博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
在局域网中通过端口号扫描服务器IP
阅读量:6267 次
发布时间:2019-06-22

本文共 6236 字,大约阅读时间需要 20 分钟。

     简要说明                                                                                                                                                                                                                                                                                                                      

      本人才接触Android不就,这个月有个任务是这样的。客户端要和服务器进行Socket通信,由于服务器不固定也就导致了服务器IP不固定,就不能实现于多个服务器通信,于是有了这样一个需求。客户端根据服务器提供的端口号来扫描服务器IP,从而满足这个需求。拿到需求一看,果断不会,于是就百度去了。搜索解雇倒是不少,可是打开一看全是重复的代码,复制——粘贴。仔细一看,就连我这个新手也看出来了很多问题,果断拿来大大修改了一番,废话不多少直接看代码吧:

 

   实现代码:

1     //本地IP 2     private String locAddress; 3     //核心线程数 4     private  int corePoolSize = 2; 5     //最大线程数 6     private  int maximumPoolSize = 5; 7     //线程池维护线程所允许的空闲时间 8     private  long keepAliveTime = 30; 9     //线程池维护线程所允许的空闲时间的单位10     private  TimeUnit unit = TimeUnit.SECONDS;1112     //线程池所使用的缓冲队列13     private  BlockingQueue
workQueue = new LinkedBlockingDeque
();14 @SuppressWarnings("unused")15 private ThreadPoolExecutor threadPoolExecutor 16 = new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue);

从这段代码可以看出,我主要是定义了一个线程池,之后要开启线程来扫描局域网IP的,百度的方法是直接开启255个线程,我对这种做法真不知道说什么了~~~不多说,直接看方法代码:

 

  

