應客戶需求,將App中的某個模塊功能做成SDK供其使用
注:以下代碼為:2018.7.28修改(之前自己寫的那個編譯不過),修改后基本都是復制而來,<來自這里>,經實踐驗證,所以更新來我之前的方法
需求分析:
- 1、介于該模塊沒有UI頁面,所以將其打包成.jar文件;
- 2、若需要UI界面,可使用.arr文件<下文不提及.arr打包和使用>;
- 3、若包含有.so文件引用,可以使用.arr,也可以打包.jar另附.so 文件
開始
一、注意
- 打包.jar文件時引用第三方框架要下載導入其.jar文件;
- 能不引入第三方的盡量使用原生Api來實現;
二、新建Module或者使用已存在的;
- 在mudule中的build.gradle設置
apply plugin: 'com.android.library'
- 在android和dependencies同級添加task makeJar如下:
task makeJar(type: Jar, dependsOn: ['compileReleaseJavaWithJavac']) {
appendix = 'sdk'
baseName = 'superPay'
version = '1.0.0'
classifier = 'release'
extension = 'jar'//后綴名
//需打包的資源所在的路徑集
def srcClassDir = ["build/intermediates/classes/release"]
//初始化資源路徑集
from srcClassDir
//去除路徑集下部分的資源
exclude 'com/along/pay/BuildConfig.class'
exclude '**/R.class'
exclude '**/R\$*.class'
//只導入資源路徑集下的部分資源
include 'com/along/pay/**/*.class'
}
task proguardJar(dependsOn: ['makeJar'], type: proguard.gradle.ProGuardTask) {
//Android 默認的 proguard 文件
configuration android.getDefaultProguardFile('proguard-android.txt')
//manifest 注冊的組件對應的 proguard 文件
configuration 'proguard-rules.pro'
String inJar = makeJar.archivePath.getAbsolutePath()
//輸入 jar
injars inJar
//輸出 jar
outjars inJar.substring(0, inJar.lastIndexOf(File.separator)) + "/proguard-${makeJar.archiveName}"
//設置不刪除未引用的資源(類,方法等)
dontshrink
Plugin plugin = getPlugins().hasPlugin("AppPlugin") ?
getPlugins().findPlugin("AppPlugin") :
getPlugins().findPlugin("LibraryPlugin")
if (plugin != null) {
List<String> runtimeJarList
if (plugin.getMetaClass().getMetaMethod("getRuntimeJarList")) {
runtimeJarList = plugin.getRuntimeJarList()
} else if (android.getMetaClass().getMetaMethod("getBootClasspath")) {
runtimeJarList = android.getBootClasspath()
} else {
runtimeJarList = plugin.getBootClasspath()
}
for (String runtimeJar : runtimeJarList) {
//給 proguard 添加 runtime
libraryjars(runtimeJar)
}
}
}
完成以上代碼之后,
三、配置混淆代碼
1、配置混淆文件:
當前module的混淆文件proguard-rules.pro,當然還是要按需求自己增刪改查一下的
-ignorewarnings # 忽略警告,避免打包時某些警告出現
-optimizationpasses 5 # 指定代碼的壓縮級別
-dontusemixedcaseclassnames # 是否使用大小寫混合
-dontskipnonpubliclibraryclasses # 是否混淆第三方jar
-dontpreverify # 混淆時是否做預校驗
-verbose # 混淆時是否記錄日志
-optimizations !code/simplification/arithmetic,!field/*,!class/merging/* # 混淆時所采用的算法
-dump class_files.txt #記錄生成的日志數據,gradle build時在本項目根目錄輸出 apk 包內所有 class 的內部結構
-printmapping mapping.txt #混淆前后的映射 (我是在使用了這條語句后,就在build\outputs\mapping\你的項目名\release\文件夾下,出現了dump.txt, mapping.txt, seeds.txt, usage.txt)
-printseeds seeds.txt #未混淆的類和成員
-printusage unused.txt (dump.txt) #列出從 apk 中刪除的代碼
-dontwarn android.support.v4.** #缺省proguard 會檢查每一個引用是否正確,但是第三方庫里面往往有些不會用到的類,沒有正確引用。如果不配置的話,系統就會報錯。
-dontwarn android.os.**
-keep class android.support.v4.** { *; } # 保持哪些類不被混淆
-keep class com.baidu.** { *; }
-keep class android.os.**{*;}
-keep interface android.support.v4.app.** { *; }
-keep public class * extends android.support.v4.**
-keep public class * extends android.app.Fragment
-keep public class * extends android.app.Activity
-keep public class * extends android.app.Application
-keep public class * extends android.app.Service
-keep public class * extends android.content.BroadcastReceiver
-keep public class * extends android.content.ContentProvider
-keep public class * extends android.support.v4.widget
-keepclasseswithmembernames class * { # 保持 native 方法不被混淆
native <methods>;
}
-keepclasseswithmembers class * { # 保持自定義控件類不被混淆
public <init>(android.content.Context, android.util.AttributeSet);
}
-keepclasseswithmembers class * { # 保持自定義控件類不被混淆
public <init>(android.content.Context, android.util.AttributeSet, int);
}
-keepclassmembers class * extends android.app.Activity { #保持類成員
public void *(android.view.View);
}
-keepclassmembers enum * { # 保持枚舉 enum 類不被混淆
public static **[] values();
public static ** valueOf(java.lang.String);
}
-keep class * implements android.os.Parcelable { # 保持 Parcelable 不被混淆
public static final android.os.Parcelable$Creator *;
}
#上面是公共可用的,這個是我自己實際測試用的
-keep public class com.along.pay.Functions{ # 保持 提供外部調用的類和方法不被混淆
public <fields>;
public <methods>;
}
四、編譯打包
在Android Studio的Terminal中輸入 gradlew makeJar回車;
直到顯示"BUILD SUCCESSFUL"即為成功!否則視錯誤信息修改
但是此時的jar包是還沒有混淆的
打混淆包建議使用點擊右側Gradle中pay/other下的proguardJar,就好了;如下:
20180728235238.png
接著就可以在其他工程直接使用了,如果有.so文件需另外復制過去,否則編譯之后找不到.so文件中的引用
Thank you all,中間查閱了許多的資料,填了不少坑,遂以此為記;歡迎補充和討論,也希望能幫到有需要的人!