• MFC 打印图片 dc.EndDoc(); 并没有释放怎么办? 原因:打印到pdf才会出现,打印到真实打印机就正常


    最后找到原因:打印到pdf才会出现,打印到真实打印机就正常

    MFC释放资源 BOOL bPrintingOK = dc.StartDoc(&di);

    在MFC (Microsoft Foundation Classes) 应用程序中,当你使用 CDC::StartDoc 方法开始一个打印任务时,如果该任务成功开始,则需要确保在任务完成后正确地结束打印任务,并且释放与之相关的资源。CDC::StartDoc 成功返回一个正值表示打印任务已经开始,这时你应该在打印完成后调用 CDC::EndDoc 方法来结束打印任务。如果在打印过程中遇到需要取消打印的情况,应该调用 CDC::AbortDoc

    下面是如何根据 StartDoc 的返回值来处理资源释放的一个示例:

    // 假设 di 是一个已经正确配置的 CPrintInfo 对象
    // 并且 dc 是一个准备好的打印设备上下文(CDC 对象)
    BOOL bPrintingOK = dc.StartDoc(&di);
    
    if (bPrintingOK > 0) {
        // 打印任务成功开始
        // 进行打印操作...
        
        // 假设打印操作成功完成
        if (/* 打印成功 */) {
            dc.EndDoc(); // 正常结束打印任务
        } else {
            // 如果打印过程中发生错误,取消打印任务
            dc.AbortDoc();
        }
    } else {
        // StartDoc 失败,处理错误
    }
    
    // 释放与 CDC 相关的资源
    // 如果是通过 CWnd::GetDC 获得的 CDC 对象,则需要调用 CWnd::ReleaseDC
    // 如果 CDC 对象是动态创建的(通过 new),则需要在适当的时候删除它
    // 如果是函数内局部变量或类的成员,通常不需要手动释放,除非你手动调用了 CreateDC 或 CreateIC 创建了设备上下文
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    这里的关键点是,在使用 StartDoc 开始打印任务后,根据打印任务的结果(成功或失败),使用 EndDocAbortDoc 来正确结束或取消任务。这样做是为了确保资源被适当管理,并且打印作业能够正确地在操作系统中被关闭。

    此外,对于通过 GetDC 获得的 CDC 对象,你需要使用 ReleaseDC 来释放。如果你是通过 new 创建的 CDC 对象,则需要使用 delete 来手动释放。如果 CDC 对象是自动变量或类的成员变量,那么当它们的作用域结束或类被销毁时,它们会自动被释放。

    MFC dc.EndDoc(); 并没有释放怎么办?

    在这里插入图片描述
    如果你调用了 dc.EndDoc() 来结束一个打印任务,但感觉资源并没有被释放,这里有几个可能的原因和解决方案:

    1. 确认资源未释放的具体表现

    首先,确认“资源未释放”的具体表现是什么。是内存没有回收,还是某些设备上下文(DC)或GDI对象(如字体、位图等)没有被释放?这有助于确定问题的根源。

    2. 检查其它资源的释放

    dc.EndDoc() 只负责结束打印文档,它并不直接负责释放所有与设备上下文相关联的资源。如果你在打印过程中创建了GDI对象(如字体、位图等),你需要确保自己释放这些资源。例如,如果你使用 CreateFont 创建了一个字体并将其选入DC中,你需要在不再需要它时调用 DeleteObject 来释放它。

    3. 确保正确使用 CDC 对象

    如果你是通过 CWnd::GetDC() 获得 CDC 对象的,确保在结束打印任务后调用 CWnd::ReleaseDC() 来释放它。如果是通过 new 动态创建的 CDC 对象,确保在结束使用后用 delete 来释放它。

    4. 使用智能指针管理资源

    为了避免资源泄漏,你可以考虑使用智能指针(如 std::unique_ptrstd::shared_ptr)来管理动态分配的资源,这样即使发生异常也能保证资源的正确释放。

    5. 检查设备驱动程序

    如果你确定代码中已正确处理了所有资源的释放,但仍然存在问题,问题可能出在设备驱动程序上。确保打印机和图形设备的驱动程序是最新的,有时旧的或不兼容的驱动程序会导致资源管理问题。

    6. 使用诊断工具

    使用Visual Studio的诊断工具(如诊断会话窗口、性能分析器)来跟踪资源使用情况和潜在的泄漏。这些工具可以帮助你识别程序中的资源泄漏。

    7. 联系技术支持

    如果问题仍然无法解决,考虑联系微软的技术支持或在相关开发者社区(如Stack Overflow)寻求帮助,可能有经验丰富的开发者遇到过类似的问题。

    记得,资源管理是开发过程中一个重要的方面,确保资源正确释放是防止内存泄漏和其它资源问题的关键。

    图片的加载与释放已经成功了。打印机这里BOOL bPrintingOK = dc.StartDoc(&di);新增的还没释放。

    		memDC.SelectObject(pBmp); //选回原来的
    		bPrintingOK = (dc.EndPage() > 0);   // end page
    		if (memDC.GetSafeHdc() != NULL) //释放
    		{
    			memDC.DeleteDC();
    			bitmap.DeleteObject();
    		}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    把图片全删了试试

    调用之前
    在这里插入图片描述
    调用BOOL bPrintingOK = dc.StartDoc(&di);之后
    在这里插入图片描述
    调用dc.EndDoc();之后 没变。
    函数退出也没变,多出近8M,应该减少到调用前相近大小
    没有打印图片,函数退出也没变,多出近8M,应该减少到调用前相近大小。

    MFC调用打印函数后,内存增加,退出打印函数后内存增加了8M,怎么办?

    内存在调用打印函数后增加了8MB,退出打印函数后没有释放,可能是因为资源没有被正确管理或释放。处理这类内存增加问题时,可以采取以下步骤来诊断和解决问题:

    1. 确认内存增加的原因

    • 资源未释放:确认是否所有创建的GDI对象(如位图、字体等)和动态分配的内存都被正确释放。
    • 资源泄漏检测:使用Visual Studio的诊断工具(例如,内存使用情况工具、性能剖析器)来检测内存泄漏。

    2. 使用RAII原则管理资源

    确保使用资源获取即初始化(RAII)原则管理所有资源。对于GDI对象和其他系统资源,当对象离开作用域时,应自动释放这些资源。在C++中,可以通过自定义的封装类或智能指针(如std::unique_ptr,配合自定义删除器)来实现。

    3. 代码审查

    • 审查创建和释放资源的代码:确保每个new都有对应的delete,每个Create函数调用(如CreatePenCreateFont等)都有对应的DeleteObject
    • 避免重复创建资源:在打印循环中重复创建而不释放资源,会导致内存迅速增加。

    4. 使用MFC诊断功能

    MFC提供了一套诊断功能来帮助开发者发现资源泄漏。可以在程序开始处定义_CrtSetDbgFlag来启用这些诊断功能,它会在程序退出时报告内存泄漏。

    #define _CRTDBG_MAP_ALLOC
    #include 
    #include 
    
    #ifdef _DEBUG
       _CrtSetDbgFlag ( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
    #endif
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    5. 分析内存使用

    使用第三方内存分析工具(如Valgrind,Visual Leak Detector等)来分析程序的内存使用情况,找出内存泄漏的具体位置。

    6. 优化打印逻辑

    • 减少不必要的打印预览和计算:打印预览和页面布局计算可能会消耗大量资源,确保这些操作只在必要时执行。
    • 重用GDI对象:如果可能,重用GDI对象而不是在每次打印时都创建新的对象。

    7. 搜索类似问题的解决方案

    在开发者社区(如Stack Overflow,Microsoft Developer Network)搜索是否有类似问题的讨论和解决方案,有时候可以找到具体的解决思路或代码示例。

    处理这类问题可能需要一些时间和耐心,通过细致的排查和逐步缩小问题范围,通常能够找到导致内存增加的原因,并采取相应的措施解决。

  • 相关阅读:
    基于scaffold的分子生成模型 + 筛选模型,对比它们
    博途PLC S7-1200/1500 ModbusTcp通信SCL代码篇(轮询)
    Head First设计模式(阅读笔记)-04.工厂模式
    6、传统的生产者,消费者问题和if判断产生的虚假唤醒线程问题
    Rdt2.1 和 Rdt2.2的详细解释
    Packed Ciphertexts in LWE-based Homomorphic Encryption:解读
    项目案例 | 丝路新贸易创新中心项目EPC模式下设计BIM技术应用
    WhatsApp账号被封?看看是不是你的原因!
    【Vue】Non-Props 属性是什么
    SpringBoot
  • 原文地址:https://blog.csdn.net/chenhao0568/article/details/136315706