Android马甲包开发之路

何为马甲包?

马甲包是指与原APP包除了包名,签名、包名称图标等给用户加以区分的东西不一样之外,其他功能基本不变的APP包。



最近公司需要一套代码,生成多个马甲包的需求,为了方便代码维护和打包方便,主要通过gradle工具配置不同包名,不同签名,不同资源名,不同马甲包部分差异化,不同兼容包名不同的差异化兼容需求(如:微信分享跟包名有关)。用Terminal命令或者Tasks一键生成多个包。

下面就描述下只需配置,就可以一键生成多个马甲包和主包(当成一个马甲包就行)?


    1.签名文件路径配置(只有一个签名文件,不同马甲包对应不同别名就行)
    2.主module的build.gradle中一些相关配置
    3.AndroidManifest.xml中的一些相关配置(${}的使用)
    4.获取MetaData值和getPackageName()获取包名
    5.如何打包


1.签名文件路径配置(只有一个签名文件,不同马甲包对应不同别名就行)

这一步主要是每个人电脑签名文件位置不一样,我把地址配置放在这里。放在其他地方也行。
signingStoreFilePath = "E:/raythinks/keystore.jks" //font color=red>签名文件目录

ext {
    signingStoreFilePath = "E:/raythinks/keystore.jks" //font color=red>签名文件目录
}
buildscript {
    repositories {
        jcenter()
    }
    dependencies {
        classpath 'com.android.tools.build:gradle:2.3.3'
        classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
    }
}

allprojects {
    repositories {
        jcenter()
    }
}

task clean(type: Delete) {
    delete rootProject.buildDir
}



2.主module的build.gradle中一些相关配置

(1)applicationId 马甲包包名配置。如: applicationId "com.raythins.herri.xinshou"//(如:新手版)
(2)signingConfigs 签名文件配置 。每个马甲包签名文件或者别名不一样。


签名文件配置.png

(3)productFlavors 配置要生成的马甲包。 如:majia_xinshou_vivo 新手版;majia_shop_vivo 商城版
(4)manifestPlaceholders 资源配置(如:马甲包app的logo、名称、微信appkey等。)


