接續上篇[NDK開發基礎②文件加密解密與分割合并](http://www.jianshu.com/p/4333d7ddf615)
### 前情提要
IO操作 , 一直在開發中占據很大比重 , 在Java中不管是網絡操作還是文件操作 , 都作為IO流來處理 , 都依靠`InputStream`和`OutputStream`這兩個輸入輸出流 。在上篇中 , 使用了C語言的IO流 , 進行了文件的加密與解密,分割與合并 。其要點是,加密解密使用了`^`運算 ,分割文件則使用了,文件大小與文件個數的`%`運算 。
### 為什么需要增量更新?
當我們開發完一個項目之后,上線維護 , 增加新功能 , 添加第三方庫 , APK大小從4 - 5M , 增長到10+M , 用戶由原來的幾十秒下載 , 到現在幾分鐘以上的下載 , 網絡情況不好的時候 , 或許就是十分鐘不等。每次全量下載 , 無論從體驗還是流量上 , 都是不友好的 , 所有增量更新還是有必要的 (小公司好像沒幾個用 , 一般大公司在用,如QQ空間)。
### 增量原理(圖)

bsdiff 進行文件內容比較,并且使用了bzip2進行文件壓縮 , 所有得出的差分包可能比理論值要小 , 進一步可以減少用戶流量 。增量更新 , 較為關鍵的部分就是生成差分包 , 將新舊APK進行比較 , 生成一個新的文件 。
### 生產資源及工具
bsdiff --- [bsdiff (win環境) ](http://sites.inka.de/tesla/download/bsdiff4.3-win32-src.zip) 生成差分包及合并差分包庫 , 源碼內已包含bzip2
bzip2 --- [bzip2](http://www.bzip.org/downloads.html) bsdiff 依賴
服務器 --- [Tomcat 7.0 ](http://tomcat.apache.org/tomcat-7.0-doc/index.html) (模擬網絡環境)放置差分包 , 供APP下載
開發工具 --- Eclipse NDK開發 , 目前建議使用Eclipse開發
開發工具 --- VS 因為服務器搭建在windows平臺 , 所以使用VS開發JNI,在JNI系列中有使用 , 不了解的可以前去了解 [JNI開發系列①JNI概念及開發流程](http://www.jianshu.com/p/68bca86a84ce)
### 編寫bsdiff JNI
當我們拿到一個C語言庫 , 首先我該怎么做 ?
> 首其一 , 了解其用法 , 找main函數
下載完bsdiff之后 , 我們看到如下目錄:

看這么多文件 , 還有一些亂七八糟的不知道什么的文件 , 那么我們只關注 , 我們知道的文件 , 將C/C++源文件以及.h頭文件,找出來 ,放到一個干凈的文件夾中 。
創建一個VS項目 , 將源文件都放入到VS項目目錄下 , 添加現有項目 , 將源碼與項目關聯 , 參考:[JNI開發系列①JNI概念及開發流程](http://www.jianshu.com/p/68bca86a84ce)
__找main函數__

在`bsdiff.cpp`文件找到帶參數的`main`函數 , 并且有一個關于用法的線索 , 那就是:
```c
if(argc!=4) errx(1,"usage: %s oldfile newfile patchfile\n",argv[0]);
```
我們可以根據這句話來推測 , 需要四個參數 , 并且三個參數必須是我呢見路徑 , 還有一個不知曉 , 暫且不管它 。
> 其二 , 創建JNI方法 , 修改main函數
既然知道了需要傳入的參數 , 那么就可以創建一個Java工程 , 編寫JNI方法了。
```java
/**
* 生成差分包
* @param oldFilePath 老版本文件路徑
* @param newFilePath 新版本文件路徑
* @param patchFilePath 生成差分包文件路徑
*/
public static native void diff(String oldFilePath , String newFilePath , String patchFilePath) ;
```
使用`javah`生成頭文件 , 引入到bsdiff.cpp文件中,編寫調用的C函數 , 并修改`main`函數名稱,`main`函數作為入口函數 , 在JNI中就適用了 , 所有將`main`函數名稱改一下 , 在JNI的C函數中調用即可 。
```c
/*文件差分*/
JNIEXPORT void JNICALL Java_com_zeno_bsdiff_BsdiffUtils_diff
(JNIEnv *env, jclass clazz, jstring joldFilePath, jstring jnewFilePath, jstring jpatchFilePath) {
char* oldFilePath = (char*)env->GetStringUTFChars(joldFilePath, 0);
char* newFilePath = (char*)env->GetStringUTFChars(jnewFilePath, 0);
char* patchFilePath = (char*)env->GetStringUTFChars(jpatchFilePath, 0);
int argc = 4;
char* argv[4];
// 無效參數 , 可以為任意字符
argv[0] = "bsdiff";
// 老版APK文件路徑
argv[1] = oldFilePath;
// 新版APK文件路徑
argv[2] = newFilePath;
// 生成的差分包文件路徑
argv[3] = patchFilePath;
// 調用bsdiff
bsdiff_main(argc, argv);
}
```
一切修改完畢 , 別忘了在項目屬性配置類型,改為生成`.dll`動態庫 。編譯生成 , 這時候會出現很多錯誤 , 舉幾個我遇到的 。
一 , 使用了非安全的函數 , 在文件中聲明`#define _CRT_SECURE_NO_WARNINGS`即可 。
二 , 在VS中通不過安全語法檢查 , 在VS中進行如下設置:

還需要在文件中添加`#define _CRT_NONSTDC_NO_DEPRECATE`預處理指令。也可以在項目屬性中配置:

三 , 變量未初始化
```c
u_char *old = nullptr,*_new = nullptr;
off_t oldsize,newsize;
off_t *I,*V = nullptr;
```
可能會是這幾個 , 將其賦值為`nullptr`就可以了。 如果還有其他錯誤 , 可執行分析 , google , 也歡迎評論留言 , 多多交流 。
> 其三 , 生成.dll動態庫 , 并使用
將生成的`.dll`動態庫 , 賦值到服務器項目的目錄下 , 或是Java項目也可以 。
差分包工具類:
```java
public class BsdiffUtils {
/**
* 生成差分包
* @param oldFilePath 老版本文件路徑
* @param newFilePath 新版本文件路徑
* @param patchFilePath 生成差分包文件路徑
*/
public static native void diff(String oldFilePath , String newFilePath , String patchFilePath) ;
static {
System.loadLibrary("BsdiffJNI");
}
}
```
使用差分包工具類:
```java
public class BsdiffApk {
public static void main(String args[]) {
/**
* 文件差分
*/
BsdiffUtils.diff(Constract.OLD_FILE_PATH, Constract.NEW_FILE_PATH, Constract.PATCH_FILE_PATH);
}
}
```
常量類:
```java
public class Constract {
public static final String OLD_FILE_PATH = "E:/javaee_workspace/AppUpdateServer/apk/app-old.apk" ;
public static final String NEW_FILE_PATH = "E:/javaee_workspace/AppUpdateServer/apk/app-new.apk" ;
public static final String PATCH_FILE_PATH = "E:/javaee_workspace/AppUpdateServer/apk/App_patch.patch" ;
}
```
生成差分包:

### 結語
使用第三方庫的時候 , 自己編寫的代碼比較少 , 主要是學會怎樣使用第三方的庫,以及編譯生成動態庫的時候 , 需要查錯找錯并修改 , 下篇 , 我們將實現 , 在客戶端合并差分包 。
- 簡介
- C語言基礎及指針①語法基礎
- C語言基礎及指針②之指針內存分析
- C語言基礎及指針③函數與二級指針
- C語言基礎及指針④函數指針
- C語言基礎及指針⑤動態內存分配
- C語言基礎及指針⑥字符操作
- C語言基礎及指針⑦結構體與指針
- C語言基礎及指針⑧文件IO
- C語言基礎及指針⑨聯合體與枚舉
- C語言基礎及指針⑩預編譯及jni.h分析
- JNI開發系列①JNI概念及開發流程
- JNI開發系列②.h頭文件分析
- JNI開發系列③C語言調用Java字段與方法
- JNI開發系列④C語言調用構造方法
- JNI開發系列⑤對象引用的處理
- NDK開發基礎①使用Android Studio編寫NDK
- NDK開發基礎②文件加密解密與分割合并
- NDK開發基礎③增量更新之服務器端生成差分包
- C++基礎①命名空間結構體和引用