最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

Before C++ 20, when the left operand is negative, why left shift is undefined behavior but right shift is implementation defined

programmeradmin4浏览0评论

I understand that standards before C++20 do not specifiy the representation of signed integers. Therefore, when left shifting a negative number, the result is undefined as it depends on the underlying representation. However, why when it comes to right shift, the behavior becomes implementation-defined instead of undefined? Doesn't the result still depend on the underlying representation?

I understand that standards before C++20 do not specifiy the representation of signed integers. Therefore, when left shifting a negative number, the result is undefined as it depends on the underlying representation. However, why when it comes to right shift, the behavior becomes implementation-defined instead of undefined? Doesn't the result still depend on the underlying representation?

Share Improve this question edited 2 days ago Yksisarvinen 22.4k5 gold badges34 silver badges66 bronze badges asked 2 days ago user18676624user18676624 1255 bronze badges 4
  • 1 that it is "implementation defined" does not imply that it would not depend on the underlying representation. It only means that the implementation must define and document the behavior (in accordance with the underlying representation) – 463035818_is_not_an_ai Commented 2 days ago
  • 1 I presume... compiler vendors prefer the behavior to be implementation defined because having a C++ defined behavior can thwart platform specific optimizations. C++20 did not want to incur making compiler vendors and their customers irate by making an adversely performance impacting language defined behavior. Another area that continues to be UB is signed integer wrap-around, which could be well-defined now that integers are 2s complement, but that would also thwart existing optimizations. – Eljay Commented 2 days ago
  • 1 related/dupe: stackoverflow/questions/3784996/… – NathanOliver Commented 2 days ago
  • @user18676624 I think that the problem is that it is unclear whether the result value will be negative or positive. Neither can be said apriori. – Vlad from Moscow Commented 2 days ago
Add a comment  | 

1 Answer 1

Reset to default 0

In C++ for signed integers when on a left shift occurs and a sign bit 1 was shifted out, the behaviour is undefined.

0b000xyz << 3 == 0bxyz
0b001xyz << 3 is undefined
0b1xyz   << 1 is undefined

For shift right the sign bit was repeatedly shifted in, the so called artithmetic bit shift. Except for -1 this is a division by 2 (-4 to -2 to -1, 4 to 2 to 1).

For the left shift one has for positive numbers this is a multiplication by 2 (1 to 2 to 4). But for negative numbers theoretically do -1 to -2 to -4 and it would behave (multiplication by 2), but if 0b10xyz << 3 happens there is a sign overflow and the choice 0b0xyz or 0b1xyz. In general a desirable behaviour cannot be found.

Hence a single test on the sign bit and then claiming "undefined" is a minimal checkable case.

与本文相关的文章

发布评论

评论列表(0)

  1. 暂无评论