同一设备安装线上线下两个应用的安装失败问题
写在前面
这个需求的情境是这样的:
在真正的项目开发的时候,会分生产环境和开发环境,开发环境都是放一些测试的数据,测试应用功能是否正常,生产环境都是一些真实的数据。所以在开发到由开发转到生产环境的时候,一台手机上就可能需要安装线上和线下两个环境的同一个应用,以便线上发现问题的时候比对下线下的环境,快速定位问题根源。
配置
我们都清楚基于AndroidStudio开发的安卓应用是以gradle配置下的applicationId为应用的唯一标示,只要我们在debug环境和release环境配置应用的不同id就可以了,接下来我们讲讲怎么实现
- 配置release和debug
这个一个很普通的配置,一般我们在debug和release环境会配置一个控制的变量来选择线上线下环境的参数,在android目录下配置
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
debug{
}
}
有一个默认的配置,叫defaultConfig,配置着一些基本信息
defaultConfig {
applicationId "com.ddstar.demo"
minSdkVersion 15
targetSdkVersion 25
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
以上的配置不管是debug还是release,applicationId都是一样的
- 配置productFlavors
同样在android目录下配置
productFlavors {
onLine {
applicationId 'com.ddstar.online'
...
}
offline {
applicationId 'com.ddstar.offline'
...
}
}
如上面代码所示,在productFlavors 目录下配置两个节点onLine 和offline ,分别配置两个key为applicationId 但是value不一样的两个值,同步一下gradle,在grede-->app-->task-->other下面
打包处可以看到有offline,online和debug,release的四种组合,那我们需要的呢,就是打包offlineDebug和onlineRelease,这样打包出来的两个应用就会以不同的applicationId和不同的环境存在了。
其实也蛮简单的,我这边主要是想讲一下一个问题,这个问题挺隐秘,网上资料也不多。
问题
当我们的应用里面有用到provider的时候,像以下代码
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.ddstar.demo.FileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/rc_file_path" />
</provider>
这边有涉及到applicationId,当初没有考虑到这个问题,于是applicationId直接就用默认的那个id写进去了,这样会造成当用下面两个id打包应用的时候,会没办法安装,因为这个applicationId是不对的呀,怎么办?所以这边有个建议,就是以后有用到应用id的,建议都用引用gradle里面的applicationId的方式来使用,因为gradle的配置优先级是最高的,你在清单文件里面配置的参数如果有在gradle里面配置了,就会被gradle覆盖掉,所以怎么改呢
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="${applicationId}.FileProvider"//这边做修改
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/rc_file_path" />
</provider>
看到没有,这样子不管你gradle怎么配置,我都不怕了
其实不管是provider还是其他什么配置有用到这个applicationId的(不是packager参数),都应该这么配置,不然会出现一些诡异的错误,好的,就写到这里。