美文网首页
Android根据内网外网连接情况配置服务器访问IP

Android根据内网外网连接情况配置服务器访问IP

作者: DuBetter | 来源:发表于2019-11-06 17:13 被阅读0次

    新项目的app,可通过内网和外网的服务器ip进行请求访问,但是客户提供了专业终端,终端在wifi情况下走外网内网都可以,但关闭wifi则只能走4G专网,也就是只能走内网。

    可前往我的小站查看:Android根据内网外网连接情况配置服务器访问IP

    方案

    Android中可以直接调用底层的shell,执行相应的命令,因此只需要执行ping命令即可。Android可以通过 Process p = Runtime.getRuntime().exec(/system/bin/ping -c 1 -w 1 " + ip)执行。
    然后通过if (p.waitFor() == 0)判断是否ping通,这里的两个1表示参数,第一个表示ping 1次,第二个表示超过1s即为失败。

    完整实现

    1. 首先声明权限,这一步非常重要,在AndroidManifes文件中
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
    <uses-permission android:name="android.permission.INTERNET" />
    
    1. 维持几个全局变量
        String outer_ip = "183.230.XXX.XXX"; // 服务器外网IP
        String inner_ip = "192.168.XXX.XXX"; // 服务器内网IP
    
        boolean outerIpAvilable = false;  // 外网可用
        boolean innerIpAvialable = false; // 内网可用
    
    1. 开启两个线程去ping两个ip,并通过CountDownLatch控制同步。因为要在两个ping结束之后,配置了ip之后才能做接下来的操作
    private void initNetworkConfig() {
            try {
                final int totalThread = 2;
                CountDownLatch countDownLatch = new CountDownLatch(totalThread);
                ExecutorService executorService = Executors.newCachedThreadPool();
                executorService.execute(new PingNetwork(outer_ip, countDownLatch, false));
                executorService.execute(new PingNetwork(inner_ip, countDownLatch, true));
                countDownLatch.await(); // 等待二者执行完毕
                Log.d(TAG, "end");
                if (innerIpAvialable && outerIpAvilable)
                    Toast.makeText(this, "内外都可使用", Toast.LENGTH_SHORT).show();
                else if (outerIpAvilable)
                    Toast.makeText(this, "外网可使用", Toast.LENGTH_SHORT).show();
                else
                    Toast.makeText(this, "内网可使用", Toast.LENGTH_SHORT).show();
                executorService.shutdown();
            }catch (Exception e) {
                e.printStackTrace();
            }
        }
    
    1. 实现ping的异步线程,
    class PingNetwork implements Runnable {
    
            String ip; // 需要ping的ip
            CountDownLatch countDownLatch;
            boolean isCheckInner;
            public PingNetwork(String ip, CountDownLatch countDownLatch, boolean isCheckInner) {
                this.ip = ip;
                this.countDownLatch = countDownLatch;
                this.isCheckInner = isCheckInner;
            }
    
            @Override
            public void run() {
                try {
                    Process p = Runtime.getRuntime().exec("/system/bin/ping -c 1 -w 1 " + ip);// ping网址3次
                    // ping的状态
                    final int status = p.waitFor();
                    if (status == 0) {
                        Log.d(TAG, "ping onSuccess");
                        if (isCheckInner){
                            innerIpAvialable = true;
                            outerIpAvilable = false;
                        }
                        else{
                            outerIpAvilable = true;
                            innerIpAvialable = false;
                        }
                    } else {
                        // 读取ping的error内容,查看无法ping通的原因
                        InputStream errorStream = p.getErrorStream();
                        BufferedReader errIn = new BufferedReader(new InputStreamReader(errorStream));
                        StringBuilder sb = new StringBuilder();
                        String err = "";
                        while ((err = errIn.readLine()) != null) {
                            sb.append(err);
                        }
                        Log.d(TAG, "result err : " + sb.toString());
                        Log.d(TAG, "ping onFailure");
                    }
                } catch (Exception e) {
                    Log.d(TAG, "ping onFailure");
                } finally {
                    countDownLatch.countDown();
                }
            }
        }
    

    这里在ping失败时候可以打印错误信息查看,还记得第一步是声明权限,本人没有声明第二个权限,在这里得到了一个错误信息Pemission denied,网上说什么root的都有,其实不然。

    完善

    通过以上的实现,可以实现通过内网外网连接情况配置访问服务器的ip,但是设想一下,如果在app启动时,手机可以访问外网,所以程序配置了外网的ip(因为外网速度快),但是在使用的过程中,关闭了外网访问,比如说wifi,此时走了专网,即内网,则无法再访问服务器了。所以在切换网络时,需要从新配置访问ip。
    因此,需要再app里面通过广播的方式,在android N(android 7)之前,可以通过android.net.conn.CONNECTIVITY_CHANGE广播,可以静态注册和动态注册,然而在7之后,改广播无效了,可以使用以下方案替换。

    ConnectivityManager connectivityManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
            connectivityManager.requestNetwork(new NetworkRequest.Builder().build(),
                    new ConnectivityManager.NetworkCallback() {
                        @Override public void onAvailable(Network network) {
                            super.onAvailable(network);
                            LogUtil.d("网络发生改变,更改配置");
                            NetworkUtils.initNetworkConfig();
                        }
                    });
    

    NetworkUtils.initNetworkConfig();是我对上面通过ping配置ip的封装。


    相关文章

      网友评论

          本文标题:Android根据内网外网连接情况配置服务器访问IP

          本文链接:https://www.haomeiwen.com/subject/gctpbctx.html