Has there been a proposal to add std::bin to the c++ standard?

前端 未结 1 937
悲&欢浪女
悲&欢浪女 2021-02-01 19:17

C++14 adds ability to use binary literals by typing 0b prefix for the value:

int v = 0b1111; // 15 in decimal

But there is no

1条回答
  •  慢半拍i
    慢半拍i (楼主)
    2021-02-01 19:57

    As far as I know there was no proposal submitted to add a formatting flag to add binary formatting and/or a manipulator std::bin. You can check the proposals at http://www.open-std.org/jtc1/sc22/wg21/docs/papers/. I'm pretty sure the proposal to add binary literals did not add this facility (a quick search revealed N3472 but I'm not sure if this is the latest version of the paper).

    From a technical point of view it may not be entirely easy to add! The various flags are normally all stored in just one word in the stream class and there are various reasons to use up all the bits. The existing three settings (std::ios_base::oct, std::ios_base::dec, std::ios_base::hex) can be nicely stored in just 2 bits. Of course, the three values would leave one value open except that this value is typically taken for the default setting, i.e., not fixing the base when reading. As a result it may be necessary to change the layout of the stream classes or to make processing less efficient (e.g., by somehow using an iword() to store the additional possibility of binary formatting). I haven't done the analysis whether there is an actual problem with any of the implementations (I know there is none for my implementation but I did use all the bits in a word if I recall correctly).

    If you want to support binary formatting it is relatively easy to add via a custom std::num_put facet. Below is a simple example. It doesn't deal with some of the formatting options which may be desirable like padding or digit separators:

    #include 
    #include 
    #include 
    
    class binary_num_put
        : public std::num_put {
        template 
        iter_type common_put(iter_type out, std::ios_base& str, char_type fill,
                             T original, unsigned long long v) const {
            if (str.flags() & std::ios_base::basefield) {
                return this->std::num_put::do_put(out, str, fill, original);
            }
            if (str.flags() & std::ios_base::showbase) {
                *out++ = '0';
                *out++ = str.flags() & std::ios_base::uppercase? 'B': 'b';
            }
            unsigned long long mask(1ull << (std::numeric_limits::digits - 1));
            while (mask && !(mask & v)) {
                mask >>= 1;
            }
            if (mask) {
                for (; mask; mask >>= 1) {
                    *out++ = v & mask? '1': '0';
                }
            }
            else {
                *out++ = '0';
            }
            return out;
        }
        iter_type do_put(iter_type out, std::ios_base& str, char_type fill, long v) const {
            return common_put(out, str, fill, v, static_cast(v));
        }
        iter_type do_put(iter_type out, std::ios_base& str, char_type fill, long long v) const {
            return common_put(out, str, fill, v, static_cast(v));
        }
        iter_type do_put(iter_type out, std::ios_base& str, char_type fill, unsigned long v) const {
            return common_put(out, str, fill, v, v);
        }
        iter_type do_put(iter_type out, std::ios_base& str, char_type fill, unsigned long long v) const {
            return common_put(out, str, fill, v, v);
        }
    };
    
    std::ostream& bin(std::ostream& out) {
        auto const& facet = std::use_facet>(out.getloc());
        if (!dynamic_cast(&facet)) {
            std::locale loc(std::locale(), new binary_num_put);
            out.imbue(loc);
        }
        out.setf(std::ios_base::fmtflags(), std::ios_base::basefield);
        return out;
    }
    
    int main()
    {
        std::cout << std::showbase << bin << 12345 << " "
                  << std::dec << 12345 << "\n";
    }
    

    0 讨论(0)
提交回复
热议问题