manifestPlaceholders资源配置(.png

(5) signingConfig 配置马甲包编译时使用的签名。如: signingConfig signingConfigs.shop 使用商城版签名

详情见gradle.gradle代码
apply plugin: 'com.android.application'
android {
    compileSdkVersion 23
    buildToolsVersion '25.0.2'
    defaultConfig {
        applicationId "com.raythins.herri"//默认包名
        minSdkVersion 15   //最小版本号
        targetSdkVersion 23  
        versionCode 110//版本code   
        versionName "1.1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
        ndk { abiFilters   'armeabi-v7a' ,'armeabi'}//指定ndk,目前市场上手机基本覆盖兼容这两种

        multiDexEnabled true//MultiDex的配置

        manifestPlaceholders = [
                JPUSH_PKGNAME    : "com.raythins.herri",//极光包名
                JPUSH_APPKEY  :  "sdfewds234324343243243243sdfdsd",//极光appkey
                JPUSH_CHANNEL    : "developer-default", //暂时填写默认值即可.
                BAIDU_APPKEY  :  "SDFDSFDFDSFRT72LSDFDSFDFDS",//
                TENCENT_ID  :  "1323123134341" , //腾讯id(应用宝)
                TENCENT_APPKEY  :  "SDFDSFDS9089SDF" , //腾讯APPKEY
                LAUNCHER_ICON  :  "@drawable/ic_launcher" ,  //logo图片路径
                WCHAT_APPID : "wxsdf4eds323r32432432432",// 微信appid
                WCHAT_TEMPLETE_ID : "sdfdsfsdfsdf233243243243243243243243243",//微信SDK  订阅id
                WCHAT_SECRET : "dfdfdsffdsfdsfdsfdsfds3432432432432432432432",//微信SDK secret
        ]
    }
    //debug和release版本的签名配置
    signingConfigs {
        xinshou{//新手版签名文件信息
            storeFile file(rootProject.ext.signingStoreFilePath)
            storePassword "123456"
            keyAlias "别名1"
            keyPassword "123456"
            v1SigningEnabled true
            v2SigningEnabled true
        }
        shop{//商城版签名文件信息
            storeFile file(rootProject.ext.signingStoreFilePath)
            storePassword "123456"
            keyAlias "别名2"
            keyPassword "123456"
            v1SigningEnabled true
            v2SigningEnabled true
        }

    }
    buildTypes {
        release {
            minifyEnabled true
            //Zipalign优化
            zipAlignEnabled true
            // 移除无用的resource文件
            shrinkResources false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }

    productFlavors {
        majia_xinshou_vivo{//新手版)
            applicationId "com.raythins.herri.xinshou"//(如:新手版)
            manifestPlaceholders = [
                 JPUSH_PKGNAME    : "com.raythins.herri.xinshou",//极光包名
                JPUSH_APPKEY  :  "sdfewds234324343243243243sdfdsd",//极光appkey
                JPUSH_CHANNEL    : "developer-default", //暂时填写默认值即可.
                BAIDU_APPKEY  :  "SDFDSFDFDSFRT72LSDFDSFDFDS",//
                TENCENT_ID  :  "1323123134341" , //腾讯id(应用宝)
                TENCENT_APPKEY  :  "SDFDSFDS9089SDF" , //腾讯APPKEY
                LAUNCHER_ICON  :  "@drawable/ic_launcher_xinshou" ,  //logo图片路径
                APP_NAME :  "Demo新手版" ,  //app名称
                WCHAT_APPID : "wxsdf4eds323r32432432432",// 微信appid
                WCHAT_TEMPLETE_ID : "sdfdsfsdfsdf233243243243243243243243243",//微信SDK  订阅id
                WCHAT_SECRET : "dfdfdsffdsfdsfdsfdsfds3432432432432432432432",//微信SDK secret
            ]
            signingConfig signingConfigs.xinshou//签名信息
        }
     majia_shop_vivo{//商城版)
            applicationId "com.raythins.herri.shop"//(如:商城版)
            manifestPlaceholders = [
                 JPUSH_PKGNAME    : "com.raythins.herri.xinshou",//极光包名
                JPUSH_APPKEY  :  "sdfewds234324343243243243sdfdsd",//极光appkey
                JPUSH_CHANNEL    : "developer-default", //暂时填写默认值即可.
                BAIDU_APPKEY  :  "SDFDSFDFDSFRT72LSDFDSFDFDS",//
                TENCENT_ID  :  "1323123134341" , //腾讯id(应用宝)
                TENCENT_APPKEY  :  "SDFDSFDS9089SDF" , //腾讯APPKEY
                LAUNCHER_ICON  :  "@drawable/ic_launcher_shop" ,  //logo图片路径
                APP_NAME :  "Demo商城版",  //app名称
                WCHAT_APPID : "wxsdf4eds323r32432432432",// 微信appid
                WCHAT_TEMPLETE_ID : "sdfdsfsdfsdf233243243243243243243243243",//微信SDK  订阅id
                WCHAT_SECRET : "dfdfdsffdsfdsfdsfdsfds3432432432432432432432",//微信SDK secret
            ]
            signingConfig signingConfigs.shop//签名信息
     }

    }

    sourceSets {
        main {
            jniLibs.srcDirs = ['libs']
        }
    }
}
repositories {
    mavenCentral()
    repositories {
        flatDir {
            dirs 'libs'
        }
    }
}
dependencies {
    compile fileTree(include: ['*.jar'], dir: 'libs')
    androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
        exclude group: 'com.android.support', module: 'support-annotations'
    })

    //自带的类库v4以及v7
    compile 'com.android.support:appcompat-v7:23.0.1'
}
3.AndroidManifest.xml中的一些相关配置(${}的使用)

