解决duilib使用zip换肤卡顿的问题(附将资源集成到程序中的操作方法)

简介: 转载请说明原出处,谢谢~~        今天在做单子是,客户要求做换肤功能,为此我专门写了一个换肤函数,并且把各种皮肤资源压缩为各个zip文件来换肤。

转载请说明原出处,谢谢~~

       今天在做单子是,客户要求做换肤功能,为此我专门写了一个换肤函数,并且把各种皮肤资源压缩为各个zip文件来换肤。但是客户反映程序运行缓慢,我测试后发现的确明显可以看出慢了不少。最后发现问题在于把皮肤资源都集成到了zip文件中,程序在刷新界面时会重新从zip文件中读取对应的资源,导致了界面反映卡顿。之前直接把z资源放到目录里或者把zip集成到程序内部,都是没问题的。但是如果要换肤就需要用到zip来压缩资源了。

       duilib的WinImplBase类为我们提供了4种加载资源的方法:


	enum UILIB_RESOURCETYPE
	{
		UILIB_FILE=1,		// 来自磁盘文件
		UILIB_ZIP,		// 来自磁盘zip压缩包
		UILIB_RESOURCE,		// 来自资源
		UILIB_ZIPRESOURCE,	// 来自资源的zip压缩包
	};

       使用磁盘文件是最简单的方法,开发时选择这个方法,但是实际发布程序后为了资源的保密就很少这样做了;使用zip文件也是常用的方法,但是问题就在于资源比较多时界面就有明显卡顿;直接使用资源肯定是快速的,但是这个方法就太繁琐了,需要逐个去处理每个资源,用法见MenuDemo;使用资源的zip压缩包,这个是我最常用的,把资源压缩为zip然后集成到程序中,这样不但可以保密资源,而且不会有卡顿的现象。

     

