Why is `boost::any` better than `void*`?

前端 未结 4 542
青春惊慌失措
青春惊慌失措 2021-02-01 07:50

What inherent advantages do boost::any and boost::any_cast offer over using void* and dynamic_cast?

相关标签:
4条回答
  • 2021-02-01 08:25

    The advantage is that boost::any is way more type-safe than void*.

    E.g.

    int i = 5;
    void* p = &i;
    static_cast<double*>(p);  //Compiler doesn't complain. Undefined Behavior.
    boost::any a;
    a = i;
    boost::any_cast<double>(a); //throws, which is good
    

    As to your comment, you cannot dynamic_cast from a void*. You can dynamic_cast only from pointers and references to class types which have at least one virtual function (aka polymorphic types)

    0 讨论(0)
  • 2021-02-01 08:32

    somehow nobody mentioned that boost::any<> acts as a value type, where as void* is a pointer. That means that any<> can store ANY object of ANY size. Once stored, you can pass the variable of any<> type anywhere you want; it lives on it's own.

    On the other hand void* is of pointer size, so you either have to make sure that sizeof (your data) <= sizeof(void*) or your void* is just a pointer to a real data which is stored somewhere else. But in that case, it is completely different from any<> because now you need to worry about that "somewhere else" and make sure it stays valid as long as void* is valid, which sometimes could become a challenge, especially in multithreaded applications.

    Plus as other's have mentioned any<> is very type safe, it'll store anything you want, but the only way to get it back is to know the exact type or it fails (which can be kinda annoying when one API gives you unsigned int and your code wants int are treated as different types). void* will let you do anything you want with it and if you start reading or paving over heap and/or uninitialized memory, it won't stop you or even let you know that you are doing that.

    0 讨论(0)
  • 2021-02-01 08:41

    This is what boost's reference says:

    It supports copying of any value type and safe checked extraction of that value strictly against its type.

    Neither of those can be done with a void*. There are no checks for you and you have to know yourself what you can cast it to.

    I don't think dynamic_cast enters the picture at all, as it hasn't directly anything to do with either.

    0 讨论(0)
  • 2021-02-01 08:42

    boost::any calls destructors:

    {
        boost::any x = std::string("Hello, world");
        x = std::wstring(L"Goodbye"); // string::~string called here
    } // wstring::~wstring called here
    
    0 讨论(0)
提交回复
热议问题