在学习C++的时候,都知道不要手动调用析构函数,也不要在构造函数、析构函数里调用虚函数。工作这么多年,这些冷门的知识极少用到,渐渐被繁杂的业务逻辑淹没掉。

  不过,最近项目里出现了析构函数没有被正确地调用,导致内存泄漏。代码大概如下:

class base_mem_alloc
{
public:
  base_mem_alloc() {}
  virtual ~base_mem_alloc() {}
}; class mem_alloc : public base_mem_alloc
{
public:
~mem_alloc() {}
virtual ~mem_alloc()
{
// 释放内存
}
}; class test_conf
{
public:
bool reload();
private:
class mem_alloc _alloc;
} bool test_conf::reload()
{
// 正常写法,释放所有内存,但没有这个接口
// _alloc.release() // 有问题的写法
class mem_alloc tmp;
_alloc.~mem_alloc(); // 通过析构函数释放内存
_alloc = tmp; // 通过拷贝构造函数将内部变量初始化 // 使用_alloc分配内存
}

  公司的框架要求使用统一的内存分配器。像读取配置这种逻辑,在配置不需要的时候(也就是关掉进程)是直接从分配器统一释放的,但这框架有点年代了,之前没有考虑游戏热更配置的问题。现在要求能重新加载配置,那么就少了一个释放内存的接口。于是,便通过析构函数释放内存,然后再用拷贝构造函数把一个临时分配器拷贝过来。虽然这种写法极其少见,但咋一看,好像也没问题。然后不幸的是,这种写法真的有问题,~mem_alloc()这个析构函数是无法正常调用的。

  代码中的内存分配器用了多态,C++的多态是依赖虚函数表实现的,虚函数表是在构造函数的时候一步步创建,在析构函数一步步销毁。之所以说是一步步,因为C++在调用构造函数时,会从基类构造函数--子类构造函数构造,析构时从子类析构函数--基类析构函数。在这个过程中,对象的类型也是会变的,调用基类构造函数的时候,他的类型就是基类,调用子类构造函数时,就是子类。析构时则反过来,所以析构完成后,对象的类型是基类(理论上讲,不再存在这个对象,但他的数据遗留是基类)。参考:https://www.artima.com/cppsource/nevercall.html

    During base class construction of a derived class object, the type of the object
is that of the base class. Not only do virtual functions resolve to the base class,
but the parts of the language using runtime type information (e.g., dynamic_cast (see Item 27)
and typeid) treat the object as a base class type.An object doesn't become a derived class object
until execution of a derived class constructor begins. The same reasoning applies during destruction. Once a derived class destructor has run,
the object's derived class data members assume undefined values, so C++ treats them as if
they no longer exist. Upon entry to the base class destructor, the object becomes a base class
object, and all parts of C++—virtual functions, dynamic_casts, etc.—treat it that way.

  由于虚函数表在构造、析构过程中是变化的,因此在这时调用虚函数可能不会得到正确的结果。而像dynamic_cast这种依赖运行时的转换,也不可用。上面出问题的代码中,手动调用析构函数,第一次是能够正常调用的,然后就变成了基类。在使用拷贝构造函数时,会拷贝临时对象的数据,但是并不会重建虚函数表。由于在我们项目的代码中大部分功能是由基类完成的,使用拷贝构造函数后,对象的内存数据也没有被破坏。因此运行起来并没有什么太大的问题,再加上这个地方是需要多次重新加载配置才能重现,导致这个问题被隐藏了一段时间。

  其实这个问题很好解决。加个释放函数就OK,或者换用指针,delete掉再new就可以了。用placement new在原来对象上重新创建一个对象也行。最后说一句,写代码,不是写得越复杂越高深才好,而是越通俗易懂越好,少用一些奇奇怪怪的写法用法。毕竟代码多数是需要维护的,公司招的人每个人的水平都不一样,通俗的代码则更容易维护。

