EXC_BAD_ACCESS with lambda capture

EXC_BAD_ACCESS with lambda capture

本文关键字:lambda capture with ACCESS BAD EXC      更新时间:2023-10-16

首先,如果这看起来很明显,请原谅我-我对c++有点陌生。我一直在调查这件事,但我没有发现任何特别有用的东西。

当试图访问lambda中捕获的变量时,我的应用程序崩溃了,我不知道为什么。我不认为这个对象已经被删除了,因为当把一个断点放在它崩溃的地方并使用CLion的调试器时,CLion显示这个对象存在。

一个代码示例可能会帮助我解释这一点:

//Create the progress dialog
QProgressDialog *progDialog = new QProgressDialog(tr("Opening Project…nExtracted: 0 (0.0%)nWaiting…"), nullptr, 0, 0, this);
// ... Some code here
//Declare a function to be passed as a callback
std::function<void (int minValue, int maxValue)> *progRangeChangedCallback = nullptr;
// ... More code here
//Create the lambda
//I capture progDialog (The progress dialog)
auto progRangeChangedCallbackLambda = [&progDialog](int newMin, int newMax) {
    //Putting a breakpoint here reveals that progDialog exists
    //CLion even autocompletes the below functions
    //when trying to evaluate an expresion
    progDialog->setMinimum(newMin); //EXC_BAD_ACCESS: Crashes happen here!
    progDialog->setMaximum(newMax);
};
// ... Even more code here
//Put the lambda in a std::function
progRangeChangedCallback = new std::function<void (int minValue, int maxValue)>(progRangeChangedCallbackLambda);
// ... More code
//Pass the std::function object as a callback to a new thread
//This extends QThread
OpenProjectThread *thread = new OpenProjectThread(filePath, this, progChangedCallback, progRangeChangedCallback, onSuccessCallback, onErrorCallback);
thread->start();

lambda通过引用捕获progDialog。这意味着,如果声明progDialog的作用域终止,则progDialog不再有效,使lambda具有悬空引用。显示的代码不足以得出一个权威的答案,但这似乎是最可能的答案。

在这里不能肯定地说progDialog应该按值捕获。即使它是按值捕获的,progDialog得到delete d,爆炸!这里没有提供任何信息来得出一个权威的结论。

对c++有点陌生的人会发现,花更多的时间学习基本原理会更有用,比如作用域规则、指针和引用,这些基本原理支撑着语言的高级特性,比如lambda和lambda捕获。在开始摆弄这些lambdas之前,您必须了解所有这些是如何工作的。