- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
所以我准备好把我的电脑扔出窗外,现在我觉得我应该在毁坏一台昂贵的设备之前寻求帮助。
我的程序有一个自动填写的列表(我手动输入项目),它只是输出它。然后,我有另一个代码块是使用赋值运算符再次输出它,然后是带有复制构造函数的第三个拷贝。
赋值运算符会使程序崩溃,但如果我将其注释掉以让它到达复制构造函数,列表就会变成空的。
您可以忽略任何显示“TODO”的内容,我会在稍后修复。
这里是所有 Action 发生的地方:
(这不是我的主要功能,那里没有错)
List::List() : m_pFront(0), m_pBack(0), _pData(0)
{}
List::~List()
{
Clear();
}
List::List(const char* p)
{
if(!p)
{
_pData = 0;
return;
}
if(strlen(p) == 0)
{
_pData = 0;
return;
}
_pData = new char[strlen(p) + 1];
strcpy(_pData, p);
}
void List::Clear()
{
//delete
if(!m_pFront)
{
return;
}
delete m_pFront;
Node* p = m_pBack;
//Walking the list
while(p)
{
//Get a pointer to the next in the list
Node* pTemp = p -> m_pNext;
delete p;
//Move p along the list
p = pTemp;
}
m_pFront = 0;
m_pBack = 0;
}
void List::PushFront(std::string data)
{
//create a new node
Node* p = new Node(data);
//Empty list
if(!m_pFront)
{
m_pFront = p;
m_pBack = p;
}
else //Not empty list
{
p -> m_pNext = m_pFront;
m_pFront -> m_pPrev = p;
m_pFront = p;
}
}
void List::PushBack(std::string data)
{
Node* p = new Node(data);
if(!m_pBack)
{
m_pFront = p;
m_pBack = p;
}
else
{
p -> m_pPrev = m_pBack;
m_pBack -> m_pNext = p;
m_pBack = p;
}
}
void List::PopFront()
{
if(m_pFront == 0)
{
//TODO: we need to handle this problem
return;
}
if(m_pBack == m_pFront)
{
Clear();
return;
}
Node* p = m_pFront;
m_pFront = m_pFront -> m_pNext;
p -> m_pNext = 0;
m_pFront -> m_pPrev = 0;
delete p;
}
void List::PopBack()
{
if(m_pBack == 0)
{
//TODO: we need to handle this problem
return;
}
if(m_pBack == m_pFront)
{
Clear();
return;
}
Node* p = m_pBack;
m_pBack = m_pBack -> m_pPrev;
p -> m_pPrev = 0;
m_pBack -> m_pNext = 0;
delete p;
}
ostream& List::OutPut(ostream& os)
{
if(Empty() == true)
{
os << "<empty>";
}
else
{
m_pFront -> OutputNode(os);
}
return os;
}
std::string& List::Back() const
{
if(m_pBack == 0)
{
//TODO: we need to handle this problem
}
return m_pBack -> GetData();
}
std::string& List::Front() const
{
if(m_pFront == 0)
{
//TODO: we need to handle this problem
}
return m_pFront -> GetData();
}
//Copy Constructor
List::List(const List& str)
{
if(Empty() == true)
{
_pData = 0;
return;
}
_pData = new char[strlen(str._pData) + 1];
strcpy(_pData, str._pData);
}
//Deep copy
List& List::operator=(const List& str)
{
if(&str == this)
{
return *this;
}
delete [] _pData;
_pData = new char[strlen(str._pData) + 1];
strcpy(_pData, str._pData);
return *this;
}
编辑:如果还需要查看,这是创建 List 类的地方
class List
{
public:
List();
List(const char* p);
//Copy constructor
List(const List& str);
//Deep Copy
List& operator=(const List& str);
~List();
void Clear();
//Adds to the front
void PushFront(std::string data);
//adds to the back
void PushBack(std::string data);
//removes from the front
void PopFront();
//removes from the back
void PopBack();
//gets the back value
std::string& Back() const;
//gets the from value
std::string& Front() const;
bool Empty() const {return m_pFront == 0;}
ostream& OutPut(ostream& os);
private:
Node* m_pFront;
Node* m_pBack;
char* _pData;
};
最佳答案
这就是发生的事情:
在你的复制构造函数中,你检查列表是否为空。此检查的结果未定义,因为 m_pFront
未初始化,但在调试版本中,此检查可能始终为真。无论哪种方式,由于您实际上并没有复制任何节点,而只是复制了 _pData
,因此结果列表将为空(可能设置了 _pData
除外)。
在您的赋值运算符中,在行 _pData = new char[strlen(str._pData) + 1];
之前,您无法检查 str._pData
是否实际指向任何事情。如果它没有,而您实际上是在执行 strlen(0)
,那么它就会崩溃和燃烧。
我的建议是正确实现复制构造函数。一个实际上做深拷贝,然后实现 copy and swap idiom为您的赋值运算符。
编辑:示例
下面的源代码实现了问题中列表类的一个小子集,以演示使用上一段中提到的 copy-and-swap 习惯用法的深层复制构造函数和赋值运算符。
在展示源代码之前,重要的是要认识到深度复制列表并不容易。必须考虑很多事情。您必须制作节点的拷贝。您可能想要制作数据的拷贝。但也许不是深拷贝。或者您的特定需求可能根本不需要数据拷贝。
在您的例子中,列表是双向链接的。如果 Node
有一个复制构造函数来执行简单的深度复制,那么由于复制构造函数的无限链接,您可能最终会导致堆栈溢出。
这是一个这种天真的实现的例子。
Node(const Node &other)
{
if (other.next)
next = new Node(*other.next);
if (other.prev)
prev = new Node(*other.prev);
}
在我的示例中,为了清楚起见,我选择不为 Node 实现复制构造函数。我选择复制数据,其形式为 std::string
以匹配问题。
list.h
#ifndef LIST_EXAMPLE_H_
#define LIST_EXAMPLE_H_
#include <string>
struct Node
{
std::string data;
Node *next, *prev;
Node(const std::string &d)
: next(0), prev(0), data(d)
{
}
};
class List
{
Node *front;
Node *back;
std::string data;
public:
List();
List(const std::string &);
List(const List &);
~List();
List& operator=(List);
void Clear();
void PushBack(const std::string&);
bool Empty() const { return front == 0; }
friend void swap(List &, List &);
void Print();
};
#endif // LIST_EXAMPLE_H_
list.cc
#include "list.h"
#include <iostream>
List::List()
: front(0), back(0), data()
{
}
List::List(const std::string &in)
: front(0), back(0), data(in)
{
}
List::~List()
{
Clear();
}
List::List(const List &other)
: front(0), back(0), data(other.data)
{
if (!other.Empty())
for (Node *node = other.front; node; node = node->next)
PushBack(node->data);
}
List& List::operator=(List other)
{
swap(*this, other);
return *this;
}
void List::Clear()
{
Node *node = front;
while (node) {
Node *to_delete = node;
node = node->next;
delete to_delete;
}
front = back = 0;
}
void List::PushBack(const std::string &data)
{
Node *node = new Node(data);
if (Empty()) {
front = back = node;
} else {
back->next = node;
node->prev = back;
back = node;
}
}
void List::Print()
{
std::cout << data << std::endl;
for (Node *node = front; node; node = node->next)
std::cout << node->data << std::endl;
std::cout << std::endl;
}
void swap(List &first, List &second)
{
using std::swap;
swap(first.front, second.front);
swap(first.back, second.back);
swap(first.data, second.data);
}
int main()
{
List a("foo");
a.PushBack("a");
a.PushBack("b");
a.PushBack("c");
a.Print();
List b("bar");
b.PushBack("d");
b.PushBack("e");
b.PushBack("f");
List c(b);
c.Print();
c = a;
c.Print();
a.Print();
return 0;
}
为什么赋值运算符和交换函数是这样的,在上述描述 copy and swap idiom 的答案中比我能解释得更好。 .这给我们留下了复制构造函数的实现。让我们逐行查看。
1. List::List(const List &other)
2. : front(0), back(0), data(other.data)
3. {
4. if (!other.Empty())
5. for (Node *node = other.front; node; node = node->next)
6. PushBack(node->data);
7. }
data
不是指针,我们不妨将其复制到初始化程序中。Node
没有复制构造函数,因此我们仅使用我们的 PushBack
方法。以这种方式遍历另一个列表中的节点并不是最好的。您应该更喜欢使用迭代器并调用 PushBack(*iter)
。
关于c++ - 复制列表时光荣崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18497658/
我有一段代码看起来像这样: void update_clock(uint8_t *time_array) { time_t time = *((time_t *) &time_array[0]
应用程序崩溃了 :( 请帮助我.. 在这方面失败了。我找不到错误?该应用程序可以连接到 iTunesConnect 但它会出错。 谁能根据下面的崩溃报告判断问题出在哪里? share_with_app
小二是新来的实习生,作为技术 leader,我给他安排了一个非常简单的练手任务,把前端 markdown 编辑器里上传的图片保存到服务器端,结果他真的就把图片直接保存到了服务器上,这下可把我气坏了,就
我正在创建一个函数,它将目录路径作为参数传递,或者如果它留空,则提示用户输入。 我已经设置了我的 PATH_MAX=100 和 if 语句来检查 if ((strlen(folder path) +
我已将“arial.ttf”文件(从我的/Windows/Fonts 文件夹中获取)加载到内存中,但是将其传递到 FT_New_Memory_Face 时会崩溃(在 FT_Open_Face 中的某处
我正在尝试在我的计算机上的两个控制台之间进行 rtsp 流。 在控制台 1 上,我有: ffmpeg -rtbufsize 100M -re -f dshow -s 320x240 -i video=
我正在尝试使用 scio_beast在一个项目中。我知道它还没有完成,但这并不重要。我已经设法让它工作得很好。 我现在正在尝试连接到 CloudFlare 后面的服务器,我知道我需要 SNI 才能工作
我有一个带有关联宏的下拉列表,如下所示: Sub Drop() If Range("Hidden1!A1") = "1" Then Sheets("Sheet1").Se
我对 bash 很陌生。我要做的就是运行这个nvvp -vm /usr/lib64/jvm/jre-1.8.0/bin/java无需记住最后的路径。我认为 instafix 就是这样做...... n
我在 Windows 上使用 XAMPP 已经两年左右了,它运行完美,没有崩溃没有问题。 (直到四个月前。) 大约四个月前,我们将服务器/系统升级到了更快的规范。 这是旧规范的内容 - Windows
我面临着一个非常烦人的 android 崩溃,它发生在大约 1% 的 PRODUCTION session 中,应用程序始终在后台运行。 Fatal Exception: android.app.Re
尝试使用下面的函数: public void createObjectType() { try { mCloudDB.createObjectType(ObjectTypeIn
由于我正在进行的一个项目,我在 CF11 管理员中弄乱了类路径,我设法使服务器崩溃,以至于我唯一得到的是一个漂亮的蓝屏和 500 错误.我已经检查了日志,我会把我能做的贴在帖子的底部,但我希望有人会启
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 10 个月前关闭。 Improve
我最近从 xcode 3.x 更新到 4.2,当我在 4.2 中运行应用程序时,我遇到了核心数据问题。我还更新到了 iOS 5,所以问题可能就在那里,我不太确定。 这些应用程序在 3.x 中运行良好,
我是一个相对较新的 iPhone 应用程序开发人员,所以我的知识有点粗略,所以如果这是一个微不足道的问题,请原谅我。 我有一个导航应用程序,它通过在navigationController对象上调用p
if ([MFMailComposeViewController canSendMail]) { MFMailComposeViewController *mailViewController
你能帮我吗? 我正在设置 UILocalNotification,当我尝试设置其 userInfo 字典时,它崩溃了。 fetchedObjects 包含 88 个对象。 这是代码: NSDi
为什么我的代码中突然出现 NSFastEnumeration Mutation Handler 崩溃。我很茫然为什么会突然出现这个崩溃以及如何解决它。 最佳答案 崩溃错误: **** 由于未捕获的异常
当我从表中删除行时,我的应用程序崩溃了。这是我检测到错误和堆栈跟踪的来源。谢谢! //delete row from database - (void)tableView:(UITableView *
我是一名优秀的程序员,十分优秀!