c++ - 当没有更多引用时如何从缓存中删除智能指针?

标签 c++ caching pointers boost smart-pointers

我一直在尝试使用智能指针来升级现有的应用程序,并且我正在尝试解决一个难题。在我的应用程序中,我有一个对象缓存,例如我们称它们为书籍。现在通过 ID 请求此图书缓存,如果它们在缓存中,则返回它们,如果不在,则从外部系统请求对象(操作缓慢)并将其添加到缓存中。一旦在缓存中,可以在应用程序中打开许多窗口,这些窗口中的每一个都可以引用这本书。在以前版本的应用程序中,程序员必须维护 AddRef 和 Release,当使用 Book 对象的每个窗口关闭时,最终的 Release(在缓存管理器上)将从缓存中删除该对象并删除该对象。

您可能已经发现了这里链条中的薄弱环节,当然是程序员记得调用 AddRef 和 Release。现在我已经转向智能指针(boost::intrusive),我不再需要担心调用 AddRef 和 Release。然而这会导致一个问题,缓存有一个对象的引用,所以当最后一个窗口关闭时,缓存不会被通知没有其他人持有一个引用。

我的第一个想法是定期遍历缓存并清除引用计数为 1 的对象。我不喜欢这个想法,因为它是一个 N 阶操作并且感觉不对。我想出了一个回调系统,它更好但不是很棒。我已经包含了回调系统的代码,但是我想知道是否有人有更好的方法来做到这一点?

class IContainer
{
public:
    virtual void FinalReference(BaseObject *in_obj)=0;
};

class BaseObject 
{
    unsigned int m_ref;

public:
    IContainer *m_container;

    BaseObject() : m_ref(0),m_container(0)
    {
    }

    void AddRef()
    {
        ++m_ref;
    }
    void Release()
    {
        // if we only have one reference left and we have a container
        if( 2 == m_ref && 0 != m_container )
        {
            m_container->FinalReference(this);
        }

        if( 0 == (--m_ref) )
        {
            delete this;
        }
    }
};

class Book : public BaseObject
{
    char *m_name;
public:
    Book()
    {
        m_name = new char[30];
        sprintf_s(m_name,30,"%07d",rand());
    }
    ~Book()
    {
        cout << "Deleting book : " << m_name;
        delete [] m_name;
    }

    const char *Name()
    {
        return m_name;
    }
};

class BookList : public IContainer
{
public:
    set<BookIPtr> m_books;

    void FinalReference(BaseObject *in_obj)
    {
        set<BookIPtr>::iterator it = m_books.find(BookIPtr((Book*)in_obj));
        if( it != m_books.end() )
        {
            in_obj->m_container = 0;
            m_books.erase( it );
        }
    }
};

namespace boost
{
    inline void intrusive_ptr_add_ref(BaseObject *p)
    {
        // increment reference count of object *p
        p->AddRef();
    }
    inline void intrusive_ptr_release(BaseObject *p)
    {
        // decrement reference count, and delete object when reference count reaches 0
        p->Release();
    } 
} // namespace boost

干杯 丰富

最佳答案

我从未使用过 boost::intrusive 智能指针,但如果您要使用 shared_ptr 智能指针,则可以将 weak_ptr 对象用于缓存。

当系统决定释放它们的内存时,这些 weak_ptr 指针不算作引用,但只要对象尚未被删除,就可以用于检索 shared_ptr。

关于c++ - 当没有更多引用时如何从缓存中删除智能指针?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2386492/

相关文章:

c - 这会释放我所有的**指针内存吗?

c++ - 执行 RSA 解密时密文无效

java - 在遍历简单的二维数组时提高缓存性能?

javascript - 如何确定从缓存中提供文件?

java - 不可能出现 NullPointerException

c - C 中的指针优先级和取消引用

c++ - 如何更改 libstdc++.so.6 的版本?

c++ - 理解 c++11 右值、 move 语义和性能

c++ - 赋值运算符(+= 和 =)之间的区别

.htaccess - 当用户使用鼠标或键盘进行回击时,Safari 似乎会忽略缓存设置