zoukankan      html  css  js  c++  java
  • 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 oolsproguardinproguardgui.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,填写输出文件,这里额外添加一个一个javajre7lib t.jar(具体以自己电脑环境为准)

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

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

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

  • 相关阅读:
    poj2478
    poj2376
    poj2192
    poj1062
    [HDOJ2639]Bone Collector II(第k优01背包)
    [HDOJ3466]Proud Merchants(贪心+01背包)
    [HDOJ5510]Bazinga(并查集)
    [POJ3264]Balanced Lineup(线段树,区间最值差)
    [HDOJ4325]Flowers(树状数组 离散化)
    [HDOJ5521]Meeting(最短路)
  • 原文地址:https://www.cnblogs.com/hsji/p/5310175.html
Copyright © 2011-2022 走看看