C++笔记:面向对象编程(Handle类)

时间:2014-05-06 15:24:34   收藏:0   阅读:514

句柄类

句柄类的出现是为了解决用户使用指针时需要控制指针的加载和释放的问题。用指针访问对象很容易出现悬垂指针或者内存泄漏的问题。

为了解决这些问题,有许多方法可以使用,句柄类就是其中之一。句柄类是一种包装类,用于存储和管理基类的对象指针,减轻用户使用对象的负担。句柄类使用指针执行操作,虚成员由于既可以指向基类型又可以指向派生类型,所以其行为将在运行时根据句柄实际绑定的对象而变化。

句柄类的设计有两个重要的考虑因素:

指针型句柄

指针型句柄例子

class Sales_item { //Sales_item 为handle类
public:
    Sales_item():p(0), use(new size_t(1)) {}
    Sales_item(const Item_base& item): p(item.clone()), use(new size_t(1)) {}
    Sales_item(const Sales_item& item):p(item.p), use(item.use) { ++*use; }
    ~Sales_item() { decr_use(); }

    Sales_item& operator=(const Sales_item&);
    const Item_base* operator->() const;
    const Item_base& operator*() const;

private:
    void decr_use();
private:
    Item_base *p;
    size_t *use;
};

Sales_item& Sales_item::operator=(const Sales_item& rhs)
{
    ++*rhs.use;
    decr_use();
    p = rhs.p;
    use = rhs.use;
    return *this;
}

const Item_base* Sales_item::operator->() const
{
     if(p)
      return p;
     else
      throw logic_error("unbound Sales_item");
}

const Item_base& Sales_item::operator*() const
{
    if(p)
      return *p;
    else
      throw logic_error("unbound Sales_item");
}

void Sales_item::decr_use()
{
    if( --*use == 0 )  
    {
        delete p;
        delete use;
    }
}

这个句柄类中要求Item_base类中有一个虚函数clone,这个虚函数的用途是解决基类型对象或者派生类型对象的复制,这样可以不用为句柄类针对每一种派生类型对象建立构造函数。
class Item_base{
public:
    virtual Item_base* clone() const { return new Item_base(*this); }
};

以上代码即可将Item_base类的指针包装起来。通过对Handler类对象的*和->操作,即可直接访问到所包装的Item_base类型对象。

值型句柄

  • 部分时刻用户代码不能直接使用句柄定义的继承层次,用户代码必须通过句柄操作Base类型对象。像一个代理对象(Proximity)一样,句柄类提供众多的对Base类型操作的成员函数及操作符。
  • 值型句柄由于常常直接参加运算符与函数操作,而这些运算符和函数中常常会访问到句柄类中的保护部分(protected和private),所以需要在句柄类中将这些运算符与函数都加为友元,使句柄类对自定义的运算符和函数开放。
  • 句柄类中同样保持指向基类对象和计数器的两个指针。指针销毁条件和句柄构造函数都与指针型句柄相同。但不同的是,值型句柄不定义*和->两个操作符,基类对象指针完全封闭在句柄类中。
  • 推荐值型句柄通过其派生类和基类派生类来完成各种值型操作,保持面向对象的设计风格。

值型句柄例子

class Query {
    friend Query operator~ (const Query&);
    friend Query operator| (const Query&, const Query&);
    friend Query operator& (const Query&, const Query&);

private:
    Query(Query_base *query): q(query), use(new size_t(1)) {  }
public:
    Query(const string& s): q(new WordQuery(s)), use(new size_t(1)) {  }  
    Query(const Query &c): q(c.q), use(c.use) { ++*use; }
    Query& operator= (const Query&);
    ~Query(){ decr_use(); }

    set<line_no> eval(const TextQuery &t) const
    { 
        return q->eval(t);
    }

private:
    void decr_use();

private:
    Query_base *q;
    size_t *use;
};

void Query::decr_use()
{
    if ( --*use == 0 ) 
    {
        delete q;
        delete use;
    }
}

Query& Query::operator=(const Query& rhs)
{
    ++*rhs.use;
    decr_use();
    q = rhs.q;
    use = rhs.use;
    return *this;
}

值型句柄不定义*和->两个操作符,基类对象指针完全封闭在句柄类中。

参考:
http://bytes.com/topic/c/insights/651599-handle-classes-smart-pointer
http://my.oschina.net/midnite/blog/156369
http://blog.csdn.net/dhuwxs/article/details/20769877

C++笔记:面向对象编程(Handle类),布布扣,bubuko.com

评论(0
© 2014 mamicode.com 版权所有 京ICP备13008772号-2  联系我们:gaon5@hotmail.com
迷上了代码!