我正在做一个学校项目,它需要一个复制构造函数,析构函数等。当我使用复制构造函数时,抛出一个错误,指出数组为空,我认为复制构造函数无法正常工作。
当我删除复制构造函数时,程序开始工作,这意味着该函数中可能发生了该问题。
template<class T>
class DynArray {
public:
DynArray<T>(){
ptr = new T[Capacity];
this->Capacity = 2;
this->Size = 0;
}
DynArray<T>(T n) {
ptr = new T[Capacity];
this->Capacity = n;
this->Size = 0;
}
DynArray<T>(const DynArray& orig) {
cout << "Copy" << endl;
ptr = new T[Size];
*ptr = *(orig.ptr);
}
DynArray<T>& operator=(const DynArray<T>& orig) {
if(this != &orig) {
delete[] ptr;
ptr = new T[Size];
*ptr = *(orig.ptr);
}
return *this;
}
void push_back(const T& n) {
if (Size >= Capacity) {
adjust(Capacity * 2);
}
ptr[Size] = n;
Size++;
}
void adjust(T a) {
cout << "grow" << endl;
T* arr = new T[a];
for (int i = 0; i < Capacity; ++i) {
arr[i] = ptr[i];
}
Capacity = a;
ptr = arr;
}
T& back() {
if(Size == 0) {
throw runtime_error("Array is empty");
}
return ptr[Size - 1];
}
T& front() {
if(Size == 0) {
throw runtime_error("Array is empty");
}
return ptr[0];
}
private:
T* ptr = nullptr;
int Capacity;
int Size;
主要:
#include <iostream>
#include "dynarray.h"
using namespace std;
int main( )
{
const char START = 'A';
const int MAX = 12;
// create a vector of chars
DynArray<char> vectD;
// push some values into the vector
for (int i = 0; i < MAX; i++)
{
vectD.push_back(START + i);
}
// remove the last element
vectD.pop_back();
// add another value
vectD.push_back('Z');
// test memory management
DynArray<char> vectD2 = vectD;
// display the contents
cout << "\n[";
for (int i = 0; i < vectD2.size() - 1; i++)
{
cout << vectD2.at(i) << ", ";
}
cout << "..., " << vectD2.back() << "]\n";
DynArray<char> vectD3;
vectD3 = vectD2;
cout << "\n[";
for (int i = 0; i < vectD3.size() - 1; i++)
{
cout << vectD3.at(i) << ", ";
}
cout << "..., " << vectD3.back() << "]\n";
vectD3.front() = '{';
vectD3.back() = '}';
cout << vectD3.front();
for (int i = 1; i < vectD3.size() - 2; i++)
{
cout << vectD3.at(i) << ", ";
}
cout << vectD3.at(vectD3.size()-2) << vectD3.back() << endl;
}
稍后在我的代码中,如果Size == 0,则将其设置为引发runtime_error,它的确会引发错误,表示该数组为空。复制构造函数是否正确复制? Main不能更改,它是教授给定的。
更新:我更改了复制构造函数以复制数组的所有元素,但是runtime_error仍然返回,表示数组为空。
template<class T>
class DynArray {
public:
DynArray<T>(){
ptr = new T[Capacity];
this->Capacity = 2;
this->Size = 0;
}
DynArray<T>(T n) {
ptr = new T[Capacity];
this->Capacity = n;
this->Size = 0;
}
DynArray<T>(const DynArray& orig) {
cout << "Copy" << endl;
ptr = new T[Size];
for (int i = 0; i < Size; i++) {
ptr[i] = orig.ptr[i];
}
}
DynArray<T>& operator=(const DynArray<T>& orig) {
if(this != &orig) {
delete[] ptr;
ptr = new T[Size];
for (int i = 0; i < Size; i++) {
ptr[i] = orig.ptr[i];
}
}
return *this;
}
void push_back(const T& n) {
if (Size >= Capacity) {
adjust(Capacity * 2);
}
ptr[Size] = n;
Size++;
}
void adjust(T a) {
cout << "grow" << endl;
T* arr = new T[a];
for (int i = 0; i < Capacity; ++i) {
arr[i] = ptr[i];
}
Capacity = a;
ptr = arr;
}
T& back() {
if(Size == 0) {
throw runtime_error("Array is empty");
}
return ptr[Size - 1];
}
T& front() {
if(Size == 0) {
throw runtime_error("Array is empty");
}
return ptr[0];
}
private:
T* ptr = nullptr;
int Capacity;
int Size;
答案 0 :(得分:1)
班上有很多错误。
在初始化Capacity
之前,默认构造函数使用Capacity
分配数组。
您的保留构造函数被声明为错误。它的输入参数应该是int
而不是T
。而且它遭受与默认构造函数相同的Capacity
错误。
您的副本构造函数也遭受相同的初始化错误,仅使用Size
替代。而且它根本不执行深层复制。它只是将ptr
指针从一个类实例复制到另一个类实例,而与所指向的内容无关。与您的副本分配运算符相同。
您的adjust()
方法也被声明为错误的,并且正在泄漏内存。
话虽如此,请尝试以下类似操作:
#include <algorithm>
#include <utility>
template<class T>
class DynArray
{
public:
DynArray(int n = 2)
: ptr(new T[n]), Capacity(n), Size(0)
{
}
DynArray(const DynArray& orig)
: DynArray(orig.Size)
{
std::cout << "Copy" << std::endl;
std::copy(orig.ptr, orig.ptr + orig.Size, ptr);
Size = orig.Size;
}
DynArray(DynArray&& orig)
: ptr(nullptr), Size(0), Capacity(0)
{
orig.swap(*this);
}
~DynArray()
{
delete[] ptr;
}
DynArray& operator=(const DynArray& orig)
{
if (this != &orig) {
DynArray(orig).swap(*this);
}
return *this;
}
DynArray& operator=(DynArray&& orig)
{
DynArray(std::move(orig)).swap(*this);
return *this;
}
void swap(DynArray &other)
{
std::swap(other.ptr, ptr);
std::swap(other.Capacity, Capacity);
std::swap(other.Size, Size);
}
void push_back(const T& n)
{
if (Size >= Capacity) {
grow();
}
ptr[Size] = n;
++Size;
}
void pop_back()
{
if (Size <= 0) {
throw std::runtime_error("Array is empty");
}
ptr[Size - 1] = T();
--Size;
}
T& front()
{
if (Size <= 0) {
throw std::runtime_error("Array is empty");
}
return ptr[0];
}
T& back()
{
if (Size <= 0) {
throw std::runtime_error("Array is empty");
}
return ptr[Size - 1];
}
T& at(int i)
{
if ((i < 0) || (i >= Size)) {
throw std::out_of_range("Index out of range");
}
return ptr[i];
}
T& operator[](int i)
{
return ptr[i];
}
int size() const {
return Size;
}
int capacity() const {
return Capacity;
}
private:
T* ptr = nullptr;
int Capacity = 0;
int Size = 0;
void grow()
{
std::cout << "grow" << std::endl;
DynArray newArr(Capacity * 2);
std::copy(ptr, ptr + Size, newArr.ptr);
newArr.Size = Size;
newArr.swap(*this);
}
};
然后,您应该考虑丢弃所有这些内容,而只需使用std::vector
即可,它会为您处理所有这些详细信息:
#include <vector>
template<class T>
class DynArray
{
public:
DynArray(int n = 2)
{
vec.reserve(n);
}
void push_back(const T& n)
{
vec.push_back(n);
}
void pop_back()
{
vec.pop_back();
}
T& front()
{
return vec.front();
}
T& back()
{
return vec.back();
}
T& at(int i)
{
return vec.at(i);
}
T& operator[](int i)
{
return vec[i];
}
int size() const {
return vec.size();
}
int capacity() const {
return vec.capacity();
}
private:
std::vector<T> vec;
};
答案 1 :(得分:0)
您的副本构造函数,
DynArray<T>(const DynArray& orig) {
cout << "Copy" << endl;
ptr = new T[Size];
*ptr = *(orig.ptr);
}
不执行深层复制。
诸如ptr
之类的指针只是资源(某个内存块)的索引。如果复制它,则该存储块仍然相同且唯一,您只需复制其地址即可。
为了执行深度复制,您需要分配一个新的内存块(执行此操作),然后将其复制。复制内存块意味着复制其每个元素。在这里,您仅复制第一个。因此,
DynArray<T>(const DynArray& orig) {
cout << "Copy" << endl;
ptr = new T[Size];
for (size_t i = 0; i < Size, i++)
ptr[i] = orig.ptr[i];
}
请注意,a[i]
与*(a+i)
相同,表示“事物在存储块中的第i个位置”。
更好的是,您可以使用std::copy
隐藏深度复制操作的详细信息,从而替换for循环:
std::copy(std::begin(orig.ptr), std::end(orig.ptr), std::begin(ptr));
在C ++ 11中,或者:
std::copy(orig.ptr, orig.ptr + Size, ptr);