Android 判断当前网络类型是否为5G

2021.1.16更新
    适配targetSdkVersion P(28)及更高的版本

项目中若存在判断当前网络类型,一般都会用到如下方法:

    /**
     * get the network type
     *
     * @param ctx Context
     * @return networktype
     */
    public static int getNetWorkType(Context ctx) {
        int networkType = TelephonyManager.NETWORK_TYPE_UNKNOWN;
        TelephonyManager tm = (TelephonyManager) ctx.getSystemService(Context.TELEPHONY_SERVICE);
        try {
            int defaultDataSubId = getSubId();
            if (defaultDataSubId == -1) {
                networkType = tm.getNetworkType();
            } else {
                try {
                    Method dataNetworkType = TelephonyManager.class
                            .getDeclaredMethod("getDataNetworkType", new Class[]{int.class});
                    dataNetworkType.setAccessible(true);
                    networkType = (int) dataNetworkType.invoke(tm, defaultDataSubId);
                } catch (Throwable t) {
                    t.printStackTrace();
                }
                if (networkType == TelephonyManager.NETWORK_TYPE_UNKNOWN) {
                    networkType = tm.getNetworkType();
                }
            }
        } catch (Throwable t) {
            t.printStackTrace();
        }
        return networkType;
    }

    /**
     * get data sub id
     *
     * @return subId
     */
    private static int getSubId() {
        int defaultDataSubId = -1;
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
            defaultDataSubId = SubscriptionManager.getDefaultDataSubscriptionId();
        }
        return defaultDataSubId;
    }

至于networkType的取值,可参考TelephonyManager类。

使用上述方法,在5G网络之前都没问题,到5G网络时,查看Android Q源码,发现TelephonyManager类里增加了这么一个类型常量:

    /** Current network is NR(New Radio) 5G. */
    public static final int NETWORK_TYPE_NR = TelephonyProtoEnums.NETWORK_TYPE_NR; // 20.

意味着在5G网络下上述两个方法返回的networkType应该是20,实际测试了下,然并卵,返回的是13(NETWORK_TYPE_LTE,表示4G)。
是Android系统出问题了?还是厂商定制ROM出问题了?
查看运营商5G网络的搭建,发现现在5G网络基本都是通过4G增强型基站发射,也就是所谓的非独立组网(NSA),所以系统获取到的是NETWORK_TYPE_LTE,是没错的。只有独立5G组网(SA),系统才会返回NETWORK_TYPE_NR。
可厂商的手机明明就可以显示5G网络标识,不一致了,怎么搞?

网传华为提供了自己的接口判断是否为5G网络,亲试确实可正常使用,如下代码:

    /**
     * get huawei network type
     *
     * @return networkType
     */
    public static int getHwNetworkType(Context ctx) {
        int networkType = TelephonyManager.NETWORK_TYPE_UNKNOWN;
        if (VERSION.SDK_INT >= VERSION_CODES.O
                && ctx.checkSelfPermission(permission.READ_PHONE_STATE) == PackageManager.PERMISSION_GRANTED) {
            try {
                TelephonyManager tm = (TelephonyManager) ctx.getSystemService(Context.TELEPHONY_SERVICE);
                ServiceState ss;
                int subId = SubscriptionManager.getDefaultDataSubscriptionId();
                if (subId < 0) {
                    ss = tm.getServiceState();
                } else {
                    Class<TelephonyManager> classTm = TelephonyManager.class;
                    Method method = classTm
                            .getDeclaredMethod("getServiceStateForSubscriber", new Class[]{int.class});
                    method.setAccessible(true);
                    Object objSs = method.invoke(tm, subId);
                    ss = (ServiceState) objSs;
                }

                Method hwGetNetworkMethod = ServiceState.class.getMethod("getHwNetworkType");
                hwGetNetworkMethod.setAccessible(true);
                Integer hwNetType = (Integer) hwGetNetworkMethod.invoke(ss);
                if (hwNetType != null) {
                    networkType = hwNetType;
                }
            } catch (Exception e) {
                QLog.e(tag, QLog.USR, "getHwNetworkType throw ex", e);
            }
        }
        return networkType;
    }

华为的搞定了,其它厂商呢?找了半天没找到类似接口,有点小无奈...

