posts - 18,  comments - 104,  trackbacks - 0

相信 xxx_cast 系列都很熟了。

static_cast, dynamic_cast, const_cast, reinterpret_cast.

但是当面对boost::shared_ptr的时候呢?

reinterpret_cast 可以转换任何类型,这个在讨论范围之外。

对于下面的这个定义:

 1 class A
 2 {
 3 public:
 4     virtual ~A() {}
 5 };
 6 
 7 class B
 8     : public A
 9 {
10 public:
11     ~B() {}
12 };

如果用boost::shared_ptr包装的话:

1 typedef boost::shared_ptr<A> APtr;
2 typedef boost::shared_ptr<B> BPtr;

想想通常对指针的使用:
1 *pA = new B();
2 *pB = dynamic_cast<B*>(pA);
3 // unsafe
4 *upB = static_cast<B*>(pA);
5 
6 pA->;
7 pB->;
8 // may crash
9 upB->;

如果使用boost::shared_ptr呢。
1 APtr pA = APtr(new B());    // OK
2 BPtr pB = pA;               // compile error

从根本上讲,APtr 和 BPtr除了里面包装的原生指针有点关系以外,他们就是完全不同的两个类型,当然A和B也是完全不同的类型呀,可是想想看其实B是知道A的存在的。可是BPtr完全不知道APtr的存在。那这儿cast怎么进行呢?别说向下转型了,向上转型都成问题。

看看这段代码:

1 template <class T>
2 class shared_ptr
3 {
4     template <class F>
5     shared_ptr(const shared_ptr<F>& p)
6         : _p(p._p)
7         , _np(p._np)
8     {}
9  private:
10    T* _p;
11    reference_counter _np;
12 };

这个构造函数可以搞定自动向上转型,因为编译器可以自动检查 _p(p._p) 的合法性。那向下转型怎么办呢?看了上面这段代码,相信很容易解决想想啊转型的问题了。 只要把  _p(p._p)  改成 _p(dynamic_cast<T*>(p._p) 就可以了,当然要检查指针的合法性,我就不多写了。
当然boost::shared_ptr的作者已经想到这个问题,他给提供了解决方案:


 1 template<class T, class U> 
 2 shared_ptr<T> static_pointer_cast(shared_ptr<U> const & r)
 3 {
 4     return shared_ptr<T>(r, boost::detail::static_cast_tag());
 5 }
 6 
 7 template<class T, class U> 
 8 shared_ptr<T> dynamic_pointer_cast(shared_ptr<U> const & r)
 9 {
10     return shared_ptr<T>(r, boost::detail::dynamic_cast_tag());
11 }

需要用static_cast 转换普通指针的地方,用shared_static_cast 转换shared_ptr,
需要用dynamic_cast 转换普通指针的地方,用shared_dynamic_cast 转换shared_ptr.

前面说过,没有const的shared_ptr,但是有


1 const A* pA = new B();
2 shared_ptr<const A> cpA(pA);         //const
3 APtr spA = const_pointer_cast<A>(cpA);

总结一下:
const_cast               const_pointer_cast
static_cast              static_pointer_cast
dynamic_cast             dynamic_pointer_cast

最后一个小问题:以前,boost中的shared_ptr的cast函数的名字是:shared_xxxx_cast,
后来,为了IDE自动提供帮助,改成了xxxx_pointer_cast。由此可见,设计库还是要用户至上。

posted on 2009-04-30 21:43 尹东斐 阅读(5194) 评论(2)  编辑 收藏 引用

FeedBack:
# re: boost::shared_ptr 的 cast
2009-05-01 19:22 | 陈梓瀚(vczh)
引用计数要变的话也可以const,因为shared_ptr存的是引用计数的指针,而不只是引用计数。  回复  更多评论
  
# re: boost::shared_ptr 的 cast
2009-05-01 23:25 | 尹东斐
@陈梓瀚(vczh)

嗯,谢谢提醒,我理解有问题。  回复  更多评论
  

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理


<2025年1月>
2930311234
567891011
12131415161718
19202122232425
2627282930311
2345678

常用链接

留言簿(4)

随笔档案

文章分类

文章档案

相册

好友博客

搜索

  •  

最新评论

阅读排行榜

评论排行榜