C++手动调用析构函数无效问题排查的更多相关文章

  1. 编写SqlHelper使用,在将ExecuteReader方法封装进而读取数据库中的数据时会产生Additional information: 阅读器关闭时尝试调用 Read 无效问题,解决方法与解释

    在自学杨中科老师的视频教学时,拓展编写SqlHelper使用,在将ExecuteReader方法封装进而读取数据库中的数据时 会产生Additional information: 阅读器关闭时尝试调用 ...

  2. 【Win 10应用开发】手动调用WCF服务

    调用服务最简单的方法就是,直接在VS里面添加服务引用,输入服务的地址即可,无论是普通Web服务,还是WCF服务均可.VS会根据获取到的元数据,自动生成客户端代码. 如果服务的调用量很大,应用广泛,可以 ...

  3. 由STL所想到的 C++显示调用析构函数

    在STL中的容器中的析构函数中,会经常调用destroy()这个函数 在STL中  destroy()被重载了 这点在这里到不去讨论 这里贴最简单的那个版本 template<class T&g ...

  4. Android 关于ListView中adapter调用notifyDataSetChanged无效的原因

    话说这个问题已经困扰我很久了,一直找不到原因,我以为只要数据变了,调用adapter的notifyDataSetChanged就会更新列表,最近在做微博帐号管理这一块,想着动态更新列表,数据是变了,但 ...

  5. Android开发之关于ListView中adapter调用notifyDataSetChanged无效的原因

    1.数据源没有更新,调用notifyDataSetChanged无效. 2.数据源更新了,但是它指向新的引用,调用notifyDataSetChanged无效. 3.数据源更新了,但是adpter没有 ...

  6. SherlockactionBar中手动调用onCreateOptionsMenu的办法

    我们有时候要做做事的时候,要提前拿到 ActionBar中的menuItem,但是,会出现为空的情况.怎么办呢? 比如这里: @Override public boolean onCreateOpti ...

  7. 手动调用run方法和普通方法调用没有区别

    手动调用run方法和普通方法调用没有区别

  8. Android 手动调用 返回键

    有人想通过下面代码来实现手动调用返回键,很可惜会出现空指针异常. this.onKeyDown(KeyEvent.KEYCODE_BACK, null); 我们可以通过调用 onBackPressed ...

  9. 关于c++显示调用析构函数的陷阱

    版权声明:欢迎转载,注明出处就好!如果不喜欢请留言说明原因再踩哦,谢谢,我也可以知道原因,不断进步!!   目录(?)[+]   一.文章来由 现在在写一个项目,需要用到多叉树存储结构,但是在某个时候 ...

随机推荐

  1. [hibernate]log4jdbc日志输出完整SQL语句

    1.在maven引入: <dependency> <groupId>log4j</groupId> <artifactId>log4j</arti ...

  2. 西湖论剑2019部分writeup

    做了一天水了几道题发现自己比较菜,mfc最后也没怼出来,被自己菜哭 easycpp c++的stl算法,先读入一个数组,再产生一个斐波拉契数列数组 main::{lambda(int)#1}::ope ...

  3. 如何为ubuntu等Linux系统扩容(LVM)

    第一步:磁盘分区 fdisk /dev/sdb root@ubuntu:/home/ubuntu# fdisk /dev/sdb Welcome to fdisk (util-linux 2.27.1 ...

  4. Core Data概述(转)

    Core Data是一个模型层的技术.Core Data帮助你建立代表程序状态的模型层.Core Data也是一种持久化技术,它能将模型对象的状态持久化到磁盘,但它最重要的特点是:Core Data不 ...

  5. Windows核心编程随笔

    最近在学习Windows底层原理,准备写个系列文章分享给大家,Michael Li(微软实习期间的Mentor,为人超好)在知乎回答过一些关于学习Windows原理的书籍推荐,大家可以拜读其中一本来入 ...

  6. elasticsearch查询操作

    #查看节点信息 curl -X GET http://localhost:9200/_nodes #打开文件数信息 curl -X GET http://localhost:9200/_nodes/s ...

  7. 机器学习(Machine Learning)与深度学习(Deep Learning)资料汇总

    <Brief History of Machine Learning> 介绍:这是一篇介绍机器学习历史的文章,介绍很全面,从感知机.神经网络.决策树.SVM.Adaboost到随机森林.D ...

  8. 性能优化-屏幕常亮与CPU唤醒

    Android在不使用的时候,屏幕在一段时间以后会变暗,再过一段时间就会熄屏,此时CPU就会休眠,那么在这个时候,Timer.Handler.Thread.Service等都会暂停,有时候我们需要屏幕 ...

  9. Go语言入门篇-gRPC基于golang & java简单实现

    一.什么是RPC 1.简介: RPC:Remote Procedure Call,远程过程调用.简单来说就是两个进程之间的数据交互. 正常服务端的接口服务是提供给用户端(在Web开发中就是浏览器)或者 ...

  10. GitHub项目管理维护实用教程

    GitHub项目维护教程   1)注册GitHub账户并登陆: 2)在Windows cmd(或Ubuntu中的terminal)中cd到自己的工作目录,将仓库clone下来: 命令: git clo ...