最近项目中有新的需求,同一份代码需要打成两个完全独立的app包,其中只有少许界面不一样;最开始方案是通过git开出分支,修改其包名,但是考虑到一旦之前的代码出现问题就必须两份代码都要更改,修改工作量大,容易有漏网之鱼; 因此,看度娘有没有类似的方法;最后发现可以通过gradle配置实现该功能;
知道了可以通过gradle实现该功能,那么怎么做呢?
- 如何配置不同的包名(applicationId);
- 遇到的问题;
如何配置不同的包名
项目gradle在android节点中存在productFlavors节点字段,该字段就是实现打多包的重点;其配置如下;
productFlavors {
// 包名及、应用名、启动图标
wlha {
applicationId 'com.vanzh.code1'
manifestPlaceholders = [app_name: "code1", app_icon: "@mipmap/ic_launcher"]
}
pady {
applicationId 'com.vanzh.code2'
manifestPlaceholders = [app_name: "code2", app_icon: "@mipmap/ic_launcher"]
}
}
上面的分别定义了两个包的包名及、应用名、启动图标。定义成功后,在Builid Variants中会多出不同的版本,选择对应的版本进行编译即可,如下;
当然也可以使用手动签名的方法,同时生成两个release包;
遇到问题
1.安装到手机上之后,发现应用名及应用图完全一样。
解决方法:由于应用图标是在AndroidManifest.xml中配置的,因此需要在AndroidManifest中设置应用名及图标的地方使用productFlavors 里配置的app_name及app_icon字段;
<application
android:name=".App"
android:allowBackup="true"
android:icon="${app_icon}"
android:label="${app_name}"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".SplashActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
即上图中的android:label、android:icon两个字段;
2.编译过程中出现与第三方库冲突的问题;
出现原因:AndroidManifest在进行合并的时候,第三方库中也存在android:label标签,导致编译冲突。
解决方法:使用本项目android:label代替第三方库的android:label标签;同理,其他标签也才用此该当,多个需要替换标签之间用","隔开,代码如下:
tools:replace="android:label"
3.安装第二个应用时出现"存在相同的内容提供者",安装失败的问题;
出现原因:第一个安装的app中AndroidManifest使用了provider,其中android:authorities一致导致的问题;如下
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.vanzh.code1.provider"
android:grantUriPermissions="true"
android:exported="false"
tools:replace="android:authorities">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/filepaths"
tools:replace="android:resource"/>
</provider>
解决方法:此处应使用本项目的applicationId,即
android:authorities="${applicationId}.provider"
4.第三方库不能正常使用;
出现原因:由于包名不一致,每个包名需要与之匹配第三方库签名。
解决方法:重新申请签名;
引申:为避名由于包名引起的问题,代码中有使用到包名的地方建议通过BuilidConfig.APPLICATIN_ID或${applicationId}来获取;否则可能包2会访问包1,出现权限问题;