删除指针向量时调试断言失败 C++
Debug assertion failed C++ while deleting a pointer vector
我已经为此工作了一段时间,可能一切都变得一团糟。我是 C++ 的初学者,在调用 delete 删除数组中的元素时我 运行 遇到了问题。它说调试断言失败。在我的书和网络中,我发现他们正在做我正在做的事情,所以我不确定为什么我会失败。谁能帮忙?
另外,我不会反对一些改进建议。
#include <iostream>
#include <string>
#include <vector>
#include <fstream>
#include "Account.h"
#include "Person.h"
#include "Checkings.h"
#include "Savings.h"
#include "AccountException.h"
using namespace std;
void printAccountSummary(const vector<Account> &acc);
string TYPE;
int main()
{
cout << "Project 3 - Greg Mora - 601\n" << endl;
cout << "%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%\n" << endl;
cout << "The Avenger's Retirement Fund\n" << endl;
cout << "#\tAccount Name \t\tAddress \t\tBalance\n" << endl;
vector<Account*> acc;
acc.push_back(new Savings(new Person("Bilbo Baggins", "43 Bag End"), 1, 500, 0.075));
acc.push_back(new Checkings(new Person("Wizard Gandalf", "Crystal Palace"), 2, 1000.00, 2.00));
acc.push_back(new Savings(new Person("Elf Elrond", "Rivendell"), 3, 1200, 0.050));
ofstream dataOut("Accounts.txt");
if (dataOut.bad())
cout << "Error: File was unable to open." << endl;
for (unsigned int i = 0; i < acc.size(); i++)
{
acc[i]->writeData(dataOut);
}
for (unsigned int i = 0; i < acc.size(); i++)
{
if (acc[i] != nullptr)
{
delete acc[i];
acc[i] = nullptr;
}
}
dataOut.close();
acc.clear();
try
{
ifstream dataInput;
dataInput.open("Accounts.txt");
if (dataInput.fail())
{
cout << "Error: File was unable to open." << endl;
}
while (!dataInput.eof())
{
getline(dataInput, TYPE);
if (TYPE=="Savings")
{
Account* save = new Savings;
save->readData(dataInput);
acc.push_back(save);
}
else if (TYPE == "Checkings")
{
Account* check = new Checkings;
check->readData(dataInput);
acc.push_back(check);
}
}
}
catch (AccountException e)
{
dataOut.close();
cout << e.getMessage();
system("PAUSE");
return 0;
}
for (unsigned int i = 0; i < acc.size(); i++)
{
acc[i]->makeWithdrawl(100.00);
}
for (unsigned int i = 0; i < acc.size(); i++)
{
acc[i]->makeDeposit(25.00);
}
for (unsigned int i = 0; i < acc.size(); i++)
{
cout << acc[i]->getAccountNumber() << "\t" << acc[i]->getPersonInfo()->getName() << "\t\t" <<
acc[i]->getPersonInfo()->getAddress() << "\t\t" << acc[i]->getAccountBalance() << endl;
}
cout << endl << endl;
system("PAUSE");
return 0;
}
Person.cpp
#include "Person.h"
using namespace std;
Person::Person(){
name = "";
address = "";
}
Person::Person(string nameIn, string addressIn)
{
name = nameIn;
address = addressIn;
}
string Person::getName() const{
return name;
}
string Person::getAddress() const{
return address;
}
void Person::writeData(ofstream& output) const
{
output << name << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
output << address << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
}
void Person::readData(ifstream& input)
{
getline(input, name);
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
getline(input, address);
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
}
Account.cpp
#include "Account.h"
#include "Person.h"
#include "Checkings.h"
#include "Savings.h"
#include <iostream>
#include <string>
#include <fstream>
using namespace std;
Account::Account()
{
accountNumber = 0;
personInfo = nullptr;
accountBalance = 0;
}
Account::Account(Person *personInfoIn, int accNumIn, double accBalIn)
{
accountNumber = accNumIn;
personInfo = personInfoIn;
accountBalance = accBalIn;
}
int Account::getAccountNumber() const
{
return accountNumber;
}
double Account::getAccountBalance() const{
return accountBalance;
}
Person *Account::getPersonInfo() const{
return personInfo;
}
void Account::makeDeposit(double depositAmt){
accountBalance = accountBalance + depositAmt;
}
void Account::makeWithdrawl(double withdrawlAmt){
accountBalance = accountBalance - withdrawlAmt;
}
Account::~Account(){
if (personInfo != nullptr)
{
delete[] personInfo;
personInfo = nullptr;
accountBalance = 0;
accountNumber = 0;
}
}
Savings.cpp
#include "Savings.h"
#include "Account.h"
using namespace std;
Savings::Savings()
{
accountBalance = 0;
intRate = 0;
}
Savings::Savings(Person *personInfoIn, int accNumIn, double accBalIn, double intRateIn)
:Account(personInfoIn, accNumIn, accBalIn)
{
intRate = intRateIn;
}
double Savings::getAccountBalance() const
{
return accountBalance + (accountBalance * intRate);
}
void Savings::readData(ifstream & input)
{
personInfo = new Person;
personInfo->readData(input);
input >> accountNumber;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
input >> accountBalance;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
input.ignore();
}
void Savings::writeData(ofstream & output) const
{
output << "Savings" << endl;
personInfo->writeData(output);
output << accountNumber << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
output << getAccountBalance() << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
}
Checkings.cpp
#include "Checkings.h"
#include "Account.h"
using namespace std;
Checkings::Checkings()
{
accountBalance = 0;
monthlyFee = 0;
}
Checkings::Checkings(Person *personInfoIn, int accNumIn, double accBalIn, double monthlyFeeIn)
: Account(personInfoIn, accNumIn, accBalIn)
{
monthlyFee = monthlyFeeIn;
}
double Checkings::getAccountBalance() const
{
return accountBalance - monthlyFee;
}
void Checkings::readData(ifstream & input)
{
personInfo = new Person;
personInfo->readData(input);
input >> accountNumber;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in checking account number");
}
input >> accountBalance;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in checking account balance");
}
input.ignore();
}
void Checkings::writeData(ofstream & output) const
{
output << "Checkings" << endl;
personInfo->writeData(output);
output << accountNumber << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out checking account number");
}
output << getAccountBalance() << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out checking account balance");
}
}
您在程序的早期有一个循环,您删除了 acc
向量中的元素并将指针设置为 nullptr
,但是这并没有从向量中删除元素。然后在程序的末尾,你有三个循环遍历向量中的 all 元素(包括你销毁的元素)并取消引用指针而不检查 nullptr
.
取消引用 nullptr
会导致 undefined behavior 并且很可能导致崩溃。
想到了两种可能的解决方案:在销毁循环后清除向量,或者在其他循环中添加对 nullptr
的检查。
我看到的问题:
您正在使用 operator new
创建 Person
但您正在使用 operator delete []
删除它。改变
Account::~Account(){
if (personInfo != nullptr)
{
delete[] personInfo; // Problem
personInfo = nullptr;
accountBalance = 0;
accountNumber = 0;
}
}
到
Account::~Account(){
if (personInfo != nullptr)
{
delete personInfo;
personInfo = nullptr;
accountBalance = 0;
accountNumber = 0;
}
}
我已经为此工作了一段时间,可能一切都变得一团糟。我是 C++ 的初学者,在调用 delete 删除数组中的元素时我 运行 遇到了问题。它说调试断言失败。在我的书和网络中,我发现他们正在做我正在做的事情,所以我不确定为什么我会失败。谁能帮忙? 另外,我不会反对一些改进建议。
#include <iostream>
#include <string>
#include <vector>
#include <fstream>
#include "Account.h"
#include "Person.h"
#include "Checkings.h"
#include "Savings.h"
#include "AccountException.h"
using namespace std;
void printAccountSummary(const vector<Account> &acc);
string TYPE;
int main()
{
cout << "Project 3 - Greg Mora - 601\n" << endl;
cout << "%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%\n" << endl;
cout << "The Avenger's Retirement Fund\n" << endl;
cout << "#\tAccount Name \t\tAddress \t\tBalance\n" << endl;
vector<Account*> acc;
acc.push_back(new Savings(new Person("Bilbo Baggins", "43 Bag End"), 1, 500, 0.075));
acc.push_back(new Checkings(new Person("Wizard Gandalf", "Crystal Palace"), 2, 1000.00, 2.00));
acc.push_back(new Savings(new Person("Elf Elrond", "Rivendell"), 3, 1200, 0.050));
ofstream dataOut("Accounts.txt");
if (dataOut.bad())
cout << "Error: File was unable to open." << endl;
for (unsigned int i = 0; i < acc.size(); i++)
{
acc[i]->writeData(dataOut);
}
for (unsigned int i = 0; i < acc.size(); i++)
{
if (acc[i] != nullptr)
{
delete acc[i];
acc[i] = nullptr;
}
}
dataOut.close();
acc.clear();
try
{
ifstream dataInput;
dataInput.open("Accounts.txt");
if (dataInput.fail())
{
cout << "Error: File was unable to open." << endl;
}
while (!dataInput.eof())
{
getline(dataInput, TYPE);
if (TYPE=="Savings")
{
Account* save = new Savings;
save->readData(dataInput);
acc.push_back(save);
}
else if (TYPE == "Checkings")
{
Account* check = new Checkings;
check->readData(dataInput);
acc.push_back(check);
}
}
}
catch (AccountException e)
{
dataOut.close();
cout << e.getMessage();
system("PAUSE");
return 0;
}
for (unsigned int i = 0; i < acc.size(); i++)
{
acc[i]->makeWithdrawl(100.00);
}
for (unsigned int i = 0; i < acc.size(); i++)
{
acc[i]->makeDeposit(25.00);
}
for (unsigned int i = 0; i < acc.size(); i++)
{
cout << acc[i]->getAccountNumber() << "\t" << acc[i]->getPersonInfo()->getName() << "\t\t" <<
acc[i]->getPersonInfo()->getAddress() << "\t\t" << acc[i]->getAccountBalance() << endl;
}
cout << endl << endl;
system("PAUSE");
return 0;
}
Person.cpp
#include "Person.h"
using namespace std;
Person::Person(){
name = "";
address = "";
}
Person::Person(string nameIn, string addressIn)
{
name = nameIn;
address = addressIn;
}
string Person::getName() const{
return name;
}
string Person::getAddress() const{
return address;
}
void Person::writeData(ofstream& output) const
{
output << name << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
output << address << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
}
void Person::readData(ifstream& input)
{
getline(input, name);
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
getline(input, address);
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
}
Account.cpp
#include "Account.h"
#include "Person.h"
#include "Checkings.h"
#include "Savings.h"
#include <iostream>
#include <string>
#include <fstream>
using namespace std;
Account::Account()
{
accountNumber = 0;
personInfo = nullptr;
accountBalance = 0;
}
Account::Account(Person *personInfoIn, int accNumIn, double accBalIn)
{
accountNumber = accNumIn;
personInfo = personInfoIn;
accountBalance = accBalIn;
}
int Account::getAccountNumber() const
{
return accountNumber;
}
double Account::getAccountBalance() const{
return accountBalance;
}
Person *Account::getPersonInfo() const{
return personInfo;
}
void Account::makeDeposit(double depositAmt){
accountBalance = accountBalance + depositAmt;
}
void Account::makeWithdrawl(double withdrawlAmt){
accountBalance = accountBalance - withdrawlAmt;
}
Account::~Account(){
if (personInfo != nullptr)
{
delete[] personInfo;
personInfo = nullptr;
accountBalance = 0;
accountNumber = 0;
}
}
Savings.cpp
#include "Savings.h"
#include "Account.h"
using namespace std;
Savings::Savings()
{
accountBalance = 0;
intRate = 0;
}
Savings::Savings(Person *personInfoIn, int accNumIn, double accBalIn, double intRateIn)
:Account(personInfoIn, accNumIn, accBalIn)
{
intRate = intRateIn;
}
double Savings::getAccountBalance() const
{
return accountBalance + (accountBalance * intRate);
}
void Savings::readData(ifstream & input)
{
personInfo = new Person;
personInfo->readData(input);
input >> accountNumber;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
input >> accountBalance;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in of data");
}
input.ignore();
}
void Savings::writeData(ofstream & output) const
{
output << "Savings" << endl;
personInfo->writeData(output);
output << accountNumber << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
output << getAccountBalance() << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out of data");
}
}
Checkings.cpp
#include "Checkings.h"
#include "Account.h"
using namespace std;
Checkings::Checkings()
{
accountBalance = 0;
monthlyFee = 0;
}
Checkings::Checkings(Person *personInfoIn, int accNumIn, double accBalIn, double monthlyFeeIn)
: Account(personInfoIn, accNumIn, accBalIn)
{
monthlyFee = monthlyFeeIn;
}
double Checkings::getAccountBalance() const
{
return accountBalance - monthlyFee;
}
void Checkings::readData(ifstream & input)
{
personInfo = new Person;
personInfo->readData(input);
input >> accountNumber;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in checking account number");
}
input >> accountBalance;
if (input.bad())
{
throw AccountException("Unexpected error occurred during reading in checking account balance");
}
input.ignore();
}
void Checkings::writeData(ofstream & output) const
{
output << "Checkings" << endl;
personInfo->writeData(output);
output << accountNumber << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out checking account number");
}
output << getAccountBalance() << endl;
if (output.bad())
{
throw AccountException("Unexpected error occurred during writing out checking account balance");
}
}
您在程序的早期有一个循环,您删除了 acc
向量中的元素并将指针设置为 nullptr
,但是这并没有从向量中删除元素。然后在程序的末尾,你有三个循环遍历向量中的 all 元素(包括你销毁的元素)并取消引用指针而不检查 nullptr
.
取消引用 nullptr
会导致 undefined behavior 并且很可能导致崩溃。
想到了两种可能的解决方案:在销毁循环后清除向量,或者在其他循环中添加对 nullptr
的检查。
我看到的问题:
您正在使用 operator new
创建 Person
但您正在使用 operator delete []
删除它。改变
Account::~Account(){
if (personInfo != nullptr)
{
delete[] personInfo; // Problem
personInfo = nullptr;
accountBalance = 0;
accountNumber = 0;
}
}
到
Account::~Account(){
if (personInfo != nullptr)
{
delete personInfo;
personInfo = nullptr;
accountBalance = 0;
accountNumber = 0;
}
}