- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试自己实现 shared_ptr
。 make_shared
有问题。 std::make_shared
的主要特点是它在连续的内存块中分配计数器 block 和对象。我怎样才能做同样的事情?
我试过这样做:
template<class T>
class shared_ptr
{
private:
class _ref_cntr
{
private:
long counter;
public:
_ref_cntr() :
counter(1)
{
}
void inc()
{
++counter;
}
void dec()
{
if (counter == 0)
{
throw std::logic_error("already zero");
}
--counter;
}
long use_count() const
{
return counter;
}
};
template<class _T>
struct _object_and_block
{
_T object;
_ref_cntr cntr_block;
template<class ... Args>
_object_and_block(Args && ...args) :
object(args...)
{
}
};
T* _obj_ptr;
_ref_cntr* _ref_counter;
void _check_delete_ptr()
{
if (_obj_ptr == nullptr)
{
return;
}
_ref_counter->dec();
if (_ref_counter->use_count() == 0)
{
_delete_ptr();
}
_obj_ptr = nullptr;
_ref_counter = nullptr;
}
void _delete_ptr()
{
delete _ref_counter;
delete _obj_ptr;
}
template<class _T, class ... Args>
friend shared_ptr<_T> make_shared(Args && ... args);
public:
shared_ptr() :
_obj_ptr(nullptr),
_ref_counter(nullptr)
{
}
template<class _T>
explicit shared_ptr(_T* ptr)
{
_ref_counter = new counter_block();
_obj_ptr = ptr;
}
template<class _T>
shared_ptr(const shared_ptr<_T> & other)
{
*this = other;
}
template<class _T>
shared_ptr<T> & operator=(const shared_ptr<_T> & other)
{
_obj_ptr = other._obj_ptr;
_ref_counter = other._ref_counter;
_ref_counter->inc();
return *this;
}
~shared_ptr()
{
_check_delete_ptr();
}
};
template<class T, class ... Args>
shared_ptr<T> make_shared(Args && ... args)
{
shared_ptr<T> ptr;
auto tmp_object = new shared_ptr<T>::_object_and_block<T>(args...);
ptr._obj_ptr = &tmp_object->object;
ptr._ref_counter = &tmp_object->cntr_block;
return ptr;
}
但是当我删除对象和计数器 block 时,出现无效堆 block 异常。
最佳答案
注意_T
是 reserved name并且您不得将它用于您自己的类型/变量/参数等的名称。
问题出在这里:
void _delete_ptr()
{
delete _ref_counter;
delete _obj_ptr;
}
这对于 make_shared
是错误的情况,因为您没有分配两个单独的对象。
为 make_shared
采取的方法在 Boost 和 GCC 中 shared_ptr
是使用新的派生类型控制 block ,在基类中包含引用计数,在派生类型中为托管对象增加存储空间。如果你做 _ref_cntr
负责通过虚函数删除对象,然后派生类型可以覆盖该虚函数来做一些不同的事情(例如,只需使用显式析构函数调用来销毁对象而不释放存储)。
如果你给_ref_cntr
然后是一个虚拟析构函数 delete _ref_counter
将正确地破坏派生类型,所以它应该变成这样的:
void _delete_ptr()
{
_ref_counter->dispose();
delete _ref_counter;
}
尽管如果您不打算添加 weak_ptr
支持那么就没有必要将托管对象和控制 block 的销毁分开,你可以让控制 block 的析构函数同时做这两件事:
void _delete_ptr()
{
delete _ref_counter;
}
您当前的设计无法支持 shared_ptr
的一个重要属性, 即 template<class Y> explicit shared_ptr(Y* ptr)
构造函数必须记住 ptr
的原始类型然后调用 delete,而不是 _obj_ptr
(已转换为 T*
)。查看note在 boost::shared_ptr
的相应构造函数的文档中.使这项工作_ref_cntr
需要使用类型删除来存储原始指针,与 _obj_ptr
分开在shared_ptr
对象,所以 _ref_cntr::dispose()
可以删除正确的值。还需要设计中的更改来支持 aliasing constructor .
class _ref_cntr
{
private:
long counter;
public:
_ref_cntr() :
counter(1)
{
}
virtual ~_ref_cntr() { dispose(); }
void inc()
{
++counter;
}
void dec()
{
if (counter == 0)
{
throw std::logic_error("already zero");
}
--counter;
}
long use_count() const
{
return counter;
}
virtual void dispose() = 0;
};
template<class Y>
struct _ptr_and_block : _ref_cntr
{
Y* _ptr;
explicit _ptr_and_block(Y* p) : _ptr(p) { }
virtual void dispose() { delete _ptr; }
};
template<class Y>
struct _object_and_block : _ref_cntr
{
Y object;
template<class ... Args>
_object_and_block(Args && ...args) :
object(args...)
{
}
virtual void dispose() { /* no-op */ }
};
有了这个设计,make_shared
变成:
template<class T, class ... Args>
shared_ptr<T> make_shared(Args && ... args)
{
shared_ptr<T> ptr;
auto tmp_object = new shared_ptr<T>::_object_and_block<T>(args...);
ptr._obj_ptr = &tmp_object->object;
ptr._ref_counter = tmp_object;
return ptr;
}
所以 _ref_counter
指向分配的控制 block ,当你做 delete _ref_counter
这意味着你有一个正确匹配的 new
/delete
分配和解除分配同一对象的对,而不是使用 new
创建一个对象然后尝试 delete
两个不同的对象。
添加weak_ptr
支持您需要向控制 block 添加第二个计数,并将调用移至 dispose()
在析构函数之外,因此当第一个计数变为零时调用它(例如在 dec()
中)并且仅在第二个计数变为零时调用析构函数。然后以线程安全的方式完成所有这些会增加很多微妙的复杂性,这比这个答案需要更长的时间来解释。
此外,您的这部分实现是错误的并且会泄漏内存:
void _check_delete_ptr()
{
if (_obj_ptr == nullptr)
{
return;
}
可以构造一个 shared_ptr
使用空指针,例如shared_ptr<int>((int*)nullptr)
,在这种情况下,构造函数将分配一个控制 block ,但是因为 _obj_ptr
为空,您将永远不会删除控制 block 。
关于c++ - Make_shared - 自己的实现,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27377868/
我在 boost::asio 中做了一个 echo-server 示例。但是使用 boost::make_shard 会导致“未知异常”,而 std::make_shared 则不会。请参阅注释行。
我一直在做一些现场性能测试 1>std::shared_ptr, std::make_shared based on 'gcc 4.7.2' & 'VC10 implementation' 2>boo
问题 1> Line 1 的使用率真的比 Line 2 好吗? boost::shared_ptr shpStr = boost::make_shared(); // Line 1 boost::sh
正如 Bjarne Stroustrup 的“C++ 之旅”中所述,作为一种已知的 C++14 实践,人们应该避免在代码中使用裸露的 new 和 delete。标准库提供 std::make_shar
这可能是重复的,但我无法在任何地方找到解决方案。 我有这样的源代码: struct Blob{ //... static void *operator new(size_t si
我有一个带有模板构造函数的类,并且想要一个 shared_pointer给它。如: class A; typedef std::shared_ptr A_ptr; class A { public:
将 VS 2015 与 v120 结合使用。 所以我得到了内存异常,因为当我在一个已经构造的对象上调用 make_shared() 时。已经构造的对象有一个指向另一个用 new 分配的对象的指针,所以
在this回答 T.C.州 boost::make_shared etc. support array types - either one of unknown size, or one of fi
虽然我有std::tr1::shared_ptr在我的编译器中可用,我不有make_shared . 谁能告诉我如何正确实现 make_shared ?我懂了我需要使用可变参数来为 T 的构造函数提供
我正在尝试自己实现 shared_ptr。 make_shared 有问题。 std::make_shared 的主要特点是它在连续的内存块中分配计数器 block 和对象。我怎样才能做同样的事情?
除非定义了 B0RKEN(就像命令行上的 -DB0RKEN 一样),否则编译以下内容: #include #include #include using boost::shared_ptr; u
有什么方法可以使用 make_shared 而不是 shared_ptr 作为抽象类型? 例子: #include #include class Foo { public: virtual
关闭。这个问题需要debugging details .它目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and th
我明白了: shared_ptr x = make_shared(); 效率高于: shared_ptr x(new X()); 而且我了解其中的优势。但是,我不明白为什么编译器不能有这样的规则 "i
使用 gcc 4.6.2,如果构造函数抛出异常,make_shared() 会给出无用的回溯(显然是由于某些重新抛出)。我正在使用 make_shared() 来节省一些输入,但这是显示停止器。我创建
从 C++11 开始,由于多种原因,开发人员倾向于将智能指针类用于动态生命周期对象。对于那些新的智能指针类、标准,甚至建议不要使用像 new 这样的运算符,而是建议使用 make_shared 或 m
从理论上讲,make_shared() 和shared_ptr 之间的一个区别是内存分配技术。 make_shared() 应该使用一个 block ,而 shared_ptr 应该使用两个 bloc
为了规避restriction on partially supplied explicit template arguments ,我将要从中推导出类模板参数 ( Internal ) 的结构嵌
为什么这个 make_shared 在两个单独的调用中分配相同的内存地址? typedef struct { int a; int b; int c; } test_t; vo
我最近正在处理 shared_ptr 的问题。我很好奇如果 make_shared 失败了,它会引发异常吗?是否存在 make_shared 返回 nullptr 但没有任何异常的情况? 最佳答案 来
我是一名优秀的程序员,十分优秀!