应客户需求,将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,就好了;如下:
接着就可以在其他工程直接使用了,如果有.so文件需另外复制过去,否则编译之后找不到.so文件中的引用
Thank you all,中间查阅了许多的资料,填了不少坑,遂以此为记;欢迎补充和讨论,也希望能帮到有需要的人!