动态转换为 void 指针有实际用途吗?
在 C++ 中,T q =dynamic_cast
构造将指针 p
执行运行时强制转换为其他指针类型 T
> 必须出现在 *p
动态类型的继承层次结构中才能成功。一切都很好。
然而,也可以执行dynamic_cast
,它将简单地返回一个指向“最派生对象”的指针(参见 C++ 中的 5.2.7::7 11)。我知道这个功能可能在动态转换的实现中免费出现,但它在实践中有用吗?毕竟,它的返回类型最多就是 void*
,那么这有什么好处呢?
In C++, the T q = dynamic_cast<T>(p);
construction performs a runtime cast of a pointer p
to some other pointer type T
that must appear in the inheritance hierarchy of the dynamic type of *p
in order to succeed. That is all fine and well.
However, it is also possible to perform dynamic_cast<void*>(p)
, which will simply return a pointer to the "most derived object" (see 5.2.7::7 in C++11). I understand that this feature probably comes out for free in the implementation of the dynamic cast, but is it useful in practice? After all, its return type is at best void*
, so what good is this?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(7)
dynamic_cast()
确实可以用于检查身份,即使处理多重继承也是如此。试试这个代码:
输出:
The
dynamic_cast<void*>()
can indeed be used to check for identity, even if dealing with multiple inheritance.Try this code:
Output:
请记住,C++ 允许您以旧的 C 方式做事。
假设我有一些 API,其中我被迫通过类型
void*
走私对象指针,但它最终传递到的回调将知道其动态类型:错误!代码是错误的,因为它在存在多重继承的情况下失败(并且也不保证在不存在多重继承的情况下工作)。
当然,该 API 并不是很 C++ 风格,如果我继承自
ActualType
,即使是“正确”的代码也可能会出错。因此,我不会说这是对dynamic_cast
的出色使用,但它确实是一种使用。Bear in mind that C++ lets you do things the old C way.
Suppose I have some API in which I'm forced to smuggle an object pointer through the type
void*
, but where the callback it's eventually passed to will know its dynamic type:The WRONG! code is wrong because it fails in the presence of multiple inheritance (and isn't guaranteed to work in the absence, either).
Of course, the API isn't very C++-style, and even the "right" code can go wrong if I inherit from
ActualType
. So I wouldn't claim that this is a brilliant use ofdynamic_cast<void*>
, but it's a use.自 C 时代以来,将指针强制转换为
void*
就具有其重要性。最合适的地方是操作系统的内存管理器内部。它必须存储您创建的所有指针和对象。通过将其存储在 void* 中,他们将其概括为将任何对象存储到内存管理器数据结构上,该数据结构可以是堆/B+树或简单的数组列表。
为了简单起见,以创建通用项目的
列表
为例(列表包含完全不同类的项目)。这只有使用void*
才能实现。标准规定,对于非法类型转换,dynamic_cast 应该返回 null,并且标准还保证任何指针都应该能够将其类型转换为 void* 并从中返回,只有函数指针除外。
正常应用程序级别的
void*
类型转换的实际使用量非常少,但它广泛用于低级/嵌入式系统。通常你会想使用reinterpret_cast来处理低级的东西,比如在8086中,它用于偏移相同基址的指针来获取地址,但不限于此。
编辑:
标准表示,即使使用
dynamic_cast<>
,您也可以将任何指针转换为void*
,但没有任何地方表明您不能转换void*
> 回到对象。对于大多数用途来说,它是一条单向街道,但也有一些不可避免的用途。
它只是说
dynamic_cast<>
需要类型信息才能将其转换回请求的类型。有许多 API 要求您将
void*
传递给某些对象,例如。 java/Jni 代码将对象作为void*
传递。如果没有类型信息,您将无法进行转换。如果您有足够的信心,请求的类型是正确的,您可以要求编译器使用一个技巧来执行
dynmaic_cast
。看看这段代码:
如果这有任何不正确的地方,请纠正我。
Casting pointers to
void*
has its importance since way back in C days.Most suitable place is inside the memory manager of Operating System. It has to store all the pointer and the object of what you create. By storing it in void* they generalize it to store any object on to the memory manager data structure which could be
heap/B+Tree
or simplearraylist
.For simplicity take example of creating a
list
of generic items(List contains items of completely different classes). That would be possible only usingvoid*
.standard says that dynamic_cast should return null for illegal type casting and standard also guarantees that any pointer should be able to type cast it to void* and back from it with only exception of function pointers.
Normal application level practical usage is very less for
void*
typecasting but it is used extensively in low level/embedded systems.Normally you would want to use reinterpret_cast for low level stuff, like in 8086 it is used to offset pointer of same base to get the address but not restricted to this.
Edit:
Standard says that you can convert any pointer to
void*
even withdynamic_cast<>
but it no where states that you can not convert thevoid*
back to the object.For most usage, its a one way street but there are some unavoidable usage.
It just says that
dynamic_cast<>
needs type information for converting it back to the requested type.There are many API's that require you to pass
void*
to some object eg. java/Jni Code passes the object asvoid*
.Without type info you cannot do the casting.If you are confident enough that type requested is correct you can ask compiler to do the
dynmaic_cast<>
with a trick.Look at this code:
Please correct me if this is not correct in any way.
扩展@BruceAdi的答案并受到此讨论的启发,这里有一个可能需要指针调整的多态情况。假设我们有这个工厂类型的设置:
现在我可以说:
但是我如何手动清理它?我需要实际的内存地址来调用
::operator delete
:或者我可以重新使用内存:
Expanding on @BruceAdi's answer and inspired by this discussion, here's a polymorphic situation which may require pointer adjustment. Suppose we have this factory-type setup:
Now I could say:
But how would I clean this up manually? I need the actual memory address to call
::operator delete
:Or I could re-use the memory:
当我们将存储放回内存池但我们只保留指向基类的指针时,它很有用。这种情况我们应该弄清楚原来的地址。
it is usefull when we put the storage back to memory pool but we only keep a pointer to the base class. This case we should figure out the original address.
不要在家里这样做
警告:如果
D
定义为:则这不会工作,并且没有办法让它工作。
Don't do that at home
Warning: This will not work if
D
is defined as:and there is no way to make it work.
这可能是通过 ABI 提供不透明指针的一种方法。不透明指针——更一般地说,不透明数据类型——用于传递对象和其他库代码和客户端代码之间的资源,使得客户端代码可以与库的实现细节隔离。还有其他方法 可以肯定的是,其中一些方法可能更适合特定的用例。
Windows 在其 API 中大量使用不透明指针。例如,我相信,
HANDLE
通常是一个不透明的指针,指向您拥有HANDLE
的实际资源。HANDLE
可以是内核对象,如文件、GDI 对象和各种类型的用户对象 - 所有这些对象在实现上都必须有很大不同,但都作为HANDLE 返回
给用户。This might be one way to provide an Opaque Pointer through an ABI. Opaque Pointers -- and, more generally, Opaque Data Types -- are used to pass objects and other resources around between library code and client code in such a way that the client code can be isolated from the implementation details of the library. There are other ways to accomplish this, to be sure, and maybe some of them would be better for a particular use case.
Windows makes a lot of use of Opaque Pointers in its API.
HANDLE
is, I believe, generally an opaque pointer to the actual resource you have aHANDLE
to, for example.HANDLE
s can be Kernel Objects like files, GDI objects, and all sorts of User Objects of various kinds -- all of which must be vastly different in implementation, but all are returned as aHANDLE
to the user.