1      2           public  void scan(final int SERVERPORT){ 3             locAddress = getLocAddrIndex();//获取本地ip前缀 4            5           //开启线程 6             threadPoolExecutor.execute( new connectionServer(SERVERPORT, locAddress,0,100)); 7             threadPoolExecutor.execute( new connectionServer(SERVERPORT, locAddress,101,200)); 8             threadPoolExecutor.execute( new connectionServer(SERVERPORT, locAddress,201,256)); 9             threadPoolExecutor.shutdownNow();10          11           }12

  从这个方法是用于外部调用的scan方法,传递端口号,然后开启线程进行扫描,大家都知道,如果在一个局域网的话设备的网段是一样的,比如说:一台电脑和手机都连接了同一个wifi。那么电脑的IP就可能是192.168.8.X,手机IP就是192.168.8.Y。就是说前三个字段是一致的,那我们只要去适配最后一位,那手机就一定有几率来匹配到电脑。这样一来就简单了,最后一位无非就是1~255所以我把这个字段分开来,开启三个线程分别扫描。扫描方法之后会列出来,我们先看看怎么得到局域网前三个字段,也就是举例中的192.168.8

getLocAddrIndex()

方法是获得IP前缀的

1         /** 2          * @return 3          * 获取IP前缀 4          */ 5            6         private String getLocAddrIndex() { 7              String str = getLocAddress(); 8                  9                 if(!str.equals("")){10                   return str.substring(0,str.lastIndexOf(".")+1);11                 }12             return null;13         }

可以看到,这个方法只是将getLocAddress()方法获得的本地IP进行了截取,也就是截取出网段。下来放出如何获取本地IP:

1     public static String getLocAddress() { 2             String ipaddress = ""; 3             try { 4               Enumeration
en = NetworkInterface.getNetworkInterfaces(); 5 // 遍历所用的网络接口 6 while (en.hasMoreElements()) { 7 NetworkInterface networks = en.nextElement(); 8 // 得到每一个网络接口绑定的所有ip 9 Enumeration
address = networks.getInetAddresses();10 // 遍历每一个接口绑定的所有ip11 while (address.hasMoreElements()) {12 InetAddress ip = address.nextElement();13 if (!ip.isLoopbackAddress()14 && InetAddressUtils.isIPv4Address(ip.getHostAddress())) {15 ipaddress = ip.getHostAddress();16 17 }18 }19 }20 } catch (SocketException e) {21 22 e.printStackTrace();23 }24 25 26 return ipaddress;27 }

这个方法是百度找的,可以获取到设备的IP但是别忘记权限哦。到这里我们也就获取到了网段地址也就是我们所说的192.168.8

接下来放出线程扫描端口的方法:

1         public  class connectionServer extends Thread{ 2             private int sERVERPORT;//端口号 3             private String slocAddress;//网段 4             private int sNum;//开始扫描点 5             private int eNum;//结束扫描点 6             public  connectionServer(int SERVERPORT, String locAddress,int startNum,int endNum) { 7                 sERVERPORT = SERVERPORT; 8                 slocAddress = locAddress; 9                 sNum = startNum;10                 eNum = endNum;11                  12             }13     14             @Override15             public void run() {16                 for(int i = sNum;i < eNum; i ++){17                     18                     String current_ip = slocAddress+i;19                     if(Call(current_ip, sERVERPORT)){20                         onSeekIp(current_ip);21                         22                     }23                     24                 }  25                 26             }27 }

可以看出来是用了一个构造将需要的参数传递进来。然后在run()方法里用for循环进来匹配 ,关于匹配的方法我要说几句,第一次我用的是命令行匹配的方法也就是- p - l  也就是先用了下Window DOS下的ping命令能平通的再尝试和服务器连接。可是后来发现有好多设备无法执行这个命令,比如 鄙人用的小米手机可以,同事的魅族就无法执行了,找来找去发现socket就有一个判断端口是否打开的方法,正式踏破铁屑无觅处,得来全不费工夫啊,各位看官,请看代码:

1         //判断端口是否打开 2           public  boolean Call(String ip, int port) { 3                4                 try { 5                     Socket socket = new Socket(); 6                     socket.connect(new InetSocketAddress(ip, port), 15); 7                     socket.close(); 8                     return true; 9                    }catch(Exception e) {
10 return false;11 }12 13 }

connect这个方法,没错就是这个方法可以判断端口是否打开,后面我用了阻塞连接,timeout设置了15如果连接不上就返回fasle一旦连接上了就返回true,在线程里面可以看到返回true就调用了一个onSeepIP的方法,这个是什么了,其实这个是写了一个供外部访问的接口,已经和这个需求没有多大关系了。另外在附上调取的Demo,如果能帮到大家就再好不过了。

1      2     public void setOnSeekListener(OnSeekIpListener listener){ 3         this.listener = listener; 4     } 5      6     private void onSeekIp(String ip){ 7         if( listener != null){ 8             listener.onSeekIp(ip); 9         }10     };

这些还是写在这个类里面的,OnSeekIPListener 是从新定义了一个接口类,里面只有一个简单的方法:

1 public interface OnSeekIpListener {2 3     public void onSeekIp(String ip);4 5 }

到这里这个可供外部调用的的 寻找服务器IP的需求已经完成了。下来些一个简单的Demo来调用接口:

1 public class MainActivity extends Activity { 2      3     String date ; 4     @Override 5     protected void onCreate(Bundle savedInstanceState) { 6         // TODO Auto-generated method stub 7      8         super.onCreate(savedInstanceState); 9         UDPClient udpClient = new UDPClient();10         udpClient.setOnSeekListener(new UDPSeekListenner());11         udpClient.scan(6666);12     }13 14     class UDPSeekListenner implements OnSeekIpListener {15 16         @Override17         public void onSeekIp(String ip) {18             Log.e("UDP", "demo server_ip:"+ip);19             20         }21     }22 }

至此,所有流程已经完毕,还请各位大大轻喷。

 

转载于:https://www.cnblogs.com/yishuifenglan/p/4164273.html

你可能感兴趣的文章
ajax提交多个对象,使用序列化表单和FormData
查看>>
深入分析由前序和中序重构二叉树问题
查看>>
leetcode 题解 || Valid Parentheses 问题
查看>>
将图片转成base64字符串并在JSP页面显示的Java代码
查看>>
什么是WeakHashMap--转
查看>>
js 面试题
查看>>
第二十二节,三元运算
查看>>
Yacc 与 Lex 快速入门
查看>>
Unity中HDR外发光的使用
查看>>
Flume负载均衡配置
查看>>
Ajax详解
查看>>
Ubuntu C/C++开发环境的安装和配置
查看>>
百世汇通快递地区选择插件,单独剥离
查看>>
Linux系统调用---同步IO: sync、fsync与fdatasync【转】
查看>>
【MyBatis学习06】输入映射和输出映射
查看>>
[LeetCode] Decode String 解码字符串
查看>>
数字逻辑的一些基本运算和概念
查看>>
ant重新编译打包hadoop-core-1.2.1.jar时遇到的错
查看>>
【★★★★★】提高PHP代码质量的36个技巧
查看>>
3 weekend110的配置hadoop(格式化) + 一些问题解决 + 未免密码配置
查看>>