00001 /* 00002 * Copyright (c) 2000, 2020, Oracle and/or its affiliates. 00003 * 00004 * Licensed under the Universal Permissive License v 1.0 as shown at 00005 * http://oss.oracle.com/licenses/upl. 00006 */ 00007 #ifndef COH_ARRAY_HPP 00008 #define COH_ARRAY_HPP 00009 00010 #include "coherence/lang/compatibility.hpp" 00011 00012 #include "coherence/lang/Object.hpp" 00013 #include "coherence/lang/MemberHolder.hpp" 00014 #include "coherence/lang/SubscriptHandle.hpp" 00015 #include "coherence/lang/SubscriptHolder.hpp" 00016 00017 00018 00019 COH_OPEN_NAMESPACE2(coherence,lang) 00020 00021 extern COH_EXPORT void coh_throw_ioob(size32_t i); 00022 00023 00024 /** 00025 * A fixed size collection of values, indexed by integer position. 00026 * 00027 * Array is a wrapper around standard C++ arrays, adding thread-safety and 00028 * memory management. Array provides a "length" field which can be used to 00029 * safely determine the last index of the array. 00030 * 00031 * Array's handle type supports direct subscripting: 00032 * 00033 * @code 00034 * for (size32_t i = 0, c = hArr->length; i < c; ++i) 00035 * { 00036 * std::cout << hArr[i] << std::endl; 00037 * } 00038 * @endcode 00039 * 00040 * Only a limited number of element types are supported, including: 00041 * bool, octect_t, char, wchar_t, wchar16_t, int8_t, uint8_t, int16_t, 00042 * uint16_t, int32_t, uint32_t, int64_t, uint64_t, float32_t, and float64_t. 00043 * 00044 * Arrays of Handles and Views to Objects are supported via ObjectArray. 00045 * 00046 * @see ObjectArray 00047 * 00048 * @author mf 2007.07.05 00049 */ 00050 template<class T> 00051 class COH_EXPORT Array 00052 : public class_spec<Array<T> > 00053 { 00054 friend class factory<Array<T> >; 00055 00056 // ----- constants ------------------------------------------------------ 00057 00058 public: 00059 /** 00060 * The largest possible value of type size32_t. 00061 */ 00062 static const size32_t npos = size32_t(-1); 00063 00064 00065 // ----- handle definitions --------------------------------------------- 00066 00067 public: 00068 /** 00069 * Handle definition. 00070 */ 00071 typedef SubscriptHandle<Array<T>, T> Handle; 00072 00073 /** 00074 * View definition. 00075 */ 00076 typedef SubscriptHandle<const Array<T>, const T> View; 00077 00078 /** 00079 * Holder definition. 00080 */ 00081 typedef SubscriptHolder<Array<T>, const T> Holder; 00082 00083 00084 // ----- typedefs ------------------------------------------------------- 00085 00086 public: 00087 /** 00088 * Element type 00089 */ 00090 typedef T ElementType; 00091 00092 00093 // ----- factory methods ------------------------------------------------ 00094 00095 public: 00096 /** 00097 * Create a new Array instance. 00098 * 00099 * @param cValue the length of the returned Array 00100 */ 00101 static Handle create(size32_t cValue = 0); 00102 00103 00104 // ----- constructors --------------------------------------------------- 00105 00106 protected: 00107 /** 00108 * @internal 00109 * 00110 * Create a new Array instance. 00111 * 00112 * This constructor is not intended for general use. It assumes 00113 * partial ownership of the supplied array. That is upon destruction 00114 * of the managed Array, the destructor of each raw array element 00115 * will be executed, but the raw array itself will not be explicitly 00116 * deleted. 00117 * 00118 * @param cValue the length of the returned Array 00119 * @param aValue the raw array to be referenced by this Array 00120 */ 00121 Array(size32_t cValue, T* aValue); 00122 00123 /** 00124 * @internal 00125 * 00126 * Create a new Array which is a sub-array of an existing Array. 00127 * 00128 * The new Array uses the same storage as the original, and an update 00129 * to one will be reflected in the other. If this method is supplied 00130 * with a View to an Array, the caller must store the resulting created 00131 * Array as a View. Storing as a Handle is not const correct. It is 00132 * for this reason that this constructor is not made available via a 00133 * create method, instead it should be safely accessed via the 00134 * subArray methods. 00135 * 00136 * @param oha the Array to delegate to 00137 * @param iFrom the starting index (inclusive) 00138 * @param iTo the ending index (exclusive), or npos 00139 * 00140 * @see subArray 00141 */ 00142 Array(typename Array<T>::Holder oha, size32_t iFrom, size32_t iTo); 00143 00144 /** 00145 * @internal 00146 * 00147 * Destructor 00148 */ 00149 virtual ~Array(); 00150 00151 private: 00152 /** 00153 * @internal 00154 * 00155 * Blocked copy constructor. 00156 */ 00157 Array(const Array<T>& that); 00158 00159 00160 // ----- Array interface ------------------------------------------------ 00161 00162 public: 00163 /** 00164 * Return the element stored at the specified index. 00165 * 00166 * @param i the index into the Array 00167 * 00168 * @return the element stored at the specified index 00169 * 00170 * @throws IndexOutOfBoundsException if i is not a valid index 00171 */ 00172 inline ElementType& operator[](size32_t i) 00173 { 00174 if (i >= length) 00175 { 00176 coh_throw_ioob(i); 00177 } 00178 return raw[i]; 00179 } 00180 00181 /** 00182 * Return the element stored at the specified index. 00183 * 00184 * @param i the index into the Array 00185 * 00186 * @return the element stored at the specified index 00187 * 00188 * @throws IndexOutOfBoundsException if i is not a valid index 00189 */ 00190 inline const ElementType& operator[](size32_t i) const 00191 { 00192 if (i >= length) 00193 { 00194 coh_throw_ioob(i); 00195 } 00196 return raw[i]; 00197 } 00198 00199 /** 00200 * A region of this Array is compared to a region of a supplied Array. 00201 * 00202 * The Array regions are considered equal if each element of the 00203 * Array region is equal to the corresponding element in the other 00204 * Array region. 00205 * 00206 * In the case of Arrays of Objects this is a deep equality test, 00207 * that is each element in the supplied Array must reference an Object 00208 * which compares equal to the Object referenced from the same element 00209 * within this Array. 00210 * 00211 * @param ofSource the starting index in this Array where the 00212 * comparison begins 00213 * @param vaOther view to the Array to compare for equality 00214 * @param ofOther the starting index in the other Array where the 00215 * comparison begins 00216 * @param c the count of elements to compare, or npos for 00217 * (vaOther->length - ofOther) 00218 * 00219 * @return true iff the two Array regions are identical 00220 * 00221 * @throws IndexOutOfBoundsException if ofSource + c or ofOther + c 00222 * are out of their respective Array's bounds 00223 * 00224 * @since Coherence 3.7.1 00225 */ 00226 virtual bool regionMatches(size32_t ofSource, 00227 typename Array<T>::View vaOther, size32_t ofOther = 0, 00228 size32_t c = npos) const; 00229 00230 /** 00231 * Return an Array over a subset of the elements in this Array. The 00232 * returned Array is backed by this Array, and changes in one will 00233 * be reflected in the other. 00234 * 00235 * @param iFrom the starting index (inclusive) 00236 * @param iTo the ending index (exclusive), or npos 00237 * 00238 * @return the sub Array 00239 * 00240 * @throws IndexOutOfBoundsException if either index is out of the 00241 * Array's bounds 00242 */ 00243 virtual Handle subArray(size32_t iFrom, size32_t iTo); 00244 00245 /** 00246 * Return an Array over a subset of the elements in this Array. The 00247 * returned Array is backed by this Array, and changes in one will 00248 * be reflected in the other. 00249 * 00250 * @param iFrom the starting index (inclusive) 00251 * @param iTo the ending index (exclusive), or npos 00252 * 00253 * @return the sub Array 00254 * 00255 * @throws IndexOutOfBoundsException if either index is out of the 00256 * Array's bounds 00257 */ 00258 virtual View subArray(size32_t iFrom, size32_t iTo) const; 00259 00260 00261 // ----- Object interface ----------------------------------------------- 00262 00263 public: 00264 /** 00265 * {@inheritDoc} 00266 */ 00267 virtual TypedHandle<const String> toString() const; 00268 00269 /** 00270 * Return a hash of all the values in the Array. 00271 * 00272 * @return a hash of all the values in the Array 00273 */ 00274 virtual size32_t hashCode() const; 00275 00276 /** 00277 * Test two Arrays for equality. 00278 * 00279 * The Arrays are considered equal if they are of the same type and 00280 * length, and each element of the Array is equal to corresponding 00281 * element in the other Array. 00282 * 00283 * In the case of Arrays of Objects this is a deep equality test, 00284 * that is each element in the supplied Array must reference an Object 00285 * which compares equal to the Object referenced from the same element 00286 * within this Array. 00287 * 00288 * @param v view to the Array to compare for equality 00289 * 00290 * @return true iff the Arrays are identical 00291 */ 00292 virtual bool equals(Object::View v) const; 00293 00294 /** 00295 * Return true iff the Array and all of its elements are immutable. 00296 * 00297 * @return true iff the Array and all of its elements are immutable 00298 */ 00299 virtual bool isImmutable() const; 00300 00301 /** 00302 * Return the size in bytes of the Array. 00303 * 00304 * @param fDeep true if nested objects are to be included 00305 * 00306 * @return the size in bytes of the Array 00307 */ 00308 virtual size64_t sizeOf(bool fDeep = false) const; 00309 00310 /** 00311 * {@inheritDoc} 00312 */ 00313 virtual Object::Handle clone() const; 00314 00315 protected: 00316 /** 00317 * {@inheritDoc} 00318 */ 00319 virtual void onConst(); 00320 00321 /** 00322 * {@inheritDoc} 00323 */ 00324 virtual void onEscape(bool fEscaped) const; 00325 00326 00327 // ----- static methods ------------------------------------------------- 00328 00329 public: 00330 /** 00331 * Perform a shallow copy of the elements from one Array to another. 00332 * 00333 * If the source and destination Arrays are the same, it is 00334 * allowable to have the source and destination ranges overlap, though 00335 * an intermediate copy may be performed internally. 00336 * 00337 * @param vaSrc the source Array 00338 * @param iSrc the source offset 00339 * @param haDes the destination Array 00340 * @param iDes the destination offset 00341 * @param ce the number of elements to copy; if npos then copy all 00342 * that is available without stepping out of bounds 00343 * 00344 * @return haDes 00345 * 00346 * @throws IndexOutOfBoundsException if either index is out of the 00347 * Array's bounds, or if haDes is too small 00348 */ 00349 static typename Array<T>::Handle copy(typename Array<T>::View vaSrc, 00350 size32_t iSrc, typename Array<T>::Handle haDes, 00351 size32_t iDes = 0, size32_t ce = npos); 00352 00353 /** 00354 * Compute the hashCode for the supplied raw array. 00355 * 00356 * @param aValue the array to hash 00357 * @param c the number of elements to hash 00358 */ 00359 static size32_t hashCode(const T* aValue, size32_t c); 00360 00361 00362 // ----- data members --------------------------------------------------- 00363 00364 00365 protected: 00366 /** 00367 * The cached hash of all the Array elements. The hash code is not 00368 * cached until the Array becomes immutable. A value of zero indicates 00369 * that a hash has not been cached. 00370 */ 00371 mutable size32_t m_nHashCode; 00372 00373 /** 00374 * The super Array in the case that this Array was produced by a call 00375 * to subArray. If non-NULL then this Array does not own the memory 00376 * associated with raw. This data member is here solely for the purpose 00377 * of ensuring that the Array which "owns" the raw array is kept alive 00378 * and potentially mutable. 00379 */ 00380 MemberHolder<Array<T> > m_ohDelegate; 00381 00382 public: 00383 /** 00384 * The number of elements in the Array. 00385 */ 00386 const size32_t length; 00387 00388 /** 00389 * The underlying Array of values. 00390 * 00391 * This inner class allows the raw array to inherit the Array Object's 00392 * constness. Direct usage of this data member is discouraged as it 00393 * does not provide any bounds checking safety. 00394 */ 00395 class MemberArray 00396 { 00397 // ----- constructors --------------------------------------- 00398 00399 public: 00400 /** 00401 * Construct MemberArray for the given array of values. 00402 */ 00403 MemberArray(T* aValue) 00404 : m_aValue(aValue) {} 00405 00406 // ---- operators ------------------------------------------- 00407 00408 public: 00409 /** 00410 * Provides a conversion of this MemberArray to the type 'T*' 00411 */ 00412 inline operator T*() 00413 { 00414 return m_aValue; 00415 } 00416 00417 /** 00418 * Provides a conversion of this constant MemberArray to the 00419 * type 'const T*' 00420 */ 00421 inline operator const T*() const 00422 { 00423 return m_aValue; 00424 } 00425 00426 // ----- data members --------------------------------------- 00427 00428 protected: 00429 /** 00430 * The underlying array of values as 'T*' 00431 */ 00432 T* m_aValue; 00433 00434 // ----- friends -------------------------------------------- 00435 00436 friend class Array<T>; 00437 } raw; 00438 }; 00439 00440 COH_CLOSE_NAMESPACE2 00441 00442 #endif // COH_ARRAY_HPP