|
|
| version 1.57.2.11, 2003/01/28 09:48:15 | version 1.78, 2009/04/30 04:40:30 |
|---|---|
| Line 1 | Line 1 |
| /** @file | /** @file |
| Parser: Array & Array_iterator classes decls. | Parser: Array & Array_iterator classes decls. |
| Copyright (c) 2001, 2003 ArtLebedev Group (http://www.artlebedev.com) | Copyright (c) 2001-2009 ArtLebedev Group (http://www.artlebedev.com) |
| Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru) | Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru) |
| */ | */ |
| #ifndef PA_ARRAY_H | #ifndef PA_ARRAY_H |
| #define PA_ARRAY_H | #define PA_ARRAY_H |
| static const char* IDENT_ARRAY_Y="$Date$"; | static const char * const IDENT_ARRAY_Y="$Date$"; |
| #include "pa_pool.h" | // includes |
| #include "pa_memory.h" | |
| #include "pa_exception.h" | #include "pa_exception.h" |
| // forwards | |
| template<typename T> class Array_iterator; | template<typename T> class Array_iterator; |
| /** | // defines |
| Simple Array. | |
| */ | #define ARRAY_OPTION_LIMIT_ALL ((size_t)-1) |
| /// Simple Array | |
| template<typename T> class Array: public PA_Object { | template<typename T> class Array: public PA_Object { |
| friend class Array_iterator<T>; | friend class Array_iterator<T>; |
| protected: | |
| /// elements[growing size] here | |
| T *felements; | |
| // allocated size | // allocated size |
| int fallocated; | size_t fallocated; |
| // array size | |
| size_t fused; | |
| public: | public: |
| struct Action_options { | |
| size_t offset; | |
| size_t limit; //< ARRAY_OPTION_LIMIT_ALL means 'all'. zero limit means 'nothing' | |
| bool reverse; | |
| bool defined; | |
| Action_options( | |
| size_t aoffset=0, | |
| size_t alimit=ARRAY_OPTION_LIMIT_ALL, | |
| bool areverse=false): | |
| offset(aoffset), limit(alimit), reverse(areverse), | |
| defined(false) {} | |
| bool adjust(size_t count) { | |
| if(!count || !limit) | |
| return false; | |
| if(offset>=count) | |
| return false; | |
| // max(limit) | |
| size_t m=reverse? | |
| offset+1 | |
| :count-offset; | |
| if(!m) | |
| return false; | |
| // fix limit | |
| if(limit==ARRAY_OPTION_LIMIT_ALL || limit>m) | |
| limit=m; | |
| return true; | |
| } | |
| }; | |
| typedef T element_type; | typedef T element_type; |
| Array(int initial=3, int delta=1): | inline Array(size_t initial=0): |
| fallocated(initial?initial:3), | fallocated(initial), |
| fdelta(delta), | |
| fused(0) | fused(0) |
| { | { |
| if(fallocated<=0 || fdelta<1) | felements=fallocated?(T *)pa_malloc(fallocated*sizeof(T)):0; |
| throw Exception(0, | |
| Exception::undefined_source, | |
| "Array::Array(%d, %d) too small", initial, delta); | |
| felements=new T[fallocated]; | |
| } | } |
| override ~Array() { | |
| T *last=felements+fused; | |
| for(T *current=felements; current<last; current++) | |
| delete current; | |
| delete felements; | #ifdef USE_DESTRUCTORS |
| inline ~Array(){ | |
| if(felements) | |
| pa_free(felements); | |
| } | } |
| #endif | |
| /// how many items are in Array | /// how many items are in Array |
| int count() const { return fused; } | inline size_t count() const { return fused; } |
| /// append to array | /// append to array |
| Array& operator += (T src) { | inline Array& operator+=(T src) { |
| if(is_full()) | if(is_full()) |
| expand(fdelta); | expand(fallocated>0?2:3); // 3 is PAF default, confirmed by tests |
| felements[fused++]=src; | felements[fused++]=src; |
| Line 62 public: | Line 103 public: |
| } | } |
| /// append other Array portion to this one. starting from offset | /// append other Array portion to this one. starting from offset |
| Array& append(const Array& src, int offset=0, int limit=0) { | Array& append(const Array& src, |
| if(!(offset>=0 && offset<src.count())) { | size_t offset=0, |
| throw Exception(0, | size_t limit=ARRAY_OPTION_LIMIT_ALL, //< negative limit means 'all'. zero limit means 'nothing' |
| Exception::undefined_source, | bool reverse=false) { |
| "Array::append(offset=%d) out of range [0..%d]", offset, src.count()-1); | |
| //return 0; // never | size_t src_count=src.count(); |
| } | // skip tivials |
| if(!src_count || !limit || offset>=src_count) | |
| return *this; | |
| // max(limit) | |
| size_t m=reverse? | |
| 1+offset | |
| :src_count-offset; | |
| if(!m) | |
| return *this; | |
| // fix limit | // fix limit |
| { | if(limit==ARRAY_OPTION_LIMIT_ALL || limit>m) |
| int m=src.count()-offset; | limit=m; |
| if(!m || limit<0) | |
| return *this; | |
| if(!limit || limit>m) | |
| limit=m; | |
| } | |
| int needed=limit-(fallocated-fused); | ssize_t delta=reverse? |
| if(needed>0) | (ssize_t)limit |
| expand(needed); | :limit-(fallocated-fused); |
| if(delta>0) | |
| memcpy(&felements[fused+=limit], &src.felements[offset], limit*sizeof(T)); | expand(delta); |
| T* from=&src.felements[offset]; | |
| T* to=&felements[fused]; | |
| if(reverse) { // reverse | |
| for(T* from_end=from-limit; from>from_end; --from) | |
| *to++=*from; | |
| } else { // forward | |
| for(T* from_end=from+limit; from<from_end; from++) | |
| *to++=*from; | |
| } | |
| fused+=limit; | |
| return *this; | return *this; |
| } | } |
| /// get index-element | /// get index-element |
| T get(int index) const { | inline T get(size_t index) const { |
| if(!(index>=0 && index<count())) { | assert(index<count()); |
| throw Exception(0, | return felements[index]; |
| Exception::undefined_source, | } |
| "Array::get(%d) out of range [0..%d]", index, count()-1); | |
| return 0; // never | |
| } | |
| /// ref version of get | |
| inline T& get_ref(size_t index) const { | |
| assert(index<count()); | |
| return felements[index]; | return felements[index]; |
| } | } |
| /// put index-element | /// put index-element |
| void put(int index, T element) { | inline void put(size_t index, T element) { |
| if(!(index>=0 && index<count())) { | assert(index<count()); |
| throw Exception(0, | |
| Exception::undefined_source, | |
| "Array::put(%d) out of range [0..%d]", index, count()-1); | |
| return; // never | |
| } | |
| felements[index]=element; | felements[index]=element; |
| } | } |
| inline T operator [](size_t index) const { return get(index); } | |
| /// iterate over all elements | /// iterate over all elements |
| template<typename I> void for_each(void (*callback)(T, I), I info) const { | template<typename I> void for_each(void (*callback)(T, I), I info) const { |
| Line 117 public: | Line 170 public: |
| callback(*current, info); | callback(*current, info); |
| } | } |
| /// iterate over all elements | |
| template<typename I> void for_each(bool (*callback)(T, I), I info) const { | |
| T *last=felements+fused; | |
| for(T *current=felements; current<last; current++) | |
| if(callback(*current, info)) | |
| return; | |
| } | |
| /// iterate over all elements | |
| template<typename I> void for_each_ref(void (*callback)(T&, I), I info) { | |
| T *last=felements+fused; | |
| for(T *current=felements; current<last; current++) | |
| callback(*current, info); | |
| } | |
| /// iterate over all elements until condition becomes true, return that element | /// iterate over all elements until condition becomes true, return that element |
| template<typename I> T first_that(bool (*callback)(T, I), I info) const { | template<typename I> T first_that(bool (*callback)(T, I), I info) const { |
| T *last=felements+fused; | T *last=felements+fused; |
| Line 124 public: | Line 192 public: |
| if(callback(*current, info)) | if(callback(*current, info)) |
| return *current; | return *current; |
| return 0;//T(0); | return T(0); |
| } | } |
| protected: | inline T* ptr(size_t index){ |
| return felements + index; | |
| // default expand delta size | } |
| int fdelta; | |
| /// elements[growing size] here | |
| T *felements; | |
| // array size | protected: |
| int fused; | |
| bool is_full() { | bool is_full() { |
| return fused == fallocated; | return fused == fallocated; |
| } | } |
| void expand(int delta) { | void expand(size_t delta) { |
| fallocated+=delta; | if(fallocated){ |
| felements = (T *)realloc(felements, fallocated*sizeof(T)); | size_t new_allocated=fallocated+delta; |
| felements=(T *)pa_realloc(felements, new_allocated*sizeof(T)); | |
| fallocated=new_allocated; | |
| } else { | |
| fallocated=delta; | |
| felements=(T *)pa_malloc(fallocated*sizeof(T)); | |
| } | |
| } | } |
| private: //disabled | private: //disabled |
| Array(const Array&) {} | |
| Array& operator = (const Array&) { return *this; } | Array& operator = (const Array&) { return *this; } |
| }; | }; |
| typedef smart_ptr<char> CharPtr; | |
| class Pool: public Array<CharPtr> { | |
| public: | |
| char *malloc(size_t size) { | |
| CharPtr result=CharPtr((char *)Array<CharPtr>::malloc(size)); | |
| *this += result; | |
| return result.get(); | |
| } | |
| char *copy(const char *buf, size_t size=0) { | |
| if(!size) | |
| size=strlen(buf)+1; | |
| char *result=malloc(size); | |
| memcpy(result, buf, size); | |
| return result; | |
| } | |
| }; | |
| void *operator new(size_t size, Pool& pool) { | |
| return pool.malloc(size); | |
| } | |
| /// handy array iterator | /** Array iterator, usage: |
| @code | |
| // Array<T> a; | |
| for(Array_iterator<T> i(a); i.has_next(); ) { | |
| T& element=i.next(); | |
| ... | |
| } | |
| @endcode | |
| */ | |
| template<typename T> class Array_iterator { | template<typename T> class Array_iterator { |
| Array<T>& farray; | const Array<T>& farray; |
| T *fcurrent; | T *fcurrent; |
| T *flast; | T *flast; |
| public: | public: |
| Array_iterator(Array<T>& aarray): farray(aarray) { | Array_iterator(const Array<T>& aarray): farray(aarray) { |
| fcurrent=farray.felements; | fcurrent=farray.felements; |
| flast=farray.felements+farray.count(); | flast=farray.felements+farray.count(); |
| } | } |
| Line 200 public: | Line 255 public: |
| } | } |
| }; | }; |
| #endif | #endif |