主要是通过${}将主module的build.gradle中一些相关配置映射到manifest中。打包自动填充进去。如:${applicationId}、${LAUNCHER_ICON}、${APP_NAME}、${BAIDU_APPKEY}等。

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    package="com.raythinks.herri">

    <!-- Required -->
    <permission
        android:name="${applicationId}.permission.JPUSH_MESSAGE"
        android:protectionLevel="signature" />

    <uses-permission android:name="android.permission.CAMERA" />
    <application
        android:name=".base.CustomApplication"
        android:allowBackup="true"
        android:icon="${LAUNCHER_ICON}"
        android:label="${APP_NAME}"
        android:largeHeap="true"
        android:supportsRtl="true"
        android:theme="@style/AppTheme"
        tools:replace="android:allowBackup">
   
       <!-- //百度appkey -->
        <meta-data
            android:name="com.baidu.lbsapi.API_KEY"
            android:value="${BAIDU_APPKEY}" />
     <!-- //微信appid-->
        <meta-data
            android:name="WCHAT_APPID"
            android:value="${WCHAT_APPID}" />
      <!-- //微信secret-->
        <meta-data
            android:name="WCHAT_SECRET"
            android:value="${WCHAT_SECRET}" />
         <!-- //微信secret-->
        <meta-data
            android:name="WCHAT_TEMPLETE_ID"
            android:value="${WCHAT_TEMPLETE_ID}" />
        <!-- //腾讯ID-->
        <meta-data
            android:name="TENCENT_ID"
            android:value="${TENCENT_ID}" />
       <!-- //腾讯appkey-->
        <meta-data
            android:name="TENCENT_APPKEY"
            android:value="${TENCENT_APPKEY}" />
        <!-- 友盟集成 -->
        <meta-data
            android:name="UMENG_APPKEY"
            android:value="586481eb82b63522b30005a9" />
        <meta-data
            android:name="UMENG_CHANNEL"
            android:value="${UMENG_CHANNEL_VALUE}" />
              <!-- 目前这个渠道统计功能的报表还未开放。 -->
        <meta-data
            android:name="JPUSH_CHANNEL"
            android:value="${JPUSH_CHANNEL}" />
        <!-- Required. AppKey copied from Portal -->

        <meta-data
            android:name="JPUSH_APPKEY"
            android:value="${JPUSH_APPKEY}" />
        <!-- ShareSDK -->
        <activity
            android:name="com.mob.tools.MobUIShell"
            android:configChanges="keyboardHidden|orientation|screenSize"
            android:screenOrientation="portrait"
            android:theme="@android:style/Theme.Translucent.NoTitleBar"
            android:windowSoftInputMode="stateHidden|adjustResize">

            <!-- QQ和QQ空间分享 QQ登录的回调必须要配置的 -->
            <intent-filter>
                <data android:scheme="tencent${TENCENT_ID}" />

                <action android:name="android.intent.action.VIEW" />

                <category android:name="android.intent.category.BROWSABLE" />
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>

            <!-- 新浪微博客户端分享回调必须配置 -->
            <intent-filter>
                <action android:name="com.sina.weibo.sdk.action.ACTION_SDK_REQ_ACTIVITY" />

                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </activity>
       
        <!-- 微信分享回调 -->
        <activity
            android:name="${applicationId}.wxapi.WXEntryActivity"
            android:configChanges="keyboardHidden|orientation|screenSize"
            android:exported="true"
            android:screenOrientation="portrait"
            android:theme="@android:style/Theme.Translucent.NoTitleBar" />
        <activity
            android:name=".ui.activity.MainActivity"
            android:configChanges="orientation|keyboardHidden"
            android:launchMode="singleTask"
            android:screenOrientation="portrait" />
      

        <!-- 极光推送 Required SDK 核心功能 -->
        <!-- 可配置android:process参数将PushService放在其他进程中 -->
        <service
            android:name="cn.jpush.android.service.PushService"
            android:enabled="true"
            android:exported="false">
            <intent-filter>
                <action android:name="cn.jpush.android.intent.REGISTER" />
                <action android:name="cn.jpush.android.intent.REPORT" />
                <action android:name="cn.jpush.android.intent.PushService" />
                <action android:name="cn.jpush.android.intent.PUSH_TIME" />
            </intent-filter>
        </service>

        <!-- since 1.8.0 option 可选项。用于同一设备中不同应用的JPush服务相互拉起的功能。 -->
        <!-- 若不启用该功能可删除该组件,将不拉起其他应用也不能被其他应用拉起 -->
        <service
            android:name="cn.jpush.android.service.DaemonService"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="cn.jpush.android.intent.DaemonService" />

                <category android:name="${applicationId}" />
            </intent-filter>
        </service>

        <!-- Required SDK核心功能 -->
        <service
            android:name="cn.jpush.android.service.DownloadService"
            android:enabled="true"
            android:exported="false" />

        <!-- Required SDK核心功能 -->
        <receiver
            android:name="cn.jpush.android.service.PushReceiver"
            android:enabled="true">
            <intent-filter android:priority="1000">
                <action android:name="cn.jpush.android.intent.NOTIFICATION_RECEIVED_PROXY" />

                <category android:name="${applicationId}" />
            </intent-filter>
            <intent-filter>
                <action android:name="android.intent.action.USER_PRESENT" />
                <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
            </intent-filter>
            <!-- Optional -->
            <intent-filter>
                <action android:name="android.intent.action.PACKAGE_ADDED" />
                <action android:name="android.intent.action.PACKAGE_REMOVED" />

                <data android:scheme="package" />
            </intent-filter>
        </receiver>

        <!-- Required SDK核心功能 -->
        <receiver android:name="cn.jpush.android.service.AlarmReceiver" />

        <!-- Required since 3.0.7 -->
        <!-- 新的tag/alias接口结果返回需要开发者配置一个自定的广播 -->
        <!-- 该广播需要继承JPush提供的JPushMessageReceiver类, 并如下新增一个 Intent-Filter -->
        <receiver
            android:name=".receiver.MyJPushMessageReceiver"
            android:enabled="true">
            <intent-filter>
                <action android:name="cn.jpush.android.intent.RECEIVE_MESSAGE" />

                <category android:name="${applicationId}" />
            </intent-filter>
        </receiver>

        <!-- 极光Required SDK核心功能 -->
        <activity
            android:name="cn.jpush.android.ui.PushActivity"
            android:configChanges="orientation|keyboardHidden"
            android:exported="false"
            android:theme="@android:style/Theme.NoTitleBar">
            <intent-filter>
                <action android:name="cn.jpush.android.ui.PushActivity" />

                <category android:name="android.intent.category.DEFAULT" />
                <category android:name="${applicationId}" />
            </intent-filter>
        </activity>

        <!-- SDK核心功能 -->
        <activity
            android:name="cn.jpush.android.ui.PopWinActivity"
            android:configChanges="orientation|keyboardHidden"
            android:exported="false"
            android:theme="@style/MyDialogStyle">
            <intent-filter>
                <category android:name="android.intent.category.DEFAULT" />
                <category android:name="${applicationId}" />
            </intent-filter>
        </activity>

        <receiver
            android:name="com.tendcloud.appcpa.ReferralReceiver"
            android:exported="true"
            tools:ignore="ExportedReceiver">
            <intent-filter>
                <action android:name="com.android.vending.INSTALL_REFERRER" />
            </intent-filter>
        </receiver>
    </application>
