公司产品集成了魅族推送平台,但是该平台并未在大部分魅族手机上启用,往往需要用户在主动反馈后,并由客服手动切换为魅族推送平台
通过进一步的调试,我们发现其内部原因就在于 魅族系统的校验。
我们都知道,现在Android系统手机各式各样,包括三星、华为、小米、魅族等等,它们的系统都不是原生的Android系统,都是经过手机厂商修改过的,并且通过刷机可以在不同手机刷各式各样的系统,在一些应用中需要根据不同的系统做不同的处理,因此我们就需要获取手机使用的什么系统,现在主要介绍一下华为、小米、魅族三种系统的获取方法。
原理:在Android系统中有一个类似Windows系统注册表的文件build.prop。这个文件内定义了系统初始(或永久)的一些参数属性、功能的开放等。
一、背景
魅族推送平台只能在魅族设备上使用,如果在非魅族设备上启用则可注册成功,但是一直收取不到信息。
因此在应用第一次启动时,我们会校验手机是否为魅族设备,如果是则启用魅族推送平台。
魅族官方并没有提供对外的系统校验方式。
android主要的校验来自于android.os.Build.MODEL,它是一个String变量,可以根据String中一些特殊字段来判断android设备属于什么设备,比如判断Nexus 7,代码入下
public static boolean isN7() {
return "Nexus 7".equals(android.os.Build.MODEL);
}
但是需要注意的是,这是一种硬件设备校验,也就是说仅是校验你的手机硬设信息为某种类型。因此对于推送来说,我们不能使用这种方式来进行校验,因为魅族推送平台是依赖于魅族系统,而非魅族设备。
在非魅族设备的手机上刷了魅族系统就可以使用魅族推送平台作为系统服务,而在魅族设备上刷了非魅族系统则不应使用魅族推送。
那么我们如何针对Android定制化的系统进行校验呢?
二、特殊系统校验
我们需要通过获取魅族系统某种特殊系统属性作为系统校验。
1.首先进入cmd,然后输入adb shell 命令进入adb shell (前提:adb必须要配置到系统环境变量中)
C:\Users\Administrator>adb shell
shell@mx4pro:/ $
2.进入system目录
shell@mx4pro:/ $ cd system
cd system
shell@mx4pro:/system $
3.查看系统属性
shell@mx4pro:/system $ cat build.prop
cat build.prop
# begin build properties
# autogenerated by buildinfo.sh
ro.build.cta=noncta
ro.build.id=KTU84P
ro.build.mask.id=4.4.4-1423716396_wo
ro.build.args=
ro.build.inside.id=4.4.4-20150212124636
ro.build.version.incremental=m76.Flyme_OS_4.2.2.1.20150212124636
ro.build.version.sdk=19
ro.build.version.codename=REL
ro.build.version.release=4.4.4
4.筛选包含Flyme字符的属性,因为既然是魅族操作系统,Flyme这个词最能代表它的属性
shell@mx4pro:/system $ cat build.prop | grep Flyme
cat build.prop | grep Flyme
ro.build.version.incremental=m76.Flyme_OS_4.2.2.1.20150212124636
ro.build.display.id=Flyme OS 4.2.2.1U
ro.build.description=meizu_mx4pro-user 4.4.4 KTU84P m76.Flyme_OS_4.2.2.1.2015021
2124636 release-keys
ro.build.fingerprint=Meizu/meizu_mx4pro/mx4pro:4.4.4/KTU84P/m76.Flyme_OS_4.2.2.1
.20150212124636:user/release-keys
shell@mx4pro:/system $
帅选到四个属性:ro.build.version.incremental ro.build.display.id ro.build.description ro.build.fingerprint
5.换一个其他的安装有魅族操作系统设备,重复上面的步骤。最终会发现 ro.build.display.id这个属性可以当做一个魅族操作系统唯一的特性(因为这个属性值几乎都会包含Flyme字段,这正是我们想要的),我们就可以用这个属性来与其他系统进行区分。
6.最后一步
// 判断是魅族操作系统
public static boolean isMeizuFlymeOS() {
return getMeizuFlymeOSFlag().toLowerCase().contains("flyme");
}
/**
* 获取魅族系统操作版本标识
*/
public static String getMeizuFlymeOSFlag() {
return getSystemProperty("ro.build.display.id", "");
}
private static String getSystemProperty(String key, String defaultValue) {
try {
Class<?> clz = Class.forName("android.os.SystemProperties");
Method get = clz.getMethod("get", String.class, String.class);
return (String)get.invoke(clz, key, defaultValue);
} catch (ClassNotFoundException e) {
} catch (NoSuchMethodException e) {
} catch (IllegalAccessException e) {
} catch (IllegalArgumentException e) {
} catch (InvocationTargetException e) {
}
return defaultValue;
}
三、其他
3.1 小米系统的类推
我们可以使用同样的方式,来找到小米系统的校验
/**
* 获取MIUI版本名
*/
public static String getMIUIVersionName() {
return getSystemProperty(“ro.miui.ui.version.name”, null);
}
// MIUI v5版本
public static boolean isV5(){
if(“V5”.equalsIgnoreCase(getMIUIVersionName())){
return true;
}
return false;
}
3.2 安卓系统的build.prop部分代码
# begin build properties (开始设置系统性能)
# autogenerated (通过设置形成系统信息)
ro.=GRI40 (版本ID)
ro.build.=GRJ22 (版本号)
ro.build.version.incremental=eng.buildbot.20110619.060228 (版本增量)
ro.build.version.sdk=10 (sdk版本)
ro.build.version.codename=REL (版本代号)
ro.build.version.release=2.3.4 (Android 2.3.4系统)
ro.build.date=Sun Jun 19 06:02:58 UTC 2011 (制作者及制作时间)
ro.build.date.utc=0
ro.build.type=user (编译模式,如user,userdebug,eng,test模式)
ro.build.user=buildbot (编译账户)
ro.build.host=bb1 (编译主机系统)
ro.build.tags=test-keys (编译标签)
ro.product.model=HTC Wildfire (HTC内部手机代号)
ro.product.brand=htc_wwe (手机品牌)
ro.product.name=htc_buzz (手机正式名称)
ro.product.device=buzz (采用的设备)
ro.product.board=buzz (采用的处理器)
ro.product.cpu.abi=armeabi-v6j (cpu的版本)
ro.product.cpu.abi2=armeabi (cpu的品牌)
ro.product.manufacturer=HTC (手机制造商)
其实这些配置参数是以键值对的方式存放,一般情况下厂商定制ROM的时候会在定制的ROM里面加入ROM特有的标识(当然不排除有些ROM在不同版本中使用不同的版本标识),我们可以根据这些ROM标识对常见的ROM进行识别。
关键代码如下:
public class OSUtils {
//MIUI标识
private static final String KEY_MIUI_VERSION_CODE = "ro.miui.ui.version.code";
private static final String KEY_MIUI_VERSION_NAME = "ro.miui.ui.version.name";
private static final String KEY_MIUI_INTERNAL_STORAGE = "ro.miui.internal.storage";
//EMUI标识
private static final String KEY_EMUI_VERSION_CODE = "ro.build.version.emui";
private static final String KEY_EMUI_API_LEVEL = "ro.build.hw_emui_api_level";
private static final String KEY_EMUI_CONFIG_HW_SYS_VERSION = "ro.confg.hw_systemversion";
//Flyme标识
private static final String KEY_FLYME_ID_FALG_KEY = "ro.build.display.id";
private static final String KEY_FLYME_ID_FALG_VALUE_KEYWORD = "Flyme";
private static final String KEY_FLYME_ICON_FALG = "persist.sys.use.flyme.icon";
private static final String KEY_FLYME_SETUP_FALG = "ro.meizu.setupwizard.flyme";
private static final String KEY_FLYME_PUBLISH_FALG = "ro.flyme.published";
/**
* @param
* @return ROM_TYPE ROM类型的枚举
* @description获取ROM类型: MIUI_ROM, FLYME_ROM, EMUI_ROM, OTHER_ROM
*/
public static ROM_TYPE getRomType() {
ROM_TYPE rom_type = ROM_TYPE.OTHER;
try {
BuildProperties buildProperties = BuildProperties.getInstance();
if (buildProperties.containsKey(KEY_EMUI_VERSION_CODE) || buildProperties.containsKey(KEY_EMUI_API_LEVEL) || buildProperties.containsKey(KEY_MIUI_INTERNAL_STORAGE)) {
return ROM_TYPE.EMUI;
}
if (buildProperties.containsKey(KEY_MIUI_VERSION_CODE) || buildProperties.containsKey(KEY_MIUI_VERSION_NAME) || buildProperties.containsKey(KEY_MIUI_VERSION_NAME)) {
return ROM_TYPE.MIUI;
}
if (buildProperties.containsKey(KEY_FLYME_ICON_FALG) || buildProperties.containsKey(KEY_FLYME_SETUP_FALG) || buildProperties.containsKey(KEY_FLYME_PUBLISH_FALG)) {
return ROM_TYPE.FLYME;
}
if (buildProperties.containsKey(KEY_FLYME_ID_FALG_KEY)) {
String romName = buildProperties.getProperty(KEY_FLYME_ID_FALG_KEY);
if (!TextUtils.isEmpty(romName) && romName.contains(KEY_FLYME_ID_FALG_VALUE_KEYWORD)) {
return ROM_TYPE.FLYME;
}
}
} catch (IOException e) {
e.printStackTrace();
}
return rom_type;
}
public enum ROM_TYPE {
MIUI,
FLYME,
EMUI,
OTHER
}
}
可能您需要对其他的ROM进行区分,那么只需三步:
1、使用BuildProperties获取到所有的key, 遍历获取到所有的value(getProperty),或者直接找到build.prop文件。
2、找到定制ROM特征的标识(key/value)
3、增加ROM_TYPE枚举类型,getRomType方法加入识别比对即可。