- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我一直致力于一个使用大量双指针的项目,我发现我遇到了一些错误。在花了一些时间深入研究之后,我意识到问题是当您通过指向 const 对象的指针的引用传递一个非常量对象时,它最终通过拷贝而不是引用传递。我不明白为什么会这样,因为引用只是一个别名,如果您尝试更改该范围内的对象的某些内容,const 部分应该只会导致错误。为了进一步混淆这种情况,当传递一个 const 对象的双指针或者当它只是对一个 const 对象的引用时,这不会发生。任何人都可以解释为什么会发生这种情况以及这个特殊案例与我包括的其他案例有何不同?
#include <iostream>
void PassByConstPtrRef(int const *const &num_ptr_ref)
{
std::cout << &num_ptr_ref << std::endl;
}
void PassByPtrRef(int *const &num_ptr_ref)
{
std::cout << &num_ptr_ref << std::endl;
}
void PassByPtrPtr(int const *const *const num_ptr_ref)
{
std::cout << num_ptr_ref << std::endl;
}
void PassByConstRef(int const &num)
{
std::cout << &num << std::endl;
}
int main()
{
int *num_ptr = new int{ 10 };
int *const *num_ptr_ptr = &num_ptr;
int *const &num_ptr_ref = *num_ptr_ptr;
std::cout << num_ptr_ptr << " : " << &num_ptr_ref << std::endl; // is equal
std::cout << num_ptr_ptr << " : ";
PassByConstPtrRef(num_ptr_ref); // is not equal
std::cout << num_ptr_ptr << " : ";
PassByPtrRef(num_ptr_ref); // is equal
std::cout << num_ptr_ptr << " : ";
PassByPtrPtr(&num_ptr_ref); // is equal
int foo = 4;
int &bar = foo;
std::cout << &foo << " : ";
PassByConstRef(bar); // is equal
}
输出:
0x7ffeeb6d59c8 : 0x7ffeeb6d59c8
0x7ffeeb6d59c8 : 0x7ffeeb6d59b0
0x7ffeeb6d59c8 : 0x7ffeeb6d59c8
0x7ffeeb6d59c8 : 0x7ffeeb6d59c8
0x7ffeeb6d59ac : 0x7ffeeb6d59ac
最佳答案
这是一个令人困惑的问题。该症状的直接原因是为了将 num_ptr_ref
传递给 PassByConstPtrRef()
,创建了一个临时文件。绑定(bind)到此临时参数的参数因此具有不同的地址。虽然这种现象在处理对象时是一种(某种程度上)已知的危险,但当唯一的区别是更严格的 cv 限定(“cv”是“const-volatile”的缩写)时,它出现在指针上似乎很奇怪。毕竟,数据是一样的;不同之处在于可以用数据做什么。
更深层次的原因是 C++17 标准中使用的措辞。相关部分 ([dcl.init.ref]) 使用短语“相同类型”,int const *
与 int *
不是同一类型。在一个类型的顶层允许不同的 cv 资格,但在更深的层次上没有。结果,该标准实际上需要在您的案例中创建一个临时文件。 (过去时是故意的。)
幸运的是,这种情况被认为是不受欢迎的,如 defect report 2352 中所述.提议的决议是将“相同类型”更改为“相似”,这确实说明了指向类型的 cv 限定。还有其他措辞变化。在我阅读它时,其他更改要么是切换到“相似”的结果,要么是整理措辞以减少 future 出现缺陷的可能性。也许其他变化比我意识到的更重要,但重要的一点是有了这些变化,您的情况不再需要临时。
该缺陷的解决方案已纳入 GCC 10 和 clang 10,但未纳入早期版本。 (我不知道 MSVC 的哪个版本(如果有的话)具有此增强功能。)较旧的编译器会给出您的结果(不同的地址),而较新的编译器不会。考虑这是升级的理由吗? :)
作为引用,这里是缺陷报告中给出的例子。它使用返回值而不是参数,但原理是相同的。有一个指向 int
的指针和一个指向 const int
的 (const) 指针的引用,并且引用不能直接绑定(bind)到指针。
In an example like
int *ptr;
const int *const &f() {
return ptr;
}What is returned is a reference to a temporary instead of binding directly to
ptr
.
关于c++ - 通过引用传递指向 const 对象的指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66497671/
有人能给我解释一下我下面程序中的一点吗我编写的以下程序是为了理解 int * const * var 的含义; #include "iostream" using namespace std ; in
我正在摆弄 C 中的指针,但仍然不确定一些非常基础的知识。我想出了以下示例代码: #include int main(void) { int num = 42; /
以下代码产生警告: const char * mystr = "\r\nHello"; void send_str(char * str); void main(void){ send_str
我正在尝试在我的 Linux 计算机上升级 cmake。这是我使用的命令。 删除旧的 cmake: sudo apt purge --auto-remove cmake 从 https://cmake
我试图借助 char 指针来追踪 union 第一个字节的值。我假设 byte1 的输出应该是 ff,但实际上是 ffffffff。如果我错了请纠正我。 #include #include #in
我正在使用微 Controller 进行一些 ADC 测量。当我尝试使用 -O2 优化编译以下代码时遇到问题,当代码中存在 PrintVal() 函数时,MCU 卡住。我做了一些调试,结果发现,当我添
#include #include using namespace std; int main () { vector qwerty; qwerty.push_back(5);
我有我的 woking setup.py 文件;其中包含以下行: home = os.environ["HOME"] # home = /home/dr in my machine; distu
这段代码 #include #include static_assert(std::is_same_v::value_type, volatile int>); 在最新的 GCC 和 clang
我对 C 中的前向声明有疑问。 代码 typedef struct yhash_s t_yhash;// forward declaration struct yhash_s { size_t
我想提取成员指针指向的成员的类型。 template void demo(myClass& instance, void* ptr) { instance.*member = *reinter
我正在尝试使用指针将一段 C 代码转换为 Swift 3。这是 C 代码中的相关部分。 Float32 sampleArray[256] = { // Array is 256 Float value
您可能知道,VLA's haves pros and cons 和它们在 C11 中是可选的。 我想使 VLA 成为可选项的主要原因是:“堆栈可能会爆炸”: int arr[n]; /* where
这段代码有什么错误?为什么我不能按照我尝试的方式取消引用该元素。 #include typedef struct { int value; struct node * left;
有什么方法可以在 JavaScript 中创建\返回指向变量的指针吗? 比如,在 PHP 中: function func() { ..... return &$result; } 我
如果您想使用方法的指针作为参数,则需要将该方法键入作为对象的函数,就像这样好 : type TAcceptor = function(filename:string):boolean of objec
很简单的问题: 我对 C++ 中的智能指针有点陌生。我想我得到了所有权的东西,但我不知道如何访问他们实际指向的内容。当我尝试使用对象的成员函数/变量时,我只是得到了 unique_ptr 类的函数,这
我得到了一个点的方位 Angular 、指南针方向和一个可以将箭头设置到某个方向的 api(0 是顶部,90 是右侧,180 是底部,360 是顶部) 如果我希望箭头指向我采用方位 Angular 形
我正在尝试找到一种方法,从单元格中获取位于当前工作表左侧(托盘下方)的工作表单元格中的数据。 我知道如何通过调用其他工作表 =Sheet1!A1 但现在我需要一些最好的解释 =Sheet[-1]!A1
所以我在 MATLAB 中有一个 for 循环,其中向量 x 将通过一个函数,比如 cos(x).^2,或者不同的选择,比如 sin(x).^2 + 9.*x。用户将在 for 循环之前选择他想使用的
我是一名优秀的程序员,十分优秀!