- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
这是 C++17 形式的规则 ([basic.lval]/8),但它在其他标准中看起来很相似(在 C++98 中是“lvalue”而不是“glvalue”):
8 If a program attempts to access the stored value of an object through a glvalue of other than one of the following types the behavior is undefined:
(8.4) — a type that is the signed or unsigned type corresponding to the dynamic type of the object
这条规则听起来像是“除非你做了 X,否则你会得到 UB”,但这并不意味着如果你做了 X,你就不会像人们预期的那样得到 UB!实际上,执行 X 是有条件或无条件 UB,具体取决于标准的版本。
让我们看下面的代码:
int i = -1;
unsigned j = reinterpret_cast<unsigned&>(i);
这段代码的行为是什么?
[expr.reinterpret.cast]/10(C++11 中的/11)(重点是我的):
An lvalue expression of type T1 can be cast to the type “reference to T2” if an expression of type “pointer to T1” can be explicitly converted to the type “pointer to T2” using a reinterpret_cast. That is, a reference cast reinterpret_cast(x) has the same effect as the conversion *reinterpret_cast(&x) with the built-in & and * operators. The result is an lvalue that refers to the same object as the source lvalue, but with a different type.
所以 reinterpret_cast<unsigned&>(i)
左值指的是 int
对象 i
, 但带有 usigned
类型。初始化需要初始化表达式的值,这正式意味着将左值到右值转换应用于左值。
[conv.lval]/1:
An lvalue of a non-function, non-array type T can be converted to an rvalue. If T is an incomplete type, a program that necessitates this conversion is ill-formed. If the object to which the lvalue refers is not an object of type T and is not an object of a type derived from T, or if the object is uninitialized, a program that necessitates this conversion has undefined behavior.
我们的左值 unsigned
类型未引用 unsigned
的对象type 这意味着行为是未定义的。
在这些标准中,情况有点复杂,但规则略有放宽。 [expr.reinterpret.cast]/11 告诉同样的:
The result refers to the same object as the source glvalue, but with the specified type.
有关 UB 的冒犯性措辞已从 [conv.lval]/1 中删除:
A glvalue of a non-function, non-array type T can be converted to a prvalue. If T is an incomplete type, a program that necessitates this conversion is ill-formed. If T is a non-class type, the type of the prvalue is the cv-unqualified version of T. Otherwise, the type of the prvalue is T.
但是 L-to-R 转换读取的是哪个值? [conv.lval]/(2.6)(C++17 中的/(3.4))回答了这个问题:
… the value contained in the object indicated by the glvalue is the prvalue result
unsigned
左值 reinterpret_cast<unsigned&>(i)
表示 i
int
值为 -1
的对象L-to-R 转换产生的纯右值有 unsigned
类型。 [expr]/4 说:
If during the evaluation of an expression, the result is not mathematically defined or not in the range of representable values for its type, the behavior is undefined.
-1
绝对不在 unsigned
的可表示值范围内纯右值表达式的类型,因此行为未定义。
如果 i
将定义行为对象包含 [0, INT_MAX] 范围内的值。
同样的推理也适用于 unsigned
的情况。通过 int
访问对象增值。这是 C++98 和 C++11 中的 UB 以及 C++14 和 C++17 中的 UB,除非对象的值在 [0, INT_MAX] 范围内。
因此,与普遍认为此别名规则允许将对象重新解释为包含相应有符号/无符号类型的值的看法相反,它不允许这样做。对于[0, INT_MAX]范围内的值,有符号和无符号类型的对象具有相同的表示("有符号整数类型的非负值范围是对应无符号整数类型的子范围,表示两种类型中相同值的值是相同的”,[basic.fundamental]/3 in C++17)。很难将这种访问称为“重新解释”,更不用说这是 C++14 之前的无条件 UB。
那么规则 ([basic.lval]/(8.4)) 的目的是什么?
最佳答案
这是 defect report 2214 的主题,它说:
Section: 6.9.1 [basic.fundamental] Status: C++17 Submitter: Richard Smith Date: 2015-12-15
[Adopted at the February/March, 2017 meeting.]
According to 6.9.1 [basic.fundamental] paragraph 3,
The range of non-negative values of a signed integer type is a subrange of the corresponding unsigned integer type, and the value representation of each corresponding signed/unsigned type shall be the same. (This is the wording in C++11 and C++14 versions, though the paragraph numbers may be different -- n.m.)
C11对应的写法是,
The range of nonnegative values of a signed integer type is a subrange of the corresponding unsigned integer type, and the representation of the same value in each type is the same.
C 的措辞可以说更清晰,但它失去了 C++ 措辞的暗示,即有符号类型的符号位是相应无符号类型的值表示的一部分。
提议的决议(2017 年 1 月):
将 6.9.1 [basic.fundamental] 第 3 段更改如下:
...The standard and extended unsigned integer types are collectively called unsigned integer types. The range of non-negative values of a signed integer type is a subrange of the corresponding unsigned integer type, the representation of the same value in each of the two types is the same, and the value representation of each corresponding signed/unsigned type shall be the same. The standard signed integer types...
所以这显然是一直以来的意图。 C++17 刚刚修正了措辞。
C 和 C++ 标准从未打算允许将负值重新解释为无符号,反之亦然。在野外有几种带符号的整数表示形式(例如一个的补码、二进制的补码、符号和大小)并且标准不强制其中任何一个,因此它不能规定这种重新解释的效果。它们可以已实现定义,但考虑到陷阱表示的可能性,这并没有真正的好处。 “实现定义的结果或陷阱”与“未定义”一样好。
关于c++ - 签名/未签名别名规则是否按预期工作?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54085710/
我得到了这个printHashKey函数,它运行良好。 fun printHashKey() { try { val info : PackageInfo = packageM
如何使用正确的签名 key 为我的 Android 应用包签名? 最佳答案 我尝试在此处和 this question 中使用多个答案, 但不知何故我收到了这个错误,因为我的 android/app/
我的 gradle 文件中有这个: android { signingConfigs { mySigningConfig { keyAlias 'the
请至少选择一个签名版本以在 Android Studio 2.3 中使用 现在在 Android Studio 中生成一个签名的 APK 时,它显示了两个选项(复选框),即 1. V1(Jar 签名)
我想表示一些标量值(例如整数或字符串)通过它的实际值或一些 NA 值,然后存储它们在集合中(例如列表)。目的是处理缺失值。 为此,我实现了一个签名 module type Scalar = sig
为什么这不完全有效? sum :: (Num a, Num b) => a -> b -> c sum a b = a + b 当然,错误消息与签名有关,但我仍然不明白原因。 Couldn't mat
谢谢帮助,我的问题是关于从下面的代码中收到的 ax 值? mov al,22h mov cl,0fdh imul cl 真机结果:ff9a 我的预期:00:9a(通过二进制相乘) 第一个数字是 22h
我有一个注释: import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.
我从对物体的思考中得出了一个术语。当我们扩展一个类时,扩展类将具有与父类相同的签名,因此术语 IS-A 来自...示例 class Foo{ } class Foo2 extends Foo{ } c
我需要在有符号整数和它们作为字节序列的内部表示之间进行转换。在 C 中,我使用的函数如下: unsigned char hibyte(unsigned short i) {return i>>8;}
我正在尝试使用给定的 RSA 参数对一些数据进行签名。 我给出了模数、指数、D、DP、DQ、P、Q 和 InverseQ。什么库或方法最容易使用来计算此签名。在 C# 中,一旦您提供参数,它们就会有一
这些签名之间有什么区别? T * f(T & identifier); T & f(T & identifier); T f(T & identifier); void f(T * identifie
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: Where and why do I have to put the “template” and “typ
我有一个签名,我需要在签名旁边添加图片。但我不确定 css 的确切程度和内容。目前它显示在文字下方,我应该把图片放在哪里?在相同的 tr 或 td 中?
查看 LinkedHashMap 的 JDK 源代码,我注意到这个类被声明为: public class LinkedHashMap extends HashMap im
背景:我继承了一个基于 linux 的嵌入式系统,其中包含一个 SMTP 代理和一些我不得不忍受的古怪限制。它位于 SMTP 客户端和服务器之间。当 SMTP 客户端连接时,代理会打开与服务器的连接,
这是 C++17 形式的规则 ([basic.lval]/8),但它在其他标准中看起来很相似(在 C++98 中是“lvalue”而不是“glvalue”): 8 If a program attem
我有一个注释: import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.
我即将开展一个项目,希望使用电子签名板使用 C# 捕获客户的签名、在设备上显示文本等。 现在,在我开始做进一步的研究之前,我想向你们征求一些意见/建议,我应该使用哪些设备.. 我现在的要求非常笼统:我
呢喃自己在心中开始扩张地盘,仿佛制式地广播了三次。 漾起的涟绮,用谈不上精腻的手段。 拒绝天亮,却又贪恋着贪恋多情的日光。 川流不息的画面是他们,而我的落幕停在右脚,它渴望着下台,而我只剩自言
我是一名优秀的程序员,十分优秀!