首页 > 后端开发 > C++ > 为什么在 C 中删除具有基指针的派生对象数组是未定义的行为?

为什么在 C 中删除具有基指针的派生对象数组是未定义的行为?

Susan Sarandon
发布: 2024-10-29 05:51:02
原创
790 人浏览过

 Why is Deleting an Array of Derived Objects with a Base Pointer Undefined Behavior in C  ?

具有派生对象的数组:Delete[]的陷阱

C 标准明确指出,使用基数删除派生对象数组指针导致未定义的行为。这个看似模棱两可的规则引发了对其基本原理和潜在含义的质疑。

为了理解这个概念,让我们回顾一下静态类型和动态类型之间的区别。假设我们有以下代码片段:

<code class="cpp">struct B { virtual ~B() {} };
struct D : B {};

B* p = new D();</code>
登录后复制

在这种情况下,p 的静态类型是 B*,而 *p 的动态类型是 D。这是因为 p 指向类型的子对象D 已被构造来代替 B 对象。

但是,当我们使用基指针声明数组时,会出现微妙的差异。考虑以下代码:

<code class="cpp">B* p = new D[20];</code>
登录后复制

这里,p 指向数组中第一个元素的基子对象,而不是第一个元素本身。因此,使用delete [] p违反了数组的静态和动态类型匹配的要求。

这种未定义行为的原因在于它会给运行时环境带来潜在的低效率和复杂性。要使用基指针正确删除派生对象的数组,实现需要动态检索数组的元素类型,并将每个指针转换为正确的类型,然后再执行删除。这种开销被认为是不必要的,特别是考虑到多态数组的有限用例。

此外,使用基指针删除派生数组会带来另一个问题。由于 p 指向子对象,因此对数组元素的后续访问(例如,i > 0 时的 p[i])将产生不正确的结果。这进一步支持了对派生对象数组使用带有基指针的delete []的限制。

总之,使用基指针删除派生对象数组的未定义行为源于其固有的复杂性和缺乏它会带来的实用性。虽然可以实现专门的删除[]来处理这种情况,但它会付出成本效益和可用性的代价,与 C 的设计理念不一致。

以上是为什么在 C 中删除具有基指针的派生对象数组是未定义的行为?的详细内容。更多信息请关注PHP中文网其他相关文章!

来源:php.cn
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
作者最新文章
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板