绑定两个参数

时间:2010-08-12 06:37:49

标签: c++ boost bind shared-ptr

考虑我有以下struct

struct IDirect3D
{
    IDirect3D() : ref_count_(0) {}
    unsigned long Release() { return --ref_count_; }
private:
    long ref_count_;
    ~IDirect3D() {}
};

我希望像以下代码(最小示例)中那样使用shared_ptr

int main()
{
    boost::shared_ptr<IDirect3D> ptr;

    IDirect3D* p = 0; // initialized somewhere
    ptr.reset( p, boost::mem_fn( &IDirect3D::Release ) );

    return 0;
}

在大多数情况下,此功能正常,但如果p等于0,则会失败。我有以下要删除的删除器:

template<typename T, typename D>
inline void SafeDeleter( T*& p, D d ) 
{ 
    if ( p != NULL ) {
        (p->*d)();
        p = NULL;
    }
}

但是下面的代码会产生很多错误(看起来它会转储整个bind.hpp):

ptr.reset( p, boost::bind( SafeDeleter, _1, &IDirect3D::Release ) );

我使用bind时出了什么问题?

1 个答案:

答案 0 :(得分:5)

Release()来自IUnknown - 所以为什么不使用它:

void my_deleter(IUnknown* p) {
    // ...
}

ptr.reset(p, &my_deleter);

请注意,Boost也有一个intrusive_ptr,在这里看起来更自然:

void intrusive_ptr_add_ref(IUnknown* p) { p->AddRef (); }
void intrusive_ptr_release(IUnknown* p) { p->Release(); }

boost::intrusive_ptr<IDirect3D> d3d(...);
IDirect3D* p = 0;
d3d.reset(p);

您的实际问题可能是存在非模板函数SafeDeleter - 要专门使用您必须使用的模板函数:

ptr.reset(p, boost::bind(&SafeDeleter<IDirect3D, ULONG (IDirect3D::*)()>, 
                         _1, &IDirect3D::Release));