Android obfuscation from entry to mastery

Android obfuscation from entry to mastery

Introduction

As an Android developer, if you do not want to open source your application, you need to obfuscate the code before publishing it, so that even if the code is decompiled, it is difficult to read. Although the concept of obfuscation is easy, many beginners just search for some established obfuscation rules on the Internet and paste them into their own projects without a deep understanding of obfuscation. The purpose of this article is to enable a beginner to independently write obfuscation rules suitable for their own code without any help after reading it.

Say it in advance

Here we will use Android Studio to explain how to perform obfuscation. Android Studio integrates the Java language ProGuard as a compression, optimization and obfuscation tool. It is very simple to use with the Gradle build tool. You only need to set minifyEnabled to true in the gradle file in the project application directory. Then we can add our obfuscation rules to the proguard-rules.pro file.

  1. android {
  2. ...
  3. buildTypes {
  4. release {
  5. minifyEnabled true  
  6. proguardFiles getDefaultProguardFile( 'proguard-android.txt' ), 'proguard-rules.pro'  
  7. }
  8. }
  9. }

The above sample code indicates that the release version is obfuscated. Next, we will first introduce the three major functions of ProGuard and briefly explain their commonly used commands.

ProGuard role

Shrinking: Enabled by default, used to reduce the size of the application, remove unused classes and members, and will be executed again after the optimization action is executed (because some unused classes and members may be exposed again after optimization).

  1. -dontshrink turns off compression

Optimization: Enabled by default, performs optimization at the bytecode level to make the application run faster.

  1. -dontoptimize turns off optimization
  2. -optimizationpasses n indicates the number of times proguard iterates to optimize the code. Android is generally 5

Obfuscation: Enabled by default, increases the difficulty of decompilation, and classes and class members will be randomly named unless protected with keep.

  1. -dontobfuscate Disable obfuscation

After obfuscation, a mapping.txt file will be generated in the project directory app/build/outputs/mapping/release by default. This is the obfuscation rule. We can use this file to reverse the obfuscated code back to the original code, so this file is very important and should be protected. In principle, the more chaotic and irregular the code is after obfuscation, the better. However, there are some places where we need to avoid obfuscation, otherwise the program will fail to run. So here is what we are going to teach you, how to avoid obfuscation of some of your code to prevent errors.

Basic Rules

Let's first look at the following two commonly used commands. Many people may be confused about the difference between the two.

  1. -keep class cn.hadcn.test.**
  2. -keep class cn.hadcn.test.*

One star means that only the class name under the package is kept, and the class names under the sub-packages will still be obfuscated; two stars mean that the class names under the package and the sub-packages are kept. After keeping the class with the above method, you will find that although the class name is not obfuscated, the specific method and variable names in it are still changed. At this time, if you want to keep the class name and keep the content inside from being obfuscated, we need the following method

  1. -keep class cn.hadcn.test.* {*;}

On this basis, we can also use basic Java rules to protect specific classes from being confused, such as using extend, implement, and other Java rules. The following example prevents all classes that inherit Activity from being confused:

  1. -keep public class * extends android.app.Activity

If we want to keep the inner classes in a class from being obfuscated, we need to use the $ symbol. The following example means keeping all public contents in the ScriptFragment inner class JavaScriptInterface from being obfuscated.

  1. -keepclassmembers class cc.ninty.chat.ui.fragment.ScriptFragment$JavaScriptInterface {
  2. public *;
  3. }

Furthermore, if you do not want to keep all the contents of a class from being obfuscated, but only want to protect specific contents under the class, you can use

  1. <init>; //Match all constructors
  2. <fields>; //match all fields
  3. <methods>; //Match all methods

You can also add private, public, native, etc. before <fields> or <methods> to further specify the content that is not to be obfuscated, such as

  1. -keep class cn.hadcn.test.One {
  2. public <methods>;
  3. }

This means that all public methods under the One class will not be obfuscated. Of course, you can also add parameters. For example, the following means that the constructor with JSONObject as input will not be obfuscated.

  1. -keep class cn.hadcn.test.One {
  2. public <init>(org.json.JSONObject);
  3. }

Sometimes you still think, I don't need to keep the class name, I just need to keep the specific methods under the class from being confused, then you can't use the keep method, the keep method will keep the class name, and you need to use keepclassmembers, so the class name will not be kept. In order to facilitate the understanding of these rules, the official website provides the following table

reserve Prevent being removed or renamed Prevent renaming
Classes and class members -keep -keepnames
Class members only -keepclassmembers -keepclassmembernames
If it has a member, keep the class and class members -keepclasseswithmembers -keepclasseswithmembernames

Removal refers to whether it will be deleted during shrinking. The above content is the key to master in the obfuscation rules. After understanding, you should be able to understand almost all obfuscation rule files. Combined with the following points,

Precautions

1. The jni method cannot be confused, because this method needs to be consistent with the native method;

  1. -keepclasseswithmembernames class * { # Keep native methods from being confused
  2. native <methods>;
  3. }

2. The classes used for reflection are not confused (otherwise reflection may cause problems);

3. Classes in AndroidMainfest are not obfuscated, so the four major components, Application subclasses, and all classes under the Framework layer are not obfuscated by default. Customized Views are also not obfuscated by default; so there is no need to add many rules posted online to exclude custom Views or obfuscation of the four major components in Android Studio;

4. When interacting with the server, when using frameworks such as GSON and fastjson to parse server data, the written JSON object class is not confused, otherwise the JSON cannot be parsed into the corresponding object;

5. When using third-party open source libraries or referencing other third-party SDK packages, if there are special requirements, you also need to add corresponding obfuscation rules to the obfuscation file;

6. If you use JS calls to WebView, you also need to ensure that the written interface methods are not confused. The reason is the same as in item ***;

7. Parcelable subclasses and Creator static member variables should not be confused, otherwise an Android.os.BadParcelableException will be generated;

  1. -keep class * implements Android.os.Parcelable { # Keep Parcelable from being confused
  2. public   static final Android.os.Parcelable$Creator *;
  3. }

8. When using the enum type, be careful to avoid confusion between the following two methods. Due to the particularity of the enum class, the following two methods will be called by reflection, see the second rule.

  1. -keepclassmembers enum * {
  2. public   static **[] values ​​();
  3. public   static ** valueOf(java.lang.String);
  4. }

Written in ***

When releasing an app, in addition to setting minifyEnabled to true, you should also set zipAlignEnabled to true. Google Play requires that developers upload apps that are zipAligned. ZipAlign can align the resources in the installation package to 4 bytes, which can reduce the memory consumption of the app at runtime.

<<:  Use Retrofit+RxJava+MVP to create a Material Design style APP

>>:  Android touch events (notes)

Recommend

Guide to building a TOB operation customer acquisition system!

“The two core elements of customer acquisition ar...

Minimalist landscape painting course while traveling

Traveling and drawing minimalist landscape painti...

Why use blogs to promote your website? How to promote your products online?

Blog promotion is a form of Internet promotion, a...

Hammer cuts prices, is sentiment bankrupt?

[[121743]] On the morning of October 27, the WeCh...

How to improve user loyalty to products?

This is a "red ocean era". Due to the r...

The basic logic of Internet advertising

No ToC product can survive without Internet adver...

Should companies develop WeChat mini-programs?

Should companies develop WeChat mini-programs? Do...

What will be the hottest thing in 2016?

[[161131]] 2016 has arrived. I wonder if you have...

How to master social e-commerce and acquire customers at low cost?

This article is the first in a series of detailed...

How to write an excellent marketing promotion plan?

Making plans is one of the daily tasks of markete...