[技术文章] cocos2d-x读取加密压缩ZIP文件

[复制链接]
作者
超哥   发布于2022-8-23 18:19:21 来自河北

1,修改源代码


这里可以修改CCFileUtils.cpp里的getFileDataFromZip()函数,也可以修改ZipUtils.cpp里的getFileData(),

这里我修改的是ZipUtils.cpp,因为这个是专门操作ZIP的。而且后面有不同的用处。

修改很简单,只需要把getFileData()函数改一下参数,和unzOpenCurrentFile()函数就可以了。

先到ZipUtils.h把函数声明加一个参数const char *password,给它一个默认值NULL,这样当你读取没有加密的ZIP时,就不需要填写密码参数了

函数声明修改后如下:

0.jpg


然后再到函数定义,把unzOpenCurrentFile()函数修改为unzOpenCurrentFilePassword()并传入密码参数


好了,这个时候就修改完了,保存,拿一个没加密的和加密的ZIP试一下,你会发现没加密的可以正常读取,加密的读取不到,为什么呢?

这个时候就要看看它们调用的unzOpenCurrentFile3()函数,看看是哪发生错误了。

进入这个函数体我们发现在1529行时,函数就被返回了,如下图


1.jpg

而这里有个预编译,如果没有定义NOUNCRYPT就往下执行,定义了就返回,搜索一下发现,在unzip.cpp头文件处被定义了

2.jpg

如果没有定义就定义,原来当初那个zlib库被下令不给支持加密,就把这货给定义出来了(个人猜测),所以cocos2dx官方也没去掉它

好,接下来就是解除封印,把这个#define NOUNCRYPT给注释掉,保存,编译,OK,可以正常读取加密了



2,使用加密ZIP里面的资源文件


经过上面的修改,我们现在已经可以正常读取加密的ZIP文件了,那读取出来数据怎么使用呢?这对于刚刚接触这方面的新人(比如我)来说,还是有

点难度的。由于接触编程的时间不长,花了些时间才知道读取出来的数据怎样使用。

3.jpg

到此,在windows平台上使用是没有问题的了,没错,只是windows平台,但是zlib是跨平台的呀,为什么只可以windows平台呢?到Android平台编译一下你就知道了。



3,Android平台,小坑


对于上面的代码有去Android平台编译过的都会收到一条错误,如下图

4.jpg

对于刚入行的我,第一次 看到的时候,这什么鬼,const z_crc_t*{aka const unsigned int*},没见过这样的提示,aka是什么鬼,JAVA的?

然后我就去想去看看get_crc_table()函数的定义,进去发现,是const long呀,为什么提示什么const z_crc_t*{aka.....}?第一反应,跨平台文件

果然,Android平台调用的是项目文件夹下cocos2d\external\zlib\prebuilt\android里面的动态链接库呀,这个想改也改不了,所以我们只能在unzip.cpp里面动手脚了。

没错,强转,因为一直使用C++嘛,所以刚开始我就用C++的static_cast强转,不行的,还是报这个错。

不管我转什么类型,错误都没有改变,奇了怪了,不是应该提示成我转的类型吗?后来发现自己C++基础问题,对static_cast不理解,以为什么都能

转的。于是使出强力的reinterpret_cast给转了,当然,使用C语言的强转也OK的,还少敲些代码,如下图

5.jpg


OK,到此,Android和Windows都可以用了,至于苹果嘛,没有试,目前暂时没接触到苹果这方面


另外,由于Android和Windows下路径是不同的,所以要注意读取zip文件时不能使用上面资源使用那样读取,不知道怎么读取的可以往下看。



4,额外的,关于cocos2dx异步加载


原先的cocos2dx资源异步加载函数Director::getInstance()->getTextureCache()->addImageAsync()是通过路径来加载,

可是现在我们是从ZIP文件里获取到的资源,没有路径的,那怎么加载?

这个时候就只能改引擎源代码了(这个仅是个人愚见哈)

我们可以查看一下Director::getInstance()->getTextureCache()->addImageAsync()函数的实现,如下图,已加注释了

6.jpg


接下来,我们再跟进那个loadImage()函数,只关注我加了注释的两句

7.jpg


继续跟进initWithImageFileThreadSafe()函数,如下图

8.jpg


源代码就看到这里了,已经找到加载的地方,接下来就是修改上面的initWithImageFileThreadSafe()函数

既然这里是加载图片的,那就在这里让它加载ZIP文件里的资源。


先在Image这个头文件定义一个自己的静态成员函数,用来打开zip,如下图,Data,ZipFile类前置声明

9.jpg


然后再来函数定义

99.jpg


