记一次网络判断问题

很久没更新,换新工作后一直比较忙,也没有什么好分享的新技术,就记录一次碰到的坑。

1. 一般情况下的网络判断

一般我们开发的时候都会碰到需要判断网络状态然后做不同逻辑的处理的需求。然后一般比较多的做法就是适应ConnectivityManager、NetworkInfo这些方法去判断,例如我随便去网上找一段代码


    public static boolean isAvailable(Context context){
        boolean isAvailable = false ;
        ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
        NetworkInfo networkInfo = cm.getActiveNetworkInfo();
        if(networkInfo!=null && networkInfo.isAvailable()){
            isAvailable = true;
        }
        return isAvailable;
    }

一般做判断确实是能判断出当前是否有网,最直接能测试到的就是你开wifi和关wifi会得到不同的结果。
但是其实这个做法并不能去适配所有的情况,比如是你的路由器开着,但是没网,实际上设备是能收到这个路由器的wifi的,只是上不了网而已。而这种情况用上边的方法就无法判断出。
其实网上很多方法能判断是否有网络,而我尝试过很多种不同的写法,都无法检测出这种情况,如果有大佬确实知道能使用Android内部的方法来判断这种情况,还请留言给我

2.使用ping

既然正常的方法我们无法使用,那就只能使用一些黑科技了,我们可以去ping一个可靠的地址来判断是否有网,毕竟我们在电脑上也经常这样做。再随意从网上找段代码抄下来

private boolean ping() {
        try {
            Process process=Runtime.getRuntime().exec("/system/bin/ping -c 4 "+"www.baidu.com") ;
            int status = process.waitFor();
            if (status == 0) {
                return true;
            } else {  
                return false;
            }
        } catch (IOException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return false;
    }

大概这种写法,我没试过,但这种方法是肯定能成功的,如果不成功,去网上抄另一份代码就行。
那么问题来了,这样的方法就一定可靠吗,其实不一定可靠,比如说你得找一个可靠的ip,不然这个地址挂了的话判断就不准了。不仅如此,以我多年使用黑科技的经验,一般使用黑科技都会很多坑。

3.请求一个自家的接口

这可以算是一个用逻辑上去解决的方案,而且比ping可靠。按照这个思路去写,相信很多人都会,就请求一个自家的接口,请求成功就是正常,请求失败就没网。当然还要做更细微的判断,总不能你家接口返回500你也把它定义成没网吧。

但是一般来说,现在大多数Android开发使用的网络请求框架都是基于okhttp的。经过测试,我发现,在连上wifi但无网的情况下去请求后台接口,会花费很长时间,不会马上就给到结果。
所以我们需要去设置一个超时时间,一般okhttp设置超时时间都是在OkhttpClient中设置

OkHttpClient.Builder builder = new OkHttpClient.Builder()
                .connectTimeout(3, TimeUnit.SECONDS)
                .readTimeout(3, TimeUnit.SECONDS)

但其实经过测试你会发现,这样设置并没有效果。直到我看到了一篇文章“Okhttp解析DNS超时”,是我才学苏浅不完全熟悉网络相关的知识点,但我认为这个超时应该指的是解析DNS的超时,而不是连接的超时,尝试去写

 OkHttpClient.Builder builder = new OkHttpClient.Builder()
                .connectTimeout(3, TimeUnit.SECONDS)
                .readTimeout(3, TimeUnit.SECONDS)
                .retryOnConnectionFailure(false)
                .dns(new Dns() {
                    @Override
                    public List<InetAddress> lookup(String hostname) throws UnknownHostException {
                        if (hostname == null) {
                            throw new UnknownHostException("hostname == null");
                        } else {
                            try {
                                FutureTask<List<InetAddress>> task = new FutureTask<>(
                                        new Callable<List<InetAddress>>() {
                                            @Override
                                            public List<InetAddress> call() throws Exception {
                                                return Arrays.asList(InetAddress.getAllByName(hostname));
                                            }
                                        });
                                new Thread(task).start();
                                return task.get(3000, TimeUnit.MILLISECONDS);
                            } catch (Exception var4) {
                                UnknownHostException unknownHostException =
                                        new UnknownHostException("Broken system behaviour for dns lookup of " + hostname);
                                unknownHostException.initCause(var4);
                                throw unknownHostException;
                            }
                        }
                    }
                })

最后发现这样的做法确实有效,无网的情况确实能很快的返回结果。
所以我认为判断网络请求的操作可以结合1和3,基本能解决大部分的这类问题

4.事情并没这么简单

你以为故事到这里就结束啦?事情远没你想象的这么简单,用上面的方法,能够正常判断出有网、无网、有wifi但无网这3种情况。
但是我发现了另一种情况,我们平时有在公共场所连接过一些网络,而那些网络会要求你去输入账号密码登录校验,是否授权给你上网。
而在那种网络的情况下,依旧会存在一些概率,你请求这个你自己的接口是有网的,但是可能下一秒实际你的设备就断网了。
实际上我们要实时监听网络的改变情况,可以注册系统的一个广播进行监听,ConnectivityManager.CONNECTIVITY_ACTION这些。
但是在这类情况下断网,这些广播是没有推送给你网络状态变化的。
不止这些情况,说不定还存在更多其他的特殊情况,所以我的结论是上面的方法能解决大部分情况下的问题,但无法解决所以问题,并不是完全可靠

其他的我需要恶补这部分的知识再来更新这篇文章。

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

推荐阅读更多精彩内容