仔细看华为提供的接口,里面用到了ServiceState,在Android Q上此类有getNrState方法,"nr"跟NETWORK_TYPE_NR貌似有点儿像,也许这个方法正确判断5G,但是由于系统限制,此方法无法反射调用。但里面的nrState在toString()方法里面出现了,是不是可以搞事情?于是有了下面通用的方法:

/**
 * network util
 */
public class NetworkUtil {

    public static final int NETWORK_TYPE_NR = 20;

    public static final int SDK_VERSION_Q = 29;

    /**
     * get the network type
     *
     * @param ctx Context
     * @return networkType
     */
    public static int getNetWorkType(Context ctx) {
        int networkType = TelephonyManager.NETWORK_TYPE_UNKNOWN;
        TelephonyManager tm = (TelephonyManager) ctx.getSystemService(Context.TELEPHONY_SERVICE);
        try {
            int defaultDataSubId = getSubId();
            if (defaultDataSubId == -1) {
                networkType = tm.getNetworkType();
            } else {
                Object obj = ReflectionUtil.invokeMethod(tm,
                        "android.telephony.TelephonyManager", "getDataNetworkType",
                        new Class[]{int.class}, defaultDataSubId);
                if (obj instanceof Integer) {
                    networkType = (Integer) obj;
                }
                if (networkType == TelephonyManager.NETWORK_TYPE_UNKNOWN) {
                    networkType = tm.getNetworkType();
                }
            }
        } catch (Throwable t) {
            t.printStackTrace();
        }
        if (networkType == TelephonyManager.NETWORK_TYPE_LTE) {
            networkType = adjustNetworkType(ctx, networkType);
        }
        return networkType;
    }

    /**
     * get the 5G network type
     *
     * @param ctx Context
     * @param networkTypeFromSys this method can be call only when networkTypeFromSys = 13(LET)
     * @return correct network type
     */
    private static int adjustNetworkType(Context ctx, int networkTypeFromSys) {
        int networkType = networkTypeFromSys;
        if (VERSION.SDK_INT >= SDK_VERSION_Q
                && ctx.checkSelfPermission(permission.READ_PHONE_STATE)
                == PackageManager.PERMISSION_GRANTED) {
            try {
                TelephonyManager tm = (TelephonyManager) ctx
                        .getSystemService(Context.TELEPHONY_SERVICE);
                ServiceState ss = null;
                int defaultDataSubId = getSubId();
                if (defaultDataSubId == -1) {
                    ss = tm.getServiceState();
                } else {
                    Object obj = ReflectionUtil
                            .invokeMethod(tm, "android.telephony.TelephonyManager",
                                    "getServiceStateForSubscriber", new Class[]{int.class},
                                    defaultDataSubId);
                    if (obj instanceof ServiceState) {
                        ss = (ServiceState) obj;
                    }

                    if (ss == null) {
                        ss = tm.getServiceState();
                    }
                }
                if (ss != null && isServiceStateFiveGAvailable(ss.toString())) {
                    networkType = NETWORK_TYPE_NR;
                }
            } catch (Throwable t) {
                t.printStackTrace();
            }
        }
        return networkType;
    }

    /**
     * get data sub id
     *
     * @return subId
     */
    private static int getSubId() {
        int defaultDataSubId = -1;
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
            defaultDataSubId = SubscriptionManager.getDefaultDataSubscriptionId();
        }
        return defaultDataSubId;
    }

    /**
     * check the service state str is 5G
     *
     * @param ss services state str
     * @return true if is 5G
     */
    private static boolean isServiceStateFiveGAvailable(String ss) {
        boolean available = false;
        if (!TextUtils.isEmpty(ss)
                && (ss.contains("nrState=NOT_RESTRICTED")
                || ss.contains("nrState=CONNECTED"))) {
            available = true;
        }
        return available;
    }
}

此方法为通用方法,使用时只需调用NetworkUtil#getNetWorkType(Context)即可,当然前提还得先获取下READ_PHONE_STATE权限,各大厂商皆可用。

上述代码已适配targetSdkVersion P(28)及更高的版本,ReflectionUtil代码可参考这篇文章

转载请注明出处:https://www.jianshu.com/p/f63cd36f7a2b

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352