一、 这里先把使用“资源的zip压缩包”方法说明一下:

        1.让自己的窗体类继承WinImplBase类,并且重写GetSkinFile、GetSkinFolder、GetResourceType、GetResourceID这四个方法

        2.在vs中添加自定义资源,找到自己的zip文件并添加,资源类型填写为“ZIPRES”,得到资源的ID号,比如这里为“IDR_ZIPRES2”

        3.GetSkinFile中返回主窗体的xml文件的名字

        4.GetSkinFolder中返回资源文件所在的目录

        5.GetResourceType中返回资源类型,此时应该写为“return UILIB_ZIPRESOURCE;”

        6.GetResourceID中返回对应的zip资源的ID,例如:“return MAKEINTRESOURCE(IDR_ZIPRES2);”

       7.编译程序,这样就可以使用资源的zip压缩包了。

    (ps:在WinMain函数里只要写一句 CPaintManagerUI::SetInstance(hInstance);代码就够了,不需要其他任何CPaintManagerUI的代码,其他代码WinImplBase会处理的!


二、再说明一下常用的zip文件换肤方法

     使用这种方法来换肤,要求加载资源的方式使用第二种“来自磁盘的zip压缩包”方式,用法我就不说明了,duilib的多数demo都是用这种方法。

     如果要换肤,直接使用如下两句代码就可以了:

   CPaintManagerUI::SetResourceZip(_T("skin2.zip")); // 这里写入新的皮肤包的文件名就行了
   CPaintManagerUI::ReloadSkin();


三、使用“来自资源的zip压缩包”方法换肤

     这样做有两个好处,第一是不会有使用单独zip文件那种卡顿现象,第二是资源文件会相对更安全一些。

     我测试了一下,默认情况下不能让duilib使用这种方法来换肤,原因会在后面给出。接下来直接说明怎么使用这个方法:

     从常用的zip文件换肤方法中可以看出,换肤的关键就是重新设置zip文件,也就是说SetResourceZip是换肤的关键函数,他重新指定了zip文件。这个函数有两个版本,一个是加载文件中的zip,另一个是加载资源中的zip,我们需要的就是第二个版本的SetResourceZip。程序调用ReloadSkin函数后,会通知所有控件去重新加载图片资源,图片资源的加载会通过LoadImage函数,这个函数会根据加载资源类型的不同而去选择从不同的地方去试图找到资源并加载。

      在使用“资源的zip压缩包”方法的前提下,如果要换肤就使用如下函数,函数的参数是新换皮肤的资源ID,比如“IDR_ZIPRES2”,函数实际就是从程序资源中找到对应的皮肤zip文件,并且调用对应的SetResourceZip函数加载资源:


void CFrameWnd::ReloadZipResource(int ID)
{

	HRSRC hResource = ::FindResource(m_PaintManager.GetResourceDll(), MAKEINTRESOURCE(ID), _T("ZIPRES"));
	if( hResource == NULL )
		return ;
	DWORD dwSize = 0;
	HGLOBAL hGlobal = ::LoadResource(m_PaintManager.GetResourceDll(), hResource);
	if( hGlobal == NULL ) 
	{
#if defined(WIN32) && !defined(UNDER_CE)
		::FreeResource(hResource);
#endif
		return ;
	}
	dwSize = ::SizeofResource(m_PaintManager.GetResourceDll(), hResource);
	if( dwSize == 0 )
		return ;

	CPaintManagerUI::SetResourceZip((LPBYTE)::LockResource(hGlobal), dwSize);

#if defined(WIN32) && !defined(UNDER_CE)
	::FreeResource(hResource);
#endif

	CPaintManagerUI::ReloadSkin();
}



           理论上这就应该就可以了,但是实际测试还有问题,后来发现是SetResourceZip函数的定义有些问题:

void CPaintManagerUI::SetResourceZip(LPVOID pVoid, unsigned int len)
{
    if( m_pStrResourceZip == _T("membuffer") ) return;
    if( m_bCachedResourceZip && m_hResourceZip != NULL ) {
        CloseZip((HZIP)m_hResourceZip);
        m_hResourceZip = NULL;
    }
    m_pStrResourceZip = _T("membuffer");
    m_bCachedResourceZip = true;
    if( m_bCachedResourceZip ) 
        m_hResourceZip = (HANDLE)OpenZip(pVoid, len, 3);
}

          可以看到如果使用资源zip文件,那么m_pStrResourceZip变量就会保存_T("membuffer")字符串,当再次调用SetResourceZip函数时,由于第一句代码的判断就会导致函数直接返回,所以这里直接注释掉第一句代码就可以了。

         至此,就可以使用资源中的zip文件来换肤了,两全其美。这里还可以扩展,可以把zip资源都继承到一个dll文件中,然后在加载函数里先加载dll,然后从dll加载资源,这样既可以让皮肤资源独立为文件,加载也快速,并且资源也安全。这个代码很好写,我这里就不提供了。

         如有错误,请在博客留言!


         Redrain   2014.10.16

目录
相关文章
|
4月前
|
Kubernetes Go 持续交付
一个基于Go程序的持续集成/持续部署(CI/CD)
本教程通过一个简单的Go程序示例,展示了如何使用GitHub Actions实现从代码提交到Kubernetes部署的CI/CD流程。首先创建并版本控制Go项目,接着编写Dockerfile构建镜像,再配置CI/CD流程自动化构建、推送Docker镜像及部署应用。此流程基于GitHub仓库,适用于快速迭代开发。
95 3
|
4月前
|
Kubernetes 持续交付 Go
创建一个基于Go程序的持续集成/持续部署(CI/CD)流水线
创建一个基于Go程序的持续集成/持续部署(CI/CD)流水线
|
3月前
|
存储 缓存 安全
如何使用 PHP 将天气跟踪集成到 Web 应用程序中
如何使用 PHP 将天气跟踪集成到 Web 应用程序中
41 0
|
5月前
|
移动开发 小程序 测试技术
项目管理和持续集成系统搭建问题之帮助以诺行管理任务和资源如何解决
项目管理和持续集成系统搭建问题之帮助以诺行管理任务和资源如何解决
44 2
|
5月前
|
SQL DataWorks 安全
DataWorks产品使用合集之调度资源组与集成资源内部的实例如何进行共用
DataWorks作为一站式的数据开发与治理平台,提供了从数据采集、清洗、开发、调度、服务化、质量监控到安全管理的全套解决方案,帮助企业构建高效、规范、安全的大数据处理体系。以下是对DataWorks产品使用合集的概述,涵盖数据处理的各个环节。
|
5月前
|
缓存 NoSQL 网络协议
【Azure Redis 缓存 Azure Cache For Redis】在创建高级层Redis(P1)集成虚拟网络(VNET)后,如何测试VNET中资源如何成功访问及配置白名单的效果
【Azure Redis 缓存 Azure Cache For Redis】在创建高级层Redis(P1)集成虚拟网络(VNET)后,如何测试VNET中资源如何成功访问及配置白名单的效果
|
5月前
|
数据采集 DataWorks 监控
DataWorks产品使用合集之公共集成资源组如何切换独享资源
DataWorks作为一站式的数据开发与治理平台,提供了从数据采集、清洗、开发、调度、服务化、质量监控到安全管理的全套解决方案,帮助企业构建高效、规范、安全的大数据处理体系。以下是对DataWorks产品使用合集的概述,涵盖数据处理的各个环节。
|
5月前
|
开发框架 NoSQL .NET
使用 Asp.net core webapi 集成配置系统,提高程序的灵活和可维护性
使用 Asp.net core webapi 集成配置系统,提高程序的灵活和可维护性
|
6月前
|
SQL 存储 JSON
DataWorks产品使用合集之没有dev环境的project,如何创建数据集成任务时完成网络与资源配置
DataWorks作为一站式的数据开发与治理平台,提供了从数据采集、清洗、开发、调度、服务化、质量监控到安全管理的全套解决方案,帮助企业构建高效、规范、安全的大数据处理体系。以下是对DataWorks产品使用合集的概述,涵盖数据处理的各个环节。
|
6月前
|
SQL Java 数据库
实时计算 Flink版产品使用问题之Spring Boot集成Flink可以通过什么方式实现通过接口启动和关闭Flink程序
实时计算Flink版作为一种强大的流处理和批处理统一的计算框架,广泛应用于各种需要实时数据处理和分析的场景。实时计算Flink版通常结合SQL接口、DataStream API、以及与上下游数据源和存储系统的丰富连接器,提供了一套全面的解决方案,以应对各种实时计算需求。其低延迟、高吞吐、容错性强的特点,使其成为众多企业和组织实时数据处理首选的技术平台。以下是实时计算Flink版的一些典型使用合集。