将对象指针转换为双空指针指向空指针的指针 §7.3.12 指针转换 [conv.ptr]

问题描述

我在库中有一个方法(无法更改它),它采用双空指针作为其参数之一。声明如下:

bool Queue::pop(void **packet,int &size,unsigned short &sn)

在我的应用程序代码中,我想将此函数传递给另一种对象的指针,在本例中为 GstBuffer 指针类型。如果我将指针转换为 (void**) 就像在下面的代码段中那样,我似乎可以在没有任何编译器错误的情况下执行此操作,但我怀疑这是否会导致正确的行为。像这样转换指针有效吗?

guint16 sn;
int size;
GstBuffer *buf; 
Queue *Q = ... // create a Queue instance
Q->pop((void **)&buf,size,sn);  // is this conversion valid?
size = gst_buffer_get_size(buf);

解决方法

出于所有意图和目的,void 指针可以保存任何对象(数据类型)的地址,它可以指向任何对象,并且可以被类型转换为任何对象,您的代码是有效的,我只会使用更惯用的转换:

Q->pop(reinterpret_cast<void**>(&buf),size,sn);

§7.3.12 指针转换 [conv.ptr]

  1. “指向 cv T 的指针”类型的纯右值,其中 T 是对象类型,可以转换为“指向 cv void”的指针类型的纯右值.指针值 (6.8.3) 不会因此转换而改变。

Example:

void example(void **packet){
    std::cout << *packet << "\n";                   // pointer value
    std::cout << packet << "\n";                    // pointer address
    std::cout << **reinterpret_cast<int**>(packet); // value
}

int main() 
{
    int* x = new int(20);
    std::cout << x << "\n";                         // pointer value
    std::cout << &x << "\n";                        // pointer address
    example(reinterpret_cast<void**>(&x));       
    delete x;
}

输出:

0xb83eb0
0x7ffc181ab2c8
0xb83eb0
0x7ffc181ab2c8
20

甚至只需要显式转换,因为它是指向指针的指针,否则转换将是隐式的,不需要转换。