Making auto_cast safe

前端 未结 3 513
长情又很酷
长情又很酷 2021-02-05 19:09

GMan has posted a code for the delicious auto_cast “operator” that allows to write code such as the following in C++:

float f = 4.0f;
int i = auto_cast(f);
// in         


        
相关标签:
3条回答
  • 2021-02-05 19:41

    I wouldn't even use auto_cast because static_cast, const_cast, dynamic_cast and reinterpret_cast are also made ugly by design to help pointing code that could need refactoring : a ugly operation should have an ugly look.

    A secondary reason for introducing the new-style cast was that C-style casts are very hard to spot in a program. For example, you can't conveniently search for casts using an ordinary editor or word processor. This near-invisibility of C-style casts is especially unfortunate because they are so potentially damaging. An ugly operation should have an ugly syntactic form. That observation was part of the reason for choosing the syntax for the new-style casts. A further reason was for the new-style casts to match the template notation, so that programmers can write their own casts, especially run-time checked casts.

    http://www2.research.att.com/~bs/bs_faq2.html#static-cast

    I prefer to see clearly in the code where it could be better, or where we explicitely need to make this ugly operation.

    0 讨论(0)
  • 2021-02-05 19:49

    You could use type-traits to disable the operator if T is a base of R. As we're in C++0x, you can explicitly static_assert(std::is_base_of<T, U>::value, "Cannot auto_cast downwards!");

    0 讨论(0)
  • 2021-02-05 19:58

    It appears you want to use the T{u} form of initialization.

    template <typename U>
    operator U()
    {
        return U{std::forward<T>(mX)};
    }
    

    One of the reasons for these uniform initialization was that to use explicit constructors for creating a temporary, you need a cast aka T(u). With T{u} that problem was solved. For C++03, I imagine you could do something like this:

    template<typename T>
    struct construct_explicit {
      template<typename U>
      construct_explicit(U &u):t(u) { }
      template<typename U>
      construct_explicit(U const &u):t(u) { }
    
      T &get() { return t; }
      T const& get() const { return t; }
    
      T t;
    };
    

    Then you can say construct_explicit<U>(mX).get(), although in a case like in your conversion function, it also works to use a named variable as an intermediary step, I think

    template <typename U>
    operator U()
    {
        // or C++03: U u(mX);
        U u(std::forward<T>(mX));
        return u;
    }
    
    0 讨论(0)
提交回复
热议问题