这里要说一下打开ZIP的方式和上面资源使用时的打开方式不一样,原因就是Android的资源是放在assets里面的,

如果直接打开是不行的,zipFIle这个类是你传什么路径,它就打开什么路径,这换到Android下是不存在的,会闪退

当然通过FileUtils::getInstance()->fullPathForFilename()来获取全路径再打开也是不行滴,

原因嘛,因为Android下读取是不同的,看网上有人说那个assets本身就是一个包,这个暂时没有去了解过。

我就按照网上那些人说的,assets如果本身是个包的话,那就从包里也把这个zip文件获取出来不就行了,于是就这样了


另外ZipFile::createWithBuffer()这个函数,里面是调用了一个unzip.cpp的unzOpenBuffer()函数,这个函数是cocos2dx官方加的,

我在下载的zlib库中没看到有这个函数,这个函数就是为了配合cocos2dx的Data类使用的。


然后我们再到initWithImageFileThreadSafe()这个函数在_filePath = fullpath下面插入一段自己的代码,如下图

999.jpg


这里要注意哈,不要把打开ZIP文件的函数写在这里,如果写在这里,那每加载一张图片,就要打开一次压缩文件,

就好比我们平时使用压缩文件一样,你要获取里面的资源,肯定先打开,然后全部获取了,再关闭呀,

如果把打开写在这里,那就相当于每获取一张图片,你就打开一次压缩文件,那就相当于有多少个资源就打开多少次

因为打开ZIP文件后,所有的资源都已经在内存中了,所以上面才把打开ZIP文件写成静态的。


好了,接下来还要改一下CCTextureCache.cpp里面的addImageAsync()函数一小部分


先把第二行的

        std::string fullpath = FileUtils::getInstance()->fullPathForFilename(path);

改成

        std::string fullpath = path;

原代码中是通过传进来的路径获取全路径,但我们要读取的是ZIP里面的,所以这里不改就会返回空值,

而这个路径不单单是用来加载图片的,这里它把路径这个字符作为这个图片的Key值,存到map容器,

如果要使用之前加载过的纹理,都是通过这个Key值来返回纹理的,比如addImage函数呀,就是通过Key值获取的。


然后再把

        if(fullpath.emty() || FileUtils::getInstance()->isFileExist( fullpath ))

改成

        if(fullpath.emty())

去掉后面的条件,这个是判断这个路径下是否存在这个文件的,因为是ZIP里的,肯定不存在的


好了,修改完了,原先异步加载怎么用,现在也怎么用,这样就可以异步加载ZIP里面的资源,

但是,使用的时候只能通过getTextureForKey()传入加载时传入的路径来获取纹理,就是Key值。

使用getTextureForKey()来获取资源就有个缺陷了,就是如果之前没有加载过的资源,那就返回空了,

[size=0.7][size=0.7]

因此我们一般都是使用addImage()传入路径来获取纹理的,因为这个函数对于之前没加载过的纹理会自动加载。

但是只会加载文件夹下的资源,不会去加载ZIP里面的资源,所以我们只有去修改源代码,把魔爪伸上addImage()。


先上源代码

9999.jpg

其实这个和那个异常加载是一样的,只是加载时调用的函数不一样,所以这次我们要改的就是那个initWithImageFile()函数

改的方法和上面那个异步加载函数一样,直接复制过来,改一个路径名,如下图

999999.jpg

但是这里还是不同的,多了一个_filePath = FileUtils::getInstance()->fullPathForFilename(path),不要删了,这个是从文件夹加载的


这个函数就改好了,然后再回到上面的addImage()函数,看到第三行那个路径代码,没错,和异步加载一样,


        std::string fullpath = FileUtils::getInstance()->fullPathForFilename(path);

改成

        std::string fullpath = path;


好了,所有代码改完了,现在可以异步加载ZIP里面的资源,也可以加载文件夹,不管之前ZIP里的资源有没有加载过,都可以直接使用

addImage()获取,另外,如果怕上面的FileUtils::getInstance()->fullPathForFilename(path)去掉后,在其它平台无法正常获取,可以加个判断,

根据ZIP文件是否存在来赋值fullpath,以上代码在windows和Android真机测试通过。


目前cocos2dx只看到了unzip.cpp文件,没有看到zip.cpp文件,所以说如果要实现压缩,需要自己下载zlib库来实现


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 创建账号

本版积分规则

Archiver|小黑屋|( 冀ICP备2021005463号 )

GMT+8, 2025-1-18 14:42 , Processed in 0.132275 second(s), 27 queries , Gzip On.

N2N1 It社区 n2n1.cn

Copyright © 2001-2021,MeiCheng.

快速回复 返回顶部 返回列表