c++ - ostream showbase 不显示零值的 "0x"

标签 c++ formatting stream manipulators

PSPS:(预先编写的后记)
我刚刚想到,一个更有先见之明的问题会包括以下概念:对于零值整数,不显示“0x”(showbase) 是一种标准行为,还是只是我的 MinGW 实现的一个怪癖?

这一切都始于一个愉快的星期天早晨……我想以一种一致、格式化的方式转储一些十六进制表示形式的句柄。
我想要一个leading 0x 和一个fixed width,但事实证明这是难以使用预期的流操纵器。
我发现这样做的唯一方法是将句柄转换为无符号长整数。 这似乎有点不合理,我希望我不是唯一想要这个的人..
我是否遗漏了标准六角操纵器中的某些东西?是因为类型 void* (HANDLE) 只是在 ostream 的正常十六进制处理之外定义的吗?

总结:我不想将 HANDLE 强制转换为它不是的东西。
而且我不想硬编码“0x”前缀。有没有办法使用标准操纵器来做到这一点?还是我需要重载 ostream 对 HANDLE 的处理? (但这可能会使我负担过重!)

这是我的测试代码(及其输出)。
我用过“。”作为填充字符,为了清楚起见,(我实际上将使用“0”)

HANDLE h; 
ULONG ul; 
int iH = sizeof(h); // how many bytes to this void* type.
int iW = iH*2;      // the max number of hex digits (width).
int iW2= iW+2;      // the max number of hex digits (+ 2 for showbase "0x").
int iX = 4;         // the number of bits per hex digit.
int ib = iH*8;      // the max number bits in HANDLE (exponent).
int i;
std::cout<<std::setfill('.'); // I actually want '0'; 
                              //   the dot is for display clarity
for( i=0; i<=ib; i+=iX )
{ ul = (pow(2,i)-1);
  h  = (HANDLE)ul;
  std::cout
  <<"//  ul "        <<std::setw(iW2)<<std::hex <<std::showbase  <<std::internal <<ul
  <<"     h "        <<std::setw(iW2) /* hex,showbase,internal have no effect */ <<h
  <<"      I want 0x"<<std::setw(iW) <<std::hex <<std::noshowbase<<std::right    <<(ULONG)h
  <<std::endl;
}

//  ul .........0     h .........0      I want 0x.......0
//  ul 0x.......f     h .......0xf      I want 0x.......f
//  ul 0x......ff     h ......0xff      I want 0x......ff
//  ul 0x.....fff     h .....0xfff      I want 0x.....fff
//  ul 0x....ffff     h ....0xffff      I want 0x....ffff
//  ul 0x...fffff     h ...0xfffff      I want 0x...fffff
//  ul 0x..ffffff     h ..0xffffff      I want 0x..ffffff
//  ul 0x.fffffff     h .0xfffffff      I want 0x.fffffff
//  ul 0xffffffff     h 0xffffffff      I want 0xffffffff

最佳答案

我在 https://bugzilla.redhat.com/show_bug.cgi?id=166735 上找到了这个- 这是直接从那里复制/粘贴的。

Doesn't sound like a bug to me. ISO C++98, 22.2.2.2.2/10 says std::showbase means prepending # printf conversion qualifier. 22.2.2.2.2/7 says std::hex means the printf conversion specifier is %x. So the behaviour is IMHO required to be the same as printf ("%#x", 0); But http://www.opengroup.org/onlinepubs/009695399/functions/fprintf.html says: "For x or X conversion specifiers, a non-zero result shall have 0x (or 0X) prefixed to it." The same is in ISO C99, 7.19.6.1(6): "For x (or X) conversion, a nonzero result has 0x (or 0X) prefixed to it."

所以这听起来像 C++98 标准(通过说“让它像 C 的 printf("%#x", 0)')需要你看到的这种愚蠢的行为。获得所需内容的唯一方法是摆脱 std::showbase 并显式输出 0x 。对不起。

关于c++ - ostream showbase 不显示零值的 "0x",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52309776/

相关文章:

c# - 我可以使用数据绑定(bind)格式化事件来有条件地检查复选框吗?

java - 关闭连接导致流关闭错误

c++ - 如何在不复制的情况下从 N 维容器中获取可迭代范围?

c++ - 为 while、if 和 for 语句获取用户定义类的对象值

date - 在 UI5 中格式化 "/Date(timestamp)/"值

haskell - 有人可以解释这个懒惰的斐波那契解决方案吗?

c# - WCF 无法将文件作为流返回

c++ - 为什么没有 std::uses_allocator<std::optional>?

c++ - C++中的指针和字节数组

python - 在 python 中格式化数字以显示