-1

我一直在为分配编写一些代码,但我遇到了嵌套模板类型的问题。

我需要以下代码来创建一个由 3 个元素数组组成的 3 个元素数组(有点像 int b[3][3]):

Array< Array<int> > b(3);

以下是我的 Array.h 的相关部分:

template <class T>
class Array{
    public:
    Array() : size(0){ data = NULL; }

    Array(int s) : size(s) { data = new T[size]; }

    Array(const Array & a) : size(a.length()) { 
       data = new T[a.length()];
       for(int i = 0; i < a.length(); ++i)
       data[i] = a[i];
    }

~Array(){ delete[] data; }  

T & operator[](int i) {
    if (i >= 0 && i < size){
        return data[i];
    } else {
        throw ArrayOutOfBounds(i);
    }
}

T operator[](int i) const{
    if (i >= 0 && i < size){
        return data[i];
    } else {
        throw ArrayOutOfBounds(i);
    }
}

Array<T> & operator=(const Array<T> &a){
    if(this == &a) return *this;
    delete[] data;
    data = new T[a.length()];
    for(int i = 0; i < a.length(); ++i)
        data[i] = a[i];
    size = a.length();
}

    int length() const { return size; }

    // Members

    private:
    int size;
    T * data;
}

更新 6/1完整的驱动程序代码:

   // Test driver for generic Array object with assignment and bounds checking

   #include "Array.h"

int main() {
    Array<int> a1(10);
for (int i = 0; i < a1.length(); ++i)
        a1[i] = i * i;
Array<int> a2 = a1;
try {
    for (int i = 0; i <= a2.length(); ++i)
        cout << a2[i] << " ";
    cout << endl;
}
catch (const ArrayOutOfBounds & e) {
    cout << endl << "ArrayOutOfBounds index=" << e.index << endl;
}
Array< Array<int> > b(3);
for (int i = 0; i < b.length(); ++i) {
    for (int j = 0; j < b[i].length(); ++j)
        b[i][j] = i*b[i].length() + j;
}
for (int i = 0; i < b.length(); ++i) {
    cout << "b[" << i << "]= ";
    for (int j = 0; j < b[i].length(); ++j)
        cout << b[i][j] << " ";
    cout << endl;
}

Array<const char *> c(3);
c[0] = "moe"; c[1] = "curly"; c[2] = "larry";
Array<const char *> d(10);
d = c;
for (int i = 0; i < d.length(); ++i)
    cout << "d[" << i << "]=" << d[i] << " ";
cout << endl;

return 0;
}     

预期输出:

0 1 4 9 16 25 36 49 64 81 
ArrayOutOfBounds index=10
b[0]= 0 1 2
b[1]= 3 4 5
b[2]= 6 7 8
d[0]=moe d[1]=curly d[2]=larry 

更新 6/2

根据纪尧姆的解决方案,这是我使用的调整大小方法:

Array<T> & resize(int newsize){
    delete[] data;
    data = new T[newsize];
    size = newsize;
    for(int i = 0; i < size; ++i)
        init(data[i], size);
}

递归调整大小适用于更高的维度,例如Array< Array< Array<int> > > q(3);

4

1 回答 1

2

根据您粘贴的代码,这不应该崩溃。你忘了析构函数了吗?您应该有一个删除内存的析构函数。当你有一个时,你需要确保数据被初始化为 nullptr(或 NULL),这样在删除一个空数组时它不会崩溃。

但是您的方法令人困惑。数组大小应该在运行时还是编译时确定?int b[3][3]在编译时确定。如果你想要这样,你应该像std::array在 C++11 中一样将大小作为模板参数,请参阅http://en.cppreference.com/w/cpp/container/array

如果要在运行时确定大小,则需要一个 resize 方法来确定第二维的大小。

编辑:根据驱动程序代码,您需要在构造函数中做一些不同的事情(如果 T 是数组,则将 int 传递给 T)。老实说,这似乎是一个错误。这种规范很难描述构造函数的作用(为除 Array 之外的所有类型 T 调用默认 ctor)

我想要这样的东西:

private:
template <typename U>
void init(U&, int) {}

template <typename U>
void init(Array<U>& a, int sz)
{
    a.resize(sz);
}

public:
Array(int s) : size(s) {
    data = new T[size];
    for (int i = 0 ; i < size; ++i) {
        init(data[i], s);
    }
}

它有效,但这很丑陋。如果你可以使用 C++11,你可以使用 std::enable_if 更好的东西

于 2013-06-01T19:29:11.733 回答