Контейнер шаблона с несколькими параметрами шаблона, взаимодействующий с другими контейнерами шаблона с другим параметром шаблона - PullRequest
3 голосов
/ 01 августа 2011

Многословное название, да, но я не был уверен, как еще это сказать.Предположим, у меня есть контейнерный класс, который имеет два параметра шаблона, первый из которых является типом, а второй - размером локального хранилища для контейнера.

Теперь у нас есть несколько контейнеров с другим размером хранилища.По сути, функции контейнера (в любом случае все общедоступные) действительно заботятся только о T;N используется только для выделения локального хранилища (распределитель используется, если N недостаточно).

Я собрал простой пример реализации, который демонстрирует мою проблему.

#include <iostream>

template <typename T, size_t N = 10>
class TestArray
{
public:
    T Local[N];

    class Iterator
    {
    public:
        T* Array;
        int Index;      

        Iterator() : Array(NULL), Index(-1) { }
        Iterator(T* _array, int _index) : Array(_array), Index(_index) { }

        bool operator == (const Iterator& _other) const 
        { 
             return _other.Index == Index && _other.Array == Array; 
        }

        void Next() { ++Index; }
        void Prev() { --Index; }

        T& Get() { return Array[Index]; }
    };

    T& operator [] (const int _index) { return Local[_index]; }

    Iterator Begin() { return Iterator(Local, 0); }
    Iterator End() { return Iterator(Local, N); }

    template <size_t _N>
    void Copy(const TestArray<T, _N> &_other, int _index, int _count)
    {   
        int i;

        for (i = 0; i < _count; i++)
            Local[_index + i] = _other[i];
    }   
};

Это действительно вопрос из двух частей.Я коснусь этого вопроса только с первой частью, а другой со второй.Я попытался использовать его следующим образом:

int main() {

    TestArray<int> testArray1;
    TestArray<int, 25> testArray2;

    TestArray<int>::Iterator itr1;
    TestArray<int, 25>::Iterator itr2;

    itr1 = testArray1.Begin();

    for (itr1 = testArray1.Begin(); itr1 != testArray1.End(); itr1.Next())
    {
        itr1.Get() = itr1.Index;
    }

    testArray2.Copy(testArray1, 0, 10);

    for (itr2 = testArray2.Begin(); itr2 != testArray2.End(); itr2.Next())
    {
        std::cout << itr2.Get() << std::endl;
    }

    return 0;
}

Вот ссылка IDEONE: http://ideone.com/1XKwD

При компиляции с gcc-4.3.4 я получаю следующее.

prog.cpp: In member function ‘void TestArray<T, N>::Copy(const TestArray<T, _N>&, int, int) [with unsigned int _N = 10u, T = int, unsigned int N = 25u]’:
prog.cpp:82:   instantiated from here
prog.cpp:63: error: passing ‘const TestArray<int, 10u>’ as ‘this’ argument of ‘T& TestArray<T, N>::operator[](int) [with T = int, unsigned int N = 10u]’ discards qualifiers

При компиляции с VS2010 я получаю следующее.

1>------ Build started: Project: testunholytemplatemess, Configuration: Debug Win32 ------
1>  main.cpp
1>c:\users\james\documents\testproj\testunholytemplatemess\testunholytemplatemess\main.cpp(63): error C2678: binary '[' : no operator found which takes a left-hand operand of type 'const TestArray<T>' (or there is no acceptable conversion)
1>          with
1>          [
1>              T=int
1>          ]
1>          c:\users\james\documents\testproj\testunholytemplatemess\testunholytemplatemess\main.cpp(44): could be 'int &TestArray<T>::operator [](const int)'
1>          with
1>          [
1>              T=int
1>          ]
1>          while trying to match the argument list '(const TestArray<T>, int)'
1>          with
1>          [
1>              T=int
1>          ]
1>          c:\users\james\documents\testproj\testunholytemplatemess\testunholytemplatemess\main.cpp(82) : see reference to function template instantiation 'void TestArray<T,N>::Copy<10>(const TestArray<T> &,int,int)' being compiled
1>          with
1>          [
1>              T=int,
1>              N=25
1>          ]

Может быть, я толстый, но мне не удается интерпретировать то, что на самом деле пытается сказать мне что-то из этого (все еще несколькоНовое в шаблонах).Я также не понимаю, почему метод operator [] должен заботиться о N или о том, что я вызываю operator [] для контейнера с другим значением N.Если вы измените _other[i] на _other.Local[i], все будет работать нормально.

У кого-нибудь есть предложения?

1 Ответ

4 голосов
/ 01 августа 2011

Вы должны перегрузить две версии для [] -оператора: константную и неконстантную:

T & operator [] (size_t _index) { return Local[_index]; }
const T & operator [] (size_t _index) const { return Local[_index]; }

Ваша постоянная функция Copy может использовать только вторую постоянную версию!

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...