Классы ShevItem и ShevList предназначены в основном для использования, как базовые классы, хотя в простейших случаях можно обойтись без наследования, используя член info. В большинстве других случаев можно применить шаблоны, которые представлены ниже: template <class T> class ListItem : public Derived<T>, public ShevItem { public: ListItem () {} explicit ListItem ( const T & t, int i = -123456789 ) : Derived<T>(t), ShevItem(i) {} }; template <class T> class List : public ShevList { // Запрет конструктора копии и оператора присваивания: List ( List & ); void operator = ( List & ); public: List() {} const T * fir() const { return (const T *) ShevList::fir(); } const T * cur() const { return (const T *) ShevList::cur(); } const T * las() const { return (const T *) ShevList::las(); } T * fir ( Predicate1 & pre ) { return (T *) ShevList::fir(pre); } T * las ( Predicate1 & pre ) { return (T *) ShevList::las(pre); } const T * fir ( Predicate1 & pre ) const { return (const T *) ShevList::fir(pre); } const T * las ( Predicate1 & pre ) const { return (const T *) ShevList::las(pre); } T * fir () { return (T *) ShevList::fir (); } T * cur () { return (T *) ShevList::cur (); } T * las () { return (T *) ShevList::las (); } T * top () { return (T *) ShevList::top (); } T * end () { return (T *) ShevList::end (); } T * prev() { return (T *) ShevList::prev(); } T * next() { return (T *) ShevList::next(); } T * cprev() { return (T *) ShevList::cprev(); } T * cnext() { return (T *) ShevList::cnext(); } // Сделать текущим данный элемент ( небезопасная функция ): List & jump ( T * i ) { return (List&) ShevList::jump(i); } // Добавить в список один элемент List & addBefFir ( T * i ) { return (List&) ShevList::addBefFir(i); } // перед первым List & addBefCur ( T * i ) { return (List&) ShevList::addBefCur(i); } // перед текущим и сделать его текущим List & addAftCur ( T * i ) { return (List&) ShevList::addAftCur(i); } // после текущего и сделать его текущим List & addAftLas ( T * i ) { return (List&) ShevList::addAftLas(i); } // после последнего элемента // Добавить из другого списка List & addAllBefFir ( List & list ) { return (List&) ShevList::addAllBefFir(list); } // все элементы в начало List & addAllBefCur ( List & list ) { return (List&) ShevList::addAllBefCur(list); } // все элементы перед текущим List & addAllAftCur ( List & list ) { return (List&) ShevList::addAllAftCur(list); } // все элементы после текущего List & addAllAftLas ( List & list ) { return (List&) ShevList::addAllAftLas(list); } // все элементы в конец T * addNewBefFir ( List & list ) { if ( list.las() ) list.ShevList::movLasBefFir ( *this ); else ShevList::addBefFir ( new T ); return fir(); } T * addNewAftLas ( List & list ) { if ( list.las() ) list.ShevList::movLasAftLas ( *this ); else ShevList::addAftLas ( new T ); return las(); } List & invert () { return (List&) ShevList::invert (); } List & makeCurFir () { return (List&) ShevList::makeCurFir (); } List & makeCurLas () { return (List&) ShevList::makeCurLas (); } List & makeFir ( T * p ) { return (List&) ShevList::makeFir ( p ); } List & makeLas ( T * p ) { return (List&) ShevList::makeLas ( p ); } List & sort123 () { return (List&) ShevList::sort123 (); } List & sort321 () { return (List&) ShevList::sort321 (); } List & movFirBefFir ( List & list ) { return (List&) ShevList::movFirBefFir ( list ); } List & movFirBefCur ( List & list ) { return (List&) ShevList::movFirBefCur ( list ); } List & movFirAftCur ( List & list ) { return (List&) ShevList::movFirAftCur ( list ); } List & movFirAftLas ( List & list ) { return (List&) ShevList::movFirAftLas ( list ); } List & movCurBefFir ( List & list ) { return (List&) ShevList::movCurBefFir ( list ); } List & movCurBefCur ( List & list ) { return (List&) ShevList::movCurBefCur ( list ); } List & movCurAftCur ( List & list ) { return (List&) ShevList::movCurAftCur ( list ); } List & movCurAftLas ( List & list ) { return (List&) ShevList::movCurAftLas ( list ); } List & movLasBefFir ( List & list ) { return (List&) ShevList::movLasBefFir ( list ); } List & movLasBefCur ( List & list ) { return (List&) ShevList::movLasBefCur ( list ); } List & movLasAftCur ( List & list ) { return (List&) ShevList::movLasAftCur ( list ); } List & movLasAftLas ( List & list ) { return (List&) ShevList::movLasAftLas ( list ); } List & movFirBefFir ( nat n, List & list ) { return (List&) ShevList::movFirBefFir ( n, list ); } List & movLasAftLas ( nat n, List & list ) { return (List&) ShevList::movLasAftLas ( n, list ); } List & movAllBefFir ( List & list ) { return (List&) ShevList::movAllBefFir ( list ); } List & movAllBefCur ( List & list ) { return (List&) ShevList::movAllBefCur ( list ); } List & movAllAftCur ( List & list ) { return (List&) ShevList::movAllAftCur ( list ); } List & movAllAftLas ( List & list ) { return (List&) ShevList::movAllAftLas ( list ); } // Обмен элементами внутри списка: ShevList & swapCurAndFir () { return (List&) ShevList::swapCurAndFir (); } ShevList & swapCurAndLas () { return (List&) ShevList::swapCurAndLas (); } ShevList & swapFirAndLas () { return (List&) ShevList::swapFirAndLas (); } // Обмен элементами между двумя списками: List & swapCurAndCur ( List & list ) { return (List&) ShevList::swapCurAndCur ( list ); } // Обмен текущими элементами List & swapAllAndAll ( List & list ) { return (List&) ShevList::swapAllAndAll ( list ); } // Обмен всеми элементами // Для всех элементов присвоить члену info значение i: List & setAllInfo ( int i ) { return (List&) ShevList::setAllInfo(i); } // Удаление элементов из списка List & delFir () { return (List&) ShevList::delFir (); } List & delCur () { return (List&) ShevList::delCur (); } List & delLas () { return (List&) ShevList::delLas (); } template <class F> List & delSel ( F & filter ) { if ( top() ) for(;;) { if ( filter ( *cur() ) ) { if ( delCur_() ) break; continue; } if ( ! next() ) break; } return *this; } template <class T2> List & operator += ( const T2 & p ) { if ( top() ) do { *cur() += p; } while ( next() ); return *this; } template <class T2> List & operator -= ( const T2 & p ) { if ( top() ) do { *cur() -= p; } while ( next() ); return *this; } template <class T2> List & operator *= ( const T2 & p ) { if ( top() ) do { *cur() *= p; } while ( next() ); return *this; } template <class T2> List & operator /= ( const T2 & p ) { if ( top() ) do { *cur() /= p; } while ( next() ); return *this; } List & resize ( nat n ) { while ( size() > n ) ShevList::delLas(); while ( size() < n ) ShevList::addAftLas ( new T ); return *this; } List & resize ( nat n, List & stor ) { stor.addAllBefFir(*this).movFirBefFir ( n, *this ); while ( size() < n ) ShevList::addAftLas ( new T ); return *this; } }; Шаблон ListItem делает производные классы от ShevItem, а шаблон List от ShevList ( в качестве параметра для него нужно задавать класс производный от ShevItem ). Иногда нужно передать список в функцию для того, чтобы его можно было просматривать, но не изменять. Для этого предназначен класс ShowList и производные от него: class ShowList { const ShevItem * cPtr; // указатель на текущий элемент списка const ShevItem * const fPtr; // указатель на первый элемент списка const ShevItem * const lPtr; // указатель на последний элемент списка const unsigned int listSize; // количество элементов в списке public: ShowList ( const ShevList & p ) : cPtr(p.cPtr), fPtr(p.fPtr), lPtr(p.lPtr), listSize(p.listSize) {} // Количество элементов в списке: unsigned int size () const { return listSize; } // Количество элементов с заданным свойством unsigned int count ( Predicate1 & ) const; // Первый элемент с заданным свойством: const ShevItem * fir ( Predicate1 & ) const; // Последний элемент с заданным свойством: const ShevItem * las ( Predicate1 & ) const; // Текущий элемент первый? bool curIsFir () const { return cPtr == fPtr; } // Текущий элемент последний? bool curIsLas () const { return cPtr == lPtr; } // Имеется ли списке данный элемент? bool has ( const ShevItem * ) const; // Доступ к отдельному элементу списка const ShevItem * fir () const { return fPtr; } const ShevItem * cur () const { return cPtr; } const ShevItem * las () const { return lPtr; } // Сделать текущим первый элемент const ShevItem * top () { return cPtr = fPtr; } // Сделать текущим последний элемент const ShevItem * end () { return cPtr = lPtr; } // Сделать текущим предыдущий элемент const ShevItem * prev () { return cPtr != fPtr ? cPtr = cPtr->prevPtr : 0; } // Сделать текущим следующий элемент const ShevItem * next () { return cPtr != lPtr ? cPtr = cPtr->nextPtr : 0; } // Сделать текущим предыдущий элемент в цикле const ShevItem * cprev () { return cPtr = (cPtr==fPtr) ? lPtr : cPtr->prevPtr; } // Сделать текущим следующий элемент в цикле const ShevItem * cnext () { return cPtr = (cPtr==lPtr) ? fPtr : cPtr->nextPtr; } // Сделать текущим данный элемент ( небезопасная функция ): void jump ( const ShevItem * m ) { cPtr = m; } // Сделать текущим первый элемент с заданным свойством: bool findFir ( Predicate1 & ); // Сделать текущим последний элемент с заданным свойством: bool findLas ( Predicate1 & ); }; template <class T> class Show : public ShowList { public: Show ( const List<T> & p ) : ShowList ( p ) {} // Доступ к отдельному элементу списка const T * fir () const { return (const T *) ShowList::fir(); } const T * cur () const { return (const T *) ShowList::cur(); } const T * las () const { return (const T *) ShowList::las(); } // Первый элемент с заданным свойством: const T * fir ( Predicate1 & p ) const { return (const T *) ShowList::fir(); } // Последний элемент с заданным свойством: const T * las ( Predicate1 & p ) const { return (const T *) ShowList::las(); } // Сделать текущим первый элемент const T * top () { return (const T *) ShowList::top(); } // Сделать текущим последний элемент const T * end () { return (const T *) ShowList::end(); } // Сделать текущим предыдущий элемент const T * prev () { return (const T *) ShowList::prev(); } // Сделать текущим следующий элемент const T * next () { return (const T *) ShowList::next(); } // Сделать текущим предыдущий элемент в цикле const T * cprev () { return (const T *) ShowList::cprev(); } // Сделать текущим следующий элемент в цикле const T * cnext () { return (const T *) ShowList::cnext(); } // Сделать текущим данный элемент ( небезопасная функция ): void jump ( const T * m ) { ShowList::jump ( m ); } }; Интерфейс класса ShowList аналогичен интерфейсу класса ShevList, но имеет только те методы, которые не меняют список и его элементы. Исходники находятся в файлах ShevList.h и ShevList.cpp |