我正在将我的代码转换为使用 unique_ptr 而不仅仅是指针。我正在尝试为 sqlite3_statement 创建一个 unique_ptr,它会在其自定义删除器中自动调用函数 sqlite3_finalize(sqlite3_stmt *pStmt)。我怎样才能让它在我的头文件中工作?我在这里完全不知所措,欢迎任何帮助。谢谢。
在 igleyy 的大力帮助下(他的解决方案在由于某种原因调用删除器时给了我内存访问错误)我想出了这个似乎有效并且看起来很优雅的解决方案;
数据源.h
#pragma once
#include <sqlite\sqlite3.h>
#include <string>
#include <vector>
#include <memory>
class Datasource
{
void statementDeleter(sqlite3_stmt* object);
std::unique_ptr<sqlite3_stmt, std::function<void(sqlite3_stmt*)>> statement;
Datasource(Datasource const &);
Datasource &operator=(Datasource const &);
public:
Datasource(void);
~Datasource(void);
};
数据源.cpp
#include "Datasource.h"
Datasource::Datasource(void) :
statement(nullptr,std::bind(&Datasource::statementDeleter,this,std::placeholders::_1))
{
}
void Datasource::statementDeleter(sqlite3_stmt * s)
{
sqlite3_finalize(s);
}
Datasource::~Datasource(void)
{
}
最佳答案
您可以像这样编写简单的 lambda 表达式:
auto d = [](sqlite3_stmt* stmt) { sqlite3_finalize(stmt); };
std::unique_ptr<sqlite3_stmt, decltype(d)> statement(new sqlite3_stmt, d);
解释
自定义删除器使用 lambda 表达式创建,并存储在自动识别类型的变量中。 decltype
用于获取表达式类型(在您的情况下,您必须在创建 unique_ptr
时提供,以使用您的自定义删除器提供它)。
编辑
你的Datasource类代码中有两个语句变量,你需要重命名其中一个。您无法初始化 d
第一个statement
这样的变量。我认为在这种情况下不可能使用 auto,但我们可以使用 std::function
.包含并替换 auto d ...
与 std::function<void(sqlite3_stmt*)> d;
.初始化 d
在构造函数初始化列表中。
更改 statement
至 std::unique_ptr<int, decltype(d)> statement;
并在构造函数初始化列表中对其进行初始化。
Datesource::Datasource() :
d([](sqlite3_stmt* stmt) { sqlite3_finalize(stmt); }),
statement(new sqlite3_stmt) { /* constructor code */ }
关于c++ - 使用外部定义的自定义删除器在头文件中定义唯一指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19912395/