--- parser3/src/include/pa_array.h 2001/05/17 10:22:24 1.41 +++ parser3/src/include/pa_array.h 2001/11/05 11:46:23 1.47 @@ -1,11 +1,10 @@ /** @file - Parser: array class decl. + Parser: Array & Array_iter classes decls. Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com) + Author: Alexander Petrosyan (http://paf.design.ru) - Author: Alexander Petrosyan (http://design.ru/paf) - - $Id: pa_array.h,v 1.41 2001/05/17 10:22:24 parser Exp $ + $Id: pa_array.h,v 1.47 2001/11/05 11:46:23 paf Exp $ */ #ifndef PA_ARRAY_H @@ -16,6 +15,8 @@ #include "pa_types.h" #include "pa_string.h" +class Array_iter; + /** Pooled Array. @@ -33,6 +34,7 @@ */ class Array : public Pooled { + friend class Array_iter; public: /// Array item type @@ -46,10 +48,10 @@ public: typedef void (*For_each_func)(Item *value, void *info); /// first_that iterator function type, const info - typedef bool (*First_that_func_const)(Item *value, const void *info); + typedef void *(*Item_that_func_const)(Item *value, const void *info); /// first_that iterator function type - typedef bool (*First_that_func)(Item *value, void *info); + typedef void *(*Item_that_func)(Item *value, void *info); enum { CR_INITIAL_ROWS_DEFAULT=3, ///< default preallocated row count @@ -60,22 +62,6 @@ public: Array(Pool& apool, int initial_rows=CR_INITIAL_ROWS_DEFAULT); - /** - size Array. how many items are in it. - must be used with quick_get like this: - @code - int size=src.quick_size(); - for(int i=0; i=0 && index=cache_chunk_base - */ - Item *quick_get(int index) const { - // next chunk will be with "index" row - if(!(indexcount)) { - int count=cache_chunk->count; - cache_chunk_base+=count; - cache_chunk=cache_chunk->rows[count].link; - } - - return cache_chunk->rows[index-cache_chunk_base].item; - } - Item *get(int index) const; int get_int(int index) const { return reinterpret_cast(get(index)); } @@ -116,9 +83,6 @@ public: const String *get_string(int index) const { return const_cast(static_cast(get(index))); } - const String *quick_get_string(int index) const { - return const_cast(static_cast(quick_get(index))); - } /*/// iterate over all elements, const info void for_each(For_each_func_const func, const void *info=0) const; @@ -128,10 +92,10 @@ public: void for_each(For_each_func func, void *info=0) const; /// iterate over all elements until condition, const info - void* first_that(First_that_func_const func, const void *info=0) const; + void* first_that(Item_that_func_const func, const void *info=0) const; /// iterate over all elements until condition - void* first_that(First_that_func func, void *info=0) const; + void* first_that(Item_that_func func, void *info=0) const; private: @@ -165,9 +129,6 @@ private: // array size int fused_rows; - mutable int cache_chunk_base; - mutable Chunk *cache_chunk; - private: bool chunk_is_full() { @@ -181,4 +142,46 @@ private: //disabled Array& operator = (const Array&) { return *this; } }; + +/// handy array iterator +class Array_iter { +public: + + Array_iter(const Array& aarray) : array(aarray), + chunk(aarray.head), + row(chunk->rows), + countdown(chunk->count) { + } + + /// there are still elements + bool has_next() { + return !(chunk==array.tail && row==array.append_here); + } + + /// quickly extracts next Array::Item + Array::Item *next() { + // assuming: never called after has_next()! + if(countdown==0) { // end of chunk? + chunk=row->link; + row=chunk->rows; + countdown=chunk->count; + } + Array::Item *result=row->item; + row++; countdown--; + return result; + } + + /// quickly extracts next Array::Item as const String + const String *next_string() { + return const_cast(static_cast(next())); + } + +private: + const Array& array; + const Array::Chunk *chunk; + const Array::Chunk::Row *row; + int countdown; + +}; + #endif