三种方案实现Android应用的环境分离

2024-09-02 15:08

本文主要是介绍三种方案实现Android应用的环境分离,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

通常产品在迭代的时候,都有测试环境和正式环境,或者说是生产和开发环境,一般软件开发阶段都是在测试环境上运行调试,而正式打包发布时会配置正式环境的服务器,也就是不同的接口URL和数据库的区别。所以开发人员经常要在测试环境与正式环境之间来回切换,这带来了很大不便。本文提供了三种种方式来解决这个问题。

第一种方案:默认情况下,项目的buildTypes包含debug和release两个构建版本,用来分别对应测试环境和生产环境,设置applicationIdSuffix ,这样测试环境和生产环境实际上等于是已经分开的两个app了,可以实现在同一个设备上来安装同一个应用的不同版本。

第二种方案:在gradle构建的时候,Product Flavors中定义两个渠道,分别对应测试环境和生产环境,并且每个渠道applicationId不同,也可以实现在同一个设备上来安装同一个应用的不同版本。

第三种方案:手机SDCARD中创建隐藏文件,在程序启动的时候,判断有没有隐藏文件,进行测试环境和正式环境的切换,这样只需要一个apk,通过隐藏配置文件,动态进行环境的切换。

方案一和方案二的本质都是修改applicationId的值,构建打包出不同包名的apk安装文件。方案三不需要同时安装两个apk文件就可以实现环境分离,不足是每次切换的时候,app要关闭重启一下。下面简述一下三种方案的实现。

方案一:buildTypes

在app/gradle中添加下列代码

   defaultConfig {applicationId "com.environment.switch"minSdkVersion 15targetSdkVersion 23versionCode 1versionName "1.0"testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"}buildTypes {release {minifyEnabled falseproguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'}debug{applicationIdSuffix ".debug"}}

将debug类型的Application Id Suffix设置为“.debug”,然后在Application的onCreate中判断是哪一个应用,设置不同的服务器地址。

public class ServerConfigManger {private ServerConfigManger(){};private static  class ServerConfigMangerHolder{private static final ServerConfigManger instance=new ServerConfigManger();}public static ServerConfigManger getInstance(){return  ServerConfigMangerHolder.instance;}public String   getServerURL(Application pApplication){//测试环境服务器地址String urlPrefix="text.xxx.com";ApplicationInfo applicationInfo = pApplication.getApplicationInfo();String processName = applicationInfo.processName;if( processName.endsWith(".debug")){return urlPrefix;}else {//正式环境服务器地址urlPrefix="release.xxx.com";return  urlPrefix;}}
}
  @Overridepublic void onCreate() {super.onCreate();String serverURL = ServerConfigManger.getInstance().getServerURL(this);}

构建类型仍然是两个,debug和release,如下图:

build_type.png
只是现在的debug对应的是测试环境,而release对应的是正式环境,debug应用的包名是com.environment.switch.debug,release应用的包名是com.environment.switch;这种方式还可以继续优化一下,因为这两个应用安装之后,桌面的应用图标是一样的,下面将debug版本的应用图标换掉。
在src目录下新建一个debug目录,将main目录下的res目录复制一份到debug目录下,修改各个分辨率下的桌面Icon和strings.xml文件中的应用名称,如下图:
修改桌面图标.png
到此第一种方案就OK了

方案二:Product Flavors

在app/gradle中添加下列代码

   productFlavors{en1{applicationId "com.environmentswitch_beta"}en2{applicationId "com.environmentswitch"}}

所产生的构建如下,总共4个
构建类型.png

可以将en1的debug和release对应测试环境,en2的debug和release对应正式环境,我们在Application中的onCreate中根据运行时选择的是en1还是en2进行服务器地址取出不同的值。同 步骤一;

方案二:隐藏配置文件

在SDCARD中设置隐藏文件,根据隐藏文件判断是哪一套环境,假设存在test1,就用测试环境1,存在test2,就用测试环境2,存在test3,就用测试环境3,现在新建ServerBuildConfig,遍历SDCARD进行判断。

public class ServerBuildConfig {public static final int TEST_ENVIRONMENT_ONE = 0;public static final int TEST_ENVIRONMENT_TWO = 1;public static final int TEST_ENVIRONMENT_THREE = 2;public static final int RELEASE_ENVIRONMENT = 3;public static int mWhichEnvironment = 0;static {loadConfig();}private static void loadConfig() {if(!Environment.getExternalStorageState().equals("mounted")) {Log.w("ServerBuildConfig", "No valid SDCard, skip ServerBuildConfig!");} else {String buildConfigPath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/.appconfig";Log.i("ServerBuildConfig", "start to load configs for VSBuildConfig from: " + buildConfigPath);File buildFile = new File(buildConfigPath);if(buildFile.exists()) {File[] listFiles = buildFile.listFiles(new ConfigFilenameFliter());File testFile = null;if(listFiles != null) {for (File file : listFiles) {if (file.getName().startsWith(".test")) {if (testFile == null || file.lastModified() > testFile.lastModified()) {testFile = file;}}}}if(testFile != null) {if(!".test".equals(testFile.getName()) && !".test1".equals(testFile.getName())) {if(".test3".equals(testFile.getName())) {mWhichEnvironment = 3;} else if(".test2".equals(testFile.getName())) {mWhichEnvironment = 2;}} else {mWhichEnvironment = 1;}Log.i("ServerBuildConfig", "mWhichEnvironment = " + mWhichEnvironment);}}}}private static class ConfigFilenameFliter implements FilenameFilter {ConfigFilenameFliter() {}public boolean accept(File dir, String filename) {return filename.startsWith(".");}}
}

然后在ServerConfigManger添加一个方法,返回不同的服务器地址

public String   getServerURL(){String urlPrefix="text1.xxx.com";switch (ServerBuildConfig.mWhichEnvironment) {case ServerBuildConfig.TEST_ENVIRONMENT_ONE:urlPrefix="text1.xxx.com";break;case ServerBuildConfig.TEST_ENVIRONMENT_TWO:urlPrefix="text2.xxx.com";break;caseServerBuildConfig.TEST_ENVIRONMENT_THREE:urlPrefix="text3.xxx.com";break;case ServerBuildConfig.RELEASE_ENVIRONMENT:urlPrefix="release.xxx.com";break;}return  urlPrefix;}

在自定义的Application的onCreate中使用

String serverURL = ServerConfigManger.getInstance().getServerURL();

到此三种方案叙述完毕,对于第一种方案是最简便的,但是在有多套设置环境的情况下,第一种就不能满足需求了,就可以采取第二种以及第三种。总之,团队有一个约定,采取哪一种无关紧要。

这篇关于三种方案实现Android应用的环境分离的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1130300

相关文章

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

通过Docker容器部署Python环境的全流程

《通过Docker容器部署Python环境的全流程》在现代化开发流程中,Docker因其轻量化、环境隔离和跨平台一致性的特性,已成为部署Python应用的标准工具,本文将详细演示如何通过Docker容... 目录引言一、docker与python的协同优势二、核心步骤详解三、进阶配置技巧四、生产环境最佳实践

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

Nginx部署HTTP/3的实现步骤

《Nginx部署HTTP/3的实现步骤》本文介绍了在Nginx中部署HTTP/3的详细步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前提条件第一步:安装必要的依赖库第二步:获取并构建 BoringSSL第三步:获取 Nginx

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima