React Native之Android多包共存解决方案

avatar
@哈啰

作者:平台前端团队 @海洋

背景

之前遇到过一个问题,当测试遇到不同环境的时候都要卸载旧的包,重新安装再去测试。是不是很麻烦呢,于是就有了这篇文章——《React native之android多包共存解决方案》

两种方案

研究发现发现两个方案

  • app内可以切不同的环境
  • 通过多包共存的方式 --- xx.正式, xx.测试、xx.开发环境 主要讲一下app 共存

多包共存

包类型

比如我有一个app 包他可能有这样几种类型

  • debug 包(默认渠道)
  • release 包(默认渠道)
  • debug 包(其他渠道 --- 比如美团)
  • release 包(其他渠道 --- 比如美团)

怎么共存

虽然我是个前端仔,but 查资料还是会的。经过不懈的努力,终于发现

安卓系统中区分不同应用使用的是 applicationId 属性

so 下一步问题变成如何去改 appId 了 查询资料发现可以利用 gradle 去设置 productFlavors 可以很方便的实现这个需求.

具体操作

 修改 android/app/build.gradle
  productFlavors {
        beta {//测试环境
            applicationId "com.x.beta"
            // applicationIdSuffix ".beta"
            manifestPlaceholders = [
                    app_name: "@string/app_name_beta",
            ]
            //  resValue("string", "envTag", "beta环境")
        }
        product {//生产环境
            applicationId "com.x.product"
            // applicationIdSuffix ".product"
            manifestPlaceholders = [
                    app_name: "@string/app_name",
            ]
            // resValue("string", "envTag", "生产环境")
        }
    }

这里有两种访问去改 appId

  • 第一种直接去设置不同的id
  • 第二种去设置 applicationIdSuffix 也就是应用标识后缀,比如我默认 appid是 com.x。执行 /gradlew assembleReleaseBeta 会在后面追加 com.x.beat。当然不用命令行的方式在 Android Studio 里面也能看到这个选项

动态设置app_name、icon、appkey

我们可以通过 manifestPlaceholders也就是占位符来设置

 // 扩展上面
 beta {
            ...
            manifestPlaceholders = [
                    app_name: "@string/app_name_beta",
                    app_icon: '@xxxx'
                    JPUSH_APPKEY: '123' // 极光推送,
                    GoogleMapKey: '456' //谷歌地图
            ]
        }
        product {}

build.gradle 修改完了,接下来还要设置 /android/app/src/main/AndroidManifest.xml里面的内容

<application
android:icon="${app_icon}"
     android:label="${app_name}"
     xxxxx>
     <!-- Google Map Key -->   
     <meta-data
         android:name="com.google.android.geo.API_KEY"
         android:value="${GoogleMapKey}" /> 

<!--  极光推送-->
   
     <!-- User defined. 用户自定义的广播接收器-->
     <receiver
         android:name="com.ablegenius.member.receiver.JpushReceiver"
         android:enabled="true">
         <!--android:process=":remote"广播运行在远端单独进程中 ,调试断点无法执行需要关闭 或者 debug时候选择 remote ! -->
         <intent-filter>
           
           xxxxx
             <!--推送包名必须一致使用Gradle中的常量才是最终的 -->
             <category android:name="${applicationId}" />
         </intent-filter>
     </receiver>

</application>

最后配置包名 android/app/src/main/res/values/strings.xml

<resources>
   <string name="app_name">x正式</string>
    <string name="app_name_beta">x测试版</string>
</resources>

开始打包了 -----

结果生成如下

开始安装两个版本

然后出现如下问题

image.png 最开始的时候我以为我的两个版本appid 是一样的,然而并不是

image.png

image.png

再次谷歌中 -----

说是某个三方库里面的 provider下的 authorities的问题

再次找本地的三方库 ----

终于发现一个库的作者是写死的不是动态的 然后改成 解决了我的问题

总结

最后成功的解决了我的问题

image.png 通过 productFlavors 这种方式动态去改 appid,其实是现在的通用的一种方式 --- 多渠道打包 所以如果用这种方式 后面可以直接打渠道包上架到不同的市场。