boost::any源代码分析

发表于:2007-07-01来源:作者:点击数: 标签:
Boost::any分析: boost::any是一个能代表任何对象类型的对象,正如COM库的Vari ant 变量类型,以及JAVA中的Object。不同的是,Variant的做法是包含所有可能类型的一个成员实现,浪费空间,而则boost::any借助于模板,没有空间浪费。 Variant的大致实现是: C

       Boost::any分析:

 

boost::any是一个能代表任何对象类型的对象,正如COM库的Variant变量类型,以及JAVA中的Object。不同的是,Variant的做法是包含所有可能类型的一个成员实现,浪费空间,而则boost::any借助于模板,没有空间浪费。

Variant的大致实现是:

Class Cvariant

{

       int   iData;

         long lData;

       ….

       Int   type;

}

而boost::any则使用模板,依靠两个内部类来封装实际数据(PlaceFolder和Folder),并对外暴露一个叫做Type()的函数暴露实际数据的类型。

为了方便分析其代码,现展示一个简单的测试代码:

#include "stdafx.h"

#include <iostream>

#include <list>

#include "boost/any.hpp"

 

typedef std::list<boost::any> list_any;

//关键部分:可以存放任意类型的对象

void fill_list(list_any& la)

{

   //存放常数

   la.push_back(10

   //存放字符串对象,

   la.push_back( std::string("dyunze") );

   //注意la.push_back(“dyunze”)错误,因为会被当错字符串数组

}

//根据类型进行显示:

void show_list(list_any& la)

{

       list_any::iterator it;

       boost::any         anyone;

       for( it = la.begin(); it != la.end(); it++ )

       {

              anyone = *it;

              if( anyone.type() == typeid(int) )

                     std::cout<<boost::any_cast<int>(*it)<<std::endl;

              else if( anyone.type() == typeid(std::string) )

                     std::cout<<boost::any_cast<std::string>(*it).c_str()<<std::endl;

       }

}

//主程序部分:

int main(int argc, char* argv[])

{

       list_any la;

       fill_list(la);

      

       show_list(la);

       return 0;

}

以下是我整理了后的boost::any的关键代码,(只是为了说明,可能无法直接运行,如需要完整代码,请到下载boost库。)如下所示:

    class any

    {

public:

     //模板构造函数,参数可以是任意类型,真正的数据保存在content中

        template<typename ValueType>

        any(const ValueType & value): content(new holder<ValueType>(value))

        {

        } 

             //析构函数,删除保存数据的content对象

        ~any()

        {

            delete content;

        }

        //一个placeholde对象指针,只想其子类folder的一个实现

        // 即content( new holder<ValueType>(value) )语句

        placeholder * content;

    public:

      

        //查询真实数据的类型,拆葙时有用。

        const std::type_info & type() const

        {

            return content ? content->type() : typeid(void);

        }

        /**一个稻草人,存在好处是没有模板参数,可以直接申明,

   *如:       placeholder * content;

   *如果使用子类folder则这能用older<Type>

   *content而申明时Type还不确定

*/

        class placeholder

        {

        public:    

            virtual ~placeholder()

            {

            }

        public:

            virtual const std::type_info & type() const = 0;

            virtual placeholder * clone() const = 0;   

        };

        //真正保存和获取数据的类。

        template<typename ValueType>

        class holder : public placeholder

        {

        public:

            holder(const ValueType & value)

              : held(value)

            {

            }

        public:

            virtual const std::type_info & type() const

            {

                return typeid(ValueType);

            }

 

            virtual placeholder * clone() const

            {

                return new holder(held);

            }

 

        public:

                    //真正的数据,就保存在这里

            ValueType held;

        };

};

/**

 *获取content->helder数据的方法。

 *

 */

    template<typename ValueType>

    ValueType * any_cast(any * operand)

    {

                                   return operand && operand->type() == typeid(ValueType) ?              &static_cast<any::holder<ValueType> *>(operand->content)->held : 0;

}

 

以上就是boost::any源代码的关键部分,其实很短小,但是,功能上非常强大,特别是在配合容器使用时。


原文转自:http://www.ltesting.net