【步兵 经验篇】图片加密之我见
来源:互联网 发布:淘宝个人中心审核进度 编辑:程序博客网 时间:2024/05/03 07:12
【步兵 经验篇】图片加密之我见 by EOS.
现在apk满大街飞,直接下载一个apk改后缀名为rar,就可以看到里面的东西,我相信大家都知道,
也就是说,如果资源不加密的话,游戏的美术资源就成了别人的换皮对象=。=
现在淘宝上5块钱一堆资源,美术大大们,如果看到自己的资源被当烂白菜,不知道什么心态。
不过就算加密了,也有人去研究解密,对解密这块我也不是太懂,感觉这是一个门高深的技术=。=
下面我就来说一个我用过的一个比较简单的图片加密方法。
PNG加密
先说一下思路:
首先要了解png二进制的前8位是PNG的文件标识,那么我破坏这个文件图片标识,
它就算被的加密了,最起码你用图片查看工具识别不出来了。
(比如png改成开封菜~)
从最简单的出发的,想到的可能就是改值或者换位,而我的加密就是基于这两点。
(ps:完全没用加密算法,不行了,感觉自己好low)
const char pngStart[16] = { 0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A, 0x00, 0x00, 0x00, 0x0D, 0x49, 0x48, 0x44, 0x52 };//89 50 4E 47 0D 0A 1A 0A 00 00 00 00 49 48 44 52const char pngEnd[12] = { 0x00, 0x00, 0x00, 0x00, 0x49, 0x45, 0x4E, 0x44, 0xAE, 0x42, 0x60, 0x82 };//00 00 00 00 49 45 4E 44 AE 42 60 82//每一位都可以是 0x00~0xff,这里用字符串偷懒了=。=char pngStartChange[16] = "what~the#fuck!e";//[15] = schar pngEndChange[12] = "oh@my%god`e";//[11] = s
png二进制流的前16位和后12位是固定的,我们把它全部干掉,然后再打乱顺序。
因为替换字符串的可以是任意的,而且又有将近30位的排列组合,我感觉加密的可行性还是很高的。
(ps:求打脸 =3= ,因为不太懂解密。)
int index = 2;for (int i = 0; i < 16; i++){ if (i % 2 == 0) { ++index; swap(buf[i], buf[size - size / index]); } else { swap(buf[i], buf[size / index]); }}for (int i = 1; i <= 12; i++){ auto c = buf[size - i]; swap(buf[size - i], buf[15 + i*2]);}
换位方式用了两种,一种是根据size,也就是图片流的大小,另一种是固定索引。
前者感觉就是感觉变数更多一些,因为每个文件大小不一样,而且换法也很多。
后者就算简单粗暴了,不过IDHR算是比较重要的所以把它也破坏掉。
文件头数据块IHDR(header chunk):它包含有PNG文件中存储的图像数据的基本信息,并要作为第一个数据块出现在PNG数据流中,而且一个PNG数据流中只能有一个文件头数据块。
经过上面的处理,我感觉我们数据流已经够复杂了,基本上已经和图片没什么关系了。
他很难知道我们篡改后的字符是什么,所以他很难换位回来。而且我们换位也是有技巧的=。=
PNG解密
解密当然就是上面的加密的逆操作,就是换回来,然后再替换回原来的png头和尾部。
是不是感觉好简单啊~ 所以这里就需要特殊的换位技巧了=。=
(ps:如果你换位的奇技淫巧!请告诉我~)
而且我们操作很简单,所以解密效率自然是杠杠的。
源码
下面是我测试用的demo源码,仅供参考。
/************************************************************************//* 文件名称: main.cpp *//* 简要描述:图片加密解密Demo *//* *//* 创建日期:忘球了 *//* *//* 作者:EOS. *//************************************************************************/#include <iostream>#include <fstream>using namespace std;const char pngStart[16] = { 0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A, 0x00, 0x00, 0x00, 0x0D, 0x49, 0x48, 0x44, 0x52 };//89 50 4E 47 0D 0A 1A 0A 00 00 00 00 49 48 44 52const char pngEnd[12] = { 0x00, 0x00, 0x00, 0x00, 0x49, 0x45, 0x4E, 0x44, 0xAE, 0x42, 0x60, 0x82 };//00 00 00 00 49 45 4E 44 AE 42 60 82char pngStartChange[16] = "what~the#fuck!e";char pngEndChange[12] = "oh@my%god`e";class PNGTransform{public: PNGTransform(fpos_t s):size(s) { buf = new char[size]; }; ~PNGTransform() { delete[] buf; }; //神一样的解说:构造析构防测漏 void transform() { for (int i = 0; i < 16; i++) { buf[i] = pngStartChange[i]; } for (int i = 1; i <= 12; i++) { buf[size - i] = pngEndChange[12-i]; } int index = 2; for (int i = 0; i < 16; i++) { if (i % 2 == 0) { ++index; swap(buf[i], buf[size - size / index]); } else { swap(buf[i], buf[size / index]); } } for (int i = 1; i <= 12; i++) { auto c = buf[size - i]; swap(buf[size - i], buf[15 + i*2]); } ofstream fout; fout.open("./mLS07.png", ios::binary); fout.write(buf, size); fout.close(); };public: char* buf; fpos_t size;};class PNGRestore{public: PNGRestore(fpos_t s) :size(s) { buf = new char[size]; }; ~PNGRestore() { delete[] buf; }; void restore() { for (int i = 1; i <= 12; i++) { swap(buf[size - i], buf[15 + i * 2]); } //逆序防止 123 213 231 -》 231 321 321 的情况 int index = 11; for (int i = 15; i >= 0; i--) { if (i % 2 == 0) { swap(buf[i], buf[size - size / index]); } else { index--; swap(buf[i], buf[size / index]); } } for (int i = 0; i < 16; i++) { buf[i] = pngStart[i]; } for (int i = 1; i <= 12; i++) { buf[size - i] = pngEnd[12 - i]; } ofstream fout; fout.open("./rLS07.png", ios::binary); fout.write(buf, size); fout.close(); };public: char* buf; fpos_t size;};template<class T>fpos_t GetFileSize(T& fs){ fs.seekg(0, ios::end); auto fileSize = fs.tellg().seekpos(); fs.seekg(0); return fileSize;}int main(){ pngStartChange[15] = pngEndChange[11] = 's';#define PNGTRANSFORM 0#define PNGBUFCHECK 0#if PNGTRANSFORM == 1 ifstream fin; fin.open("LS07.png", ios::binary); auto fileSize = GetFileSize(fin); PNGTransform trans(fileSize); fin.read(trans.buf, fileSize); fin.close(); trans.transform();#else ifstream fin; fin.open("mLS07.png", ios::binary); auto fileSize = GetFileSize(fin); PNGRestore reset(fileSize); fin.read(reset.buf, fileSize); fin.close(); reset.restore();#endif#if PNGBUFCHECK ifstream fin; fin.open("LS07.png", ios::binary); auto fileSize = GetFileSize(fin); PNGTransform trans(fileSize); fin.read(trans.buf, fileSize); fin.close(); fin.open("rLS07.png", ios::binary); fileSize = GetFileSize(fin); PNGRestore reset(fileSize); fin.read(reset.buf, fileSize); fin.close(); reset.restore(); fpos_t xx = 0; while (xx < reset.size) { if (reset.buf[xx] != trans.buf[xx]) { break; } xx++; }#endif system("pause"); return 0;}
如果觉得此可行,请参考代码自行修改。
See Again~
之前
真爱无价,欢迎打赏~
- 【步兵 经验篇】图片加密之我见
- 【步兵 经验篇】 链式编程
- 【步兵 经验篇】新手引导
- 【步兵 经验篇】one step
- 【步兵 经验篇】关于脚本语言的优势
- 【步兵 经验篇】状态机(续)
- 【步兵 经验篇】工厂模式 in Game
- 【步兵 经验篇】代码中的文件夹
- 【步兵 经验篇】组件模式的特点
- java加密之我见
- 【步兵 经验篇】 快速开发之我来带你飞
- C# SHA512 加密之我见
- 【步兵 经验篇】游戏常用的三个处理
- 【步兵 cocos2dx】图片管理 之 什么时候打pvr.ccz
- 【步兵 cocos2dx】加密和混淆
- 【步兵 shader】基础篇
- 【步兵 shader】封装篇
- 数据仓库之我见(篇)
- 【java】java.util.ArrayList类
- WebService - SOAP协议
- APP开发实战6-APP的切图工作
- Linux(Ubuntu)下如何安装JDK
- 字符串包含
- 【步兵 经验篇】图片加密之我见
- Python与shell的3种交互方式介绍
- 二十岁出头的时候上,你一无所有,你拥有一切
- 实现打招呼的功能
- Linux bash 扩展
- 网络编程知识(3)--Http Client
- iOS面试题集锦
- 即时通讯技术- 推送技术协议方案
- 【Linux 学习笔记】戏说Git (上)