指针更改地址

Pointer Changes Address

本文关键字:地址 指针      更新时间:2023-10-16

我一直在创建一个与本地SQLite数据库交互的C++桌面应用程序。我正在使用存储库模式,并将从数据库中检索到的数据存储在类的实例中(用作数据模型)。我已经在我的应用程序的其他地方实现了这种模式,一切正常。但是,似乎在这个特定的实现中,类实例(数据模型)的地址似乎在构造和销毁之间变化。

下面是如何使用我在调试执行期间记录的内存地址设置我的代码的示例...

存储库类 - 构造函数:

目前,我让这个存储库类保持对Configs数据模型类的控制。这在未来可能会改变,但我目前的想法是易于构建和破坏这个物体(或者我是这么认为的!

ConfigsRepo::ConfigsRepo()
{
this->_configs = new Models::Configs(); // Memory address: 0x039089e8
}

存储库类 - getter 方法:

我省略了数据库连接和 SQL 语句字符串,因为它们有效并且不是问题的一部分。

Models::Configs& ConfigsRepo::Get()
{
int rc = sqlite3_exec("..", "..", &Sql::Convert, &this->_configs, NULL);
// Check rc == SQLITE_OK, blah blah...
return *this->_configs;  // Memory address: 0x03908901
}

sql::转换方法:

数据库表中当前只有一行和一列。数据检索正常。

int Sql::Convert(void* ret, int count, char** data, char** columns)
{
// "ret" is a reference to the "Configs" class instance passed
// by the repository.
// Memory address of "ret": 0x03918930 - most likely different
// due to passing by reference.
// Converting retrieved data to a bool and assigning to a
// property of the referenced "Configs" class.
// (if cell equals 1 then 'true', else 'false')
(*(Models::Configs*)ret).SomeProp = atoi(data[0]) == 1;
return 0;
}

存储库类 - 析构函数:

ConfigsRepo::~ConfigsRepo()
{
delete this->_configs; // Memory address: 0x03908901 - exception thrown
this->_configs = NULL;
}

如您所见,"Configs"对象的内存地址在Sql::Convert回调方法之后发生变化。当析构ConfigsRepo类(我试图在其中delete this->_configs)时,会抛出一个异常:Invalid address specified to RtlValidateHeap

我不确定为什么内存地址似乎发生了变化,并且我看不到问题。

注意:我开始相信以这种方式调用delete并不是关于设计模式的好做法,我将在以后重构我的代码。我现在只需要它工作。:)

提前感谢!

查看有关sqlite3_exe(...)的 sqlite3 文档:

sqlite3_exec() 的第 4 个参数中继到每个回调调用的第一个参数。

因此,在您的函数上:

int rc = sqlite3_exec("..", "..", &Sql::Convert, &this->_configs, NULL);

您实际上是在传递指针指针,即 this->configs 的地址,它本身就是一个指针Models::Configs

在回调站点,您可以将其地址为:

(*(Models::Configs*)ret).SomeProp = atoi(data[0]) == 1;

这是一种令人费解的说法:

(Models::Configs*)ret->SomeProp = atoi(data[0]) == 1;

您可以看到,您并没有撤消在呼叫站点上所做的工作。

所以,我建议你也改变你的sqlite3_exec()

int rc = sqlite3_exec("..", "..", &Sql::Convert, this->_configs, NULL);
^^^^^^^^^^^^^^

旁白:你不需要像&Sql::Convert一样传递Sql::Convert,就像Sql::Convert一样