- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想知道仅将未初始化的变量传递给函数是否会导致未定义的行为?
这对我来说真的很奇怪。
假设我们有以下代码:
void open_db(db* conn)
{
// Open database connection and store it in the conn
}
int main()
{
db* conn;
open_db(conn);
}
这对我来说似乎完全合法。它不会取消引用未初始化的变量,也不会中继其状态。它只是将未初始化的指针传递给另一个函数,该函数通过operator new
在其中存储一些数据。或类似的东西。
如果是 UB,您能引用标准中所说的确切位置吗?
对于像 int
这样的其他类型也是如此吗? ?
void foo(int bar)
{
// ...
}
int main()
{
int bar;
foo(bar); // UB?
}
最佳答案
它是 UB,参数的类型并不重要。 C99的相关位是:当你声明一个具有“自动存储持续时间”的变量但不初始化它时,它的值是不确定(6.2.4p5,6.7.8p10);任何不确定值的使用都会引发未定义的行为(J.2 指 6.2.4、6.7.8 和 6.8)1。
即使它不是 UB(例如,如果 conn
已初始化),此代码也不会达到您预期的效果。正如所写,open_db
无法修改其调用者中的变量 conn
。
无论 conn
是否初始化,代码上的细微变化都是有效的,并且确实会执行您期望的操作:
void open_db(db **conn)
{
*conn = internal_open_db();
}
int main()
{
db *conn;
open_db(&conn);
}
地址运算符,一元&
,是该语言中为数不多的在以下情况不会引发未定义行为的事物之一:应用于未初始化的变量,因为它不读取变量的值。它仅确定变量的内存位置。这是一个确定的值,可以安全地传递给 open_db
(但请注意,它的类型签名已更改:它现在正在接收一个指向 db
的指针。 code>。并且 open_db
现在可以使用指针取消引用运算符一元 *
将结果写入变量。
仅在 C++ 中,这种非常常见的模式接受了一些语法糖:
void open_db(db *&conn)
{
conn = internal_open_db();
}
int main()
{
db *conn;
open_db(conn);
}
将第二个星号更改为 & 符号会使 open_db
的 conn
参数现在成为对指针的“引用”。它仍然是一个指向“底层”指针的指针,但编译器会根据需要为您填充 &
和 *
运算符。
1 对于我的语言律师同行:附件 J 是非规范性的,我找不到任何规范性声明来支持其主张,即使用不确定值总是 布。 (如果我能首先找到“使用值”的定义,这可能会有所帮助。我相信其意图是触发 6.3.2.1p2“左值转换”的任何内容,但我认为这永远不会实际上已声明。)
“不确定值”的定义是“未指定的值或陷阱表示”;使用未指定的值不会引发 UB。使用陷阱表示确实会引起 UB,但并非所有类型都有陷阱表示。 C11(但不是 C99)在 6.3.2.1p2 中有一句话非常直白地指出“如果[代码从]一个具有自动存储持续时间的对象中读取一个值,该对象可以使用寄存器存储类进行声明(从未获取过其地址),并且该对象未初始化,行为未定义”——但请注意,这里没有使用艺术术语“不确定值”,并且它将规则限制为以下变量:地址未被占用。
但是,C 编译器绝对会将读取任何未初始化的变量视为 UB,无论其类型是否有陷阱代表或其地址是否已被占用,J.2 无疑反射(reflect)了委员会的意图,许多示例也是如此在第 7 条中,“不确定”一词的出现仅仅是为了指出读取某些变量是 UB。
关于c++ - 正在将未初始化的变量传递给另一个函数 UB,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39681349/
在回答中 Is it valid to create closure (lambda) objects using `std::bit_cast` in C++20? 表明程序可能具有未定义的行为“取
C++ 20 draft [concept.default.init] 没有精确定义 default_initializable template concept default_initializa
我担心这又是一个关于解释 ISO/IEC 14882(C++ 标准)的问题,但是: 正在从程序中调用 main,例如我从 main 递归调用 main() 至少不是实现定义的行为? (更新:我的意思是
想象一下: uint64_t x = *(uint64_t *)((unsigned char[8]){'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h'}); 我有 rea
考虑以下代码: int main() { typedef struct { int first; float second; } type; type whole = { 1, 2.0
这个问题已经有答案了: (Why) is using an uninitialized variable undefined behavior? (7 个回答) 已关闭 6 年前。 SO 上各种受人尊
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 6 年前。 Improve this ques
我最近在这样的代码中遇到了一个错误 class C { public: // foo return value depends on C's state // AND each cal
我最近在升级到 gcc 7.2 后遇到了一个奇怪的 C++ 崩溃,这可以使用以下简单的完整 c++11 程序来演示: #include #include struct MyObject {
所以,我一直在阅读 the C++ standard并来到 [defns.undefined](我正在阅读的 C++17 草案中的 3.27。请注意,虽然我在这里引用 C++17,但我在 other
这不是 Rvalue reference to lvalue reference 的拷贝 由于我是针对 C++17 提出这个问题的:由于悬空引用,以下是否仍会调用未定义的行为(当然,如果使用了引用)?
这个问题在这里已经有了答案: Arithmetic right shift gives bogus result? (2 个答案) 关闭 7 年前。 我想检查一些大的计算内存需求(存储在 unsig
坦率地说,这样的代码是否有效(除了缺乏必要的错误检查,为简单起见此处省略)? 通过互联网发送数据的代码: uint16_t i = htons(500); sendto(sockfd, &i, siz
我知道 C 和 C++ 以及不同的语言,但以下内容适用于两者。 TL/DR 我知道 i = i++; 是 UB,因为 i 在表达式中被修改了两次,而 C 和 C++ 禁止它。 引用资料: C99 6.
是否有可能满足以下条件的代码在每次运行相同输入时产生不同的输出? 虽然代码是单线程的它确实链接到一个线程安全的运行时库。 没有明确调用 rand() 或 time() 或它们的 friend 。 有一
这个问题在这里已经有了答案: is there issue will stringstream.str().c_str()? [duplicate] (3 个答案) 关闭 6 年前。 考虑以下代码,
根据 C++ 标准,如果对象本身不是 const,则可以从指针中丢弃 const 并写入对象。这样: const Type* object = new Type(); const_cast( ob
下面的代码会调用 UB 吗? int main(){ volatile int i = 0; volatile int* p = &i; int j = ++i * *p; } 最佳答案
这个问题在这里已经有了答案: will casting around sockaddr_storage and sockaddr_in break strict aliasing (2 个答案) 关
假设我们有来自 C++98 的遗留代码: bool expensiveCheck(); struct Foo; bool someFunc() { Foo *ptr = 0; if(
我是一名优秀的程序员,十分优秀!