At a glance
软件开发中,可能需要把用到的资源文件嵌入到二进制执行文件中,例如生成单个执行文件、防止机密或版权信息被PE工具查看或修改、嵌入图片资源到静态库中等等。在Mac OSX Cocoa 或iOS开发中,编译生成的Product.app是一个APP包,其实就是个文件夹,右键Show Package Contents或者去掉 .app 扩展后双击打开就可以查看包内容,修改包里面的任何资源文件都不会影响程序正常运行,要提交到App Store的程序在修改资源文件后运行下codesign(安装Xcode时已经装了这个dev tool)重签名就可以了:
$ codesign -fvs "Your Identity" path/to/appfile.app
我之前写的 ElfCodeSigner 就是基于codesign的。
所以嵌入二进制资源文件也算是一种有效的保护手段。
结合我以前Windows开发中用到的措施,我想到了三种方案:
最近公司项目中需要把一些图片资源嵌入到静态库中,我考虑了下还是用bytes的方法比较好,Cocoa原生支持其不会留下被修改的余地。一些简单的图标用CG画上去。这样公开出去的库只有一个.a文件和几个必要的.h文件。
下午闲着没事,就操起Xcode在新配的MacBook上写下了我这第一个跑在Mac OS上的C程序。
Release notes
rtb(Resource To Bytes)是一个命令行小工具,将二进制资源文件转换为bytes数组,方便在程序中使用嵌入资源。
由于时间仓促,rtb仅在Mac OS SL上测试过,对应应用在Mac OS桌面程序和iOS App中测试过,以后有时间再改成跨平台的。目前没发现什么bug。
使用方法:打开Terminal,cd到rtb所在目录,运行
$ ./rtb image.png
将生成
unsigned char image_png[] = { ..... };
unsigned int image_png_len = 16045;
变量名称根据资源文件名而来,数字开头的会加前缀 "__" ,文件名中非英文和数字的字符转换为下划线"_"(使用了isalnum()测试函数),在数组变量名称添加"_len"后缀作为数组长度变量名称。
例如"123te的 st5.png"将生成变量名unsigned char __123te____st5_png[] 和 unsigned int __123te____st5_png_len 。
Example
首先执行rtb生成.h文件:
$ ./rtb test.png > test.png.h
新建一个Window Base的iPhone项目,添加test.png.h文件到项目中,在
- (BOOL)application: didFinishLaunchingWithOptions:
方法中创建一个UIImage并把它添加到一个ImageView中:
1 #import "test.png.h"
2
3 //............
4 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
5 {
6 [self.window makeKeyAndVisible];
7
8 unsigned char *imgBytes = test_png;
9 NSUInteger imgLenght = test_png_len;
10 NSData *imgData = [NSData dataWithBytesNoCopy:imgBytes length:imgLenght freeWhenDone:NO];
11 // UIImage *image = [UIImage imageWithData:imgData];
12 // or
13 UIImage *image = [[UIImage alloc] initWithData:imgData];
14
15 UIImageView *imageView = [[UIImageView alloc] initWithImage:image];
16 imageView.frame = self.window.frame;
17 imageView.contentMode = UIViewContentModeCenter;
18 [self.window addSubview:imageView];
19 [imageView release];
20
21 [image release];
22 return YES;
23 }
因为在程序加载时已经将test.png.h中的数组加载到内存中,所以使用NSData的dataWithBytesNoCopy方法即可,不需要再Copy一份,转换成NSData后也不需要释放它,所以freeWhenDonw参数值NO.
Code Review
我C语言很烂,就不贴代码了。注释、空行加起来80多行,核心功能代码就是fopen这个资源文件:
if ((fp = fopen(argv[1], "r")) != NULL)
从文件头取到(getc(fp))到EOF,fprintf(stdout,"0x%02x",ch):
for (p = 0; (length < 0 || p < length) && (ch = getc(fp)) != EOF; p++)
{
char *c = p ? ",\n " : " ";
fprintf(fpo, "%s0x%02x", (p % COLS) ? ", " : c, ch);
}
Download
已收录至Cocoa-Utilities: https://github.com/Sundae/Cocoa-Utilities