</manifest>
4.获取MetaData值和getPackageName()获取包名
  public static <T> T getMetaData(Context context, String name) {
        try {
            final ApplicationInfo ai = context.getPackageManager().getApplicationInfo(context.getPackageName(),
                    PackageManager.GET_META_DATA);

            if (ai.metaData != null) {
                return (T) ai.metaData.get(name);
            }
        }
        catch (Exception e) {
            e.printStackTrace();
        }

        return null;
    }

如:AppUtil.getMetaData(activity, "WCHAT_APPID")

5.如何打包

通过命令方式、Tasks、Generate signed APK打包生成多个马甲包。

(1)Terminal命令方式

gradle assembleRelease    // 构建productFlavors下所有Variant Release版本
gradle assembleDebug      // 构建productFlavors下所有Variant Debug版本
gradle assemble[productFlavors.name]          // 构建productFlavors下name的Variant Release和Debug版本
gradle assemble[productFlavors.name]Relase    // 构建productFlavors下name的Variant Releaseg版本
gradle assemble[productFlavors.name]Debug     // 构建productFlavors下name的Variant Debug版本
image.png
(2)Tasks或者Generate signed APK方式
如微信回调类配置
image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,458评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,454评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,171评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,062评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,440评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,661评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,906评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,609评论 0 200
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,379评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,600评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,085评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,409评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,072评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,088评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,860评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,704评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,608评论 2 270

推荐阅读更多精彩内容