AndroidStudio封装jar并混淆

    公司和A公司有技术合作,需要把接口封装成sdk供A公司调用,我被安排来做这个事情。这里记录一下封装jar并混淆的过程中遇到的一些问题,本文使用的IDE是AndroidStudio。

    1.新建项目之后看到proguard-rules.pro中并没有混淆java文件的代码,于是百度-->>复制-->>粘贴-->>运行,可是反编译apk之后发现根本没有混淆代码。这是真么回事呢?proguard-rules.pro文件写错了?没有配置到gradle文件里?查了很多资料,再加上多次的实验,我终于找到问题的所在。之前一直看到gradle文件配置混淆文件的上面还有一行——minifyEnabled false 从来没有深究过有什么意义(以后还是要多研究一下这些不起眼的小细节,没准那一天就把你拌趴下了),原来这个标志就是代码混淆的开关,默认是关闭的,所以之前虽然配置了混淆文件,但是并没有执行代码混淆。把这个标志位置为true,再运行就发现混淆起作用了。结论:进行代码混淆时,要先把开关(minifyEnabled )打开!

    2.如何封装jar包呢?只在AndroidStudio上创建过工程,怎么才能达成jar包呢?网上介绍了一些方法,让我在gradle文件里面新建任务大概是这个样子

task makeJar(type: Copy) {
    delete 'build/libs/mysdk.jar'
    from('build/intermediates/bundles/release/')
    into('build/libs/')
    include('classes.jar')
    rename ('classes.jar', 'mysdk.jar')
}

makeJar.dependsOn(build)

    一开始我也是这么做的,还真的在build/libs目录下面生成了一个mysdk.jar文件。后来仔细看了这一段代码,发现这段代码先是执行了build,然后把build/intermediates/bundles/release目录下成的classes.jar文件改了一个名字而已,实在是太鸡肋了。你自己运行工程,直接把build/intermediates/bundles/release目录下的classes.jar拷贝出来就行了,没必要这么麻烦。结论:build输出文件中已包含jar包,不必再想办法生成jar,jar包位置build/intermediates/bundles/release/classes.jar

    3.封装什么东西?由于合作的事情还在谈,我并不知道最终的交付的sdk有什么要求,所以我做了两个方面的尝试,第一:仅封装接口;第二:提供接口和UI。先说第一种,我使用的方法和一般添加Module Library并不相同,我只是添加了一个普通的java的jar,Module中只有代码和gralde没有Manifest,没有res。下面是我的gradle文件,帮助理解

apply plugin: 'java'
dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile files('libs/android.jar')
    compile 'com.google.code.gson:gson:2.5'
    compile 'com.squareup.okhttp:okhttp:2.5.0'
    compile 'com.squareup.okio:okio:1.6.0'
}

 

    这个地方有几点需要注意的,android.jar去哪里找?我注意到工程的最下面有一个External Library目录,这个目录关联了工程的依赖文件。右键之后选择file path就可以看到jar的位置了,然后去这个目录下边拷贝出来就行了。像这样

我尝试讲gson、okhttp的jar包也通过compile files引用,但是在主工程里面提示类没有定义,只有通过这种方式,才不会报错。结论:我封装的jar和界面没有关系,所以我用了一种更干净的方法,来生成jar。

    4.运行程序,因为我没有使用“apply plugin: 'com.android.library'”所以目录结构跟上面又不太一样,我的目录结构是这样的,module名为meddo

可以看到build/libs下面已经生成好了一个jar。我把meddo.jar文件拷贝出来开始进行混淆。

    5.jar混淆,这里使用的是sdk中包含的工具,具体位置sdk\tools\proguard\bin\proguardgui.bat。使用这个工具之前要做好准备工作:a.准备好要混淆的jar(上面的meddo.jar);b.准备好jar中使用的第三方jar(我这里使用到了android.jar,okhttp-2.5.0.jar,okio-1.6.0.jar,gson-2.5.jar);c.准备好混淆配置文件

    介绍一下我这边项目的具体情况:I)我写了一个外观类MeddoInterface,并在这个类中开放出两个方法init(Context)和UserLogin(HashMap<String,String>,ResultCallBack<UserLoginResponse>)  II)封装了访问网络的操作,使用的是okhttp,中间用到了Gson,反射来解析数据

下面是混淆文件,有些冗余的东西,不要见怪

-optimizationpasses 5
-dontusemixedcaseclassnames
-dontskipnonpubliclibraryclasses
-dontpreverify
-dontshrink#混淆jar的时候一定要配置,不然会把没有用到的代码全部remove   我本来封装一个jar就是给别人调用的,全部删掉就没有东西了
-verbose

-keepattributes Signature #过滤泛型 用到发射,泛型等要添加这个
-keepattributes *Annotation* 

-optimizations !code/simplification/arithmetic,!field/*,!class/merging/*

#如果你使用了ProGuard来导入第三方jar这个地方就不用配置了
#-libraryjars ../meddo/libs/android.jar
#-libraryjars ../meddo/libs/gson-2.5.jar
#-libraryjars ../meddo/libs/okhttp-2.5.0.jar
#-libraryjars ../meddo/libs/okio-1.6.0.jar

-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 com.android.vending.licensing.ILicensingService

-keepclasseswithmembernames class * {
    native <methods>;
}

-keepclasseswithmembernames class * {
    public <init>(android.content.Context, android.util.AttributeSet);
}

-keepclasseswithmembernames class * {
    public <init>(android.content.Context, android.util.AttributeSet, int);
}

-keepclassmembers enum * {
    public static **[] values();
    public static ** valueOf(java.lang.String);
}

-keep class * implements android.os.Parcelable {
  public static final android.os.Parcelable$Creator *;
}
#公开两个方法供使用者调用
-keep public class com.hsji.makejar.MeddoInterface{
 public static void UserLogin(java.util.HashMap, com.hsji.makejar.network.ResultCallback);
 public static void init(android.content.Context);
}
#解析数据是用的的bean  完全不混淆,不然解析json数据时什么都找不到
-keep public class com.hsji.makejar.UserLoginResponse{*;}
#不要混淆ResultCallback的public方法
-keep public class com.hsji.makejar.network.ResultCallback{
 public <methods>;
}

#okhttp
-dontwarn com.squareup.okhttp.**

-keep class com.squareup.okhttp.** { *;}

-dontwarn okio.**

-keep class okio.** {*;}

#json
-dontwarn com.google.gson.**

-keep class com.google.gson.**{*;}

 ok,现在我们原料都有了,开始混淆吧!

 step1:

 导入ProGuard配置文件

step2:导入需要混淆的jar和第三方jar,填写输出文件,这里额外添加一个一个java\jre7\lib\rt.jar(具体以自己电脑环境为准)

step3:进入Process标签页,点击右下角的Process!按钮,开始混淆,很快就混淆好了

 

注:以上为个人的方法,不喜勿喷

      关于proguard文件的配置可以参考这篇文章《android 混淆文件proguard.cfg详解》

posted @ 2016-03-23 11:01  听懂·动听  阅读(5480)  评论(0编辑  收藏  举报