功能点 
  • 判断某个IP地址是否合法
  • 判断两个IP地址是否在同一个网段中
  • 判断两个IP地址的大小关系
知识准备
基本原理

IP地址范围
0.0.0.0~
255.255.255.255,包括了mask地址。

IP地址划分
  • A类地址:1.0.0.1~126.255.255.254
  • B类地址:128.0.0.1~191.255.255.254
  • C类地址:192.168.0.0~192.168.255.255
  • D类地址:224.0.0.1~239.255.255.254
  • E类地址:240.0.0.1~255.255.255.254
判断两个IP地址是否是同一个网段中
     要判断两个IP地址是不是在同一个网段,就将它们的IP地址分别与子网掩码做与运算,得到的结果一网络号,如果网络号相同,就在同一子网,否则,不在同一子网。

例:假定选择了子网掩码255.255.254.0,现在分别将上述两个IP地址分别与掩码做与运算,如下图所示:

211.95.165.24 11010011 01011111 10100101 00011000
255.255.254.0 11111111 11111111 111111110 00000000
与的结果是: 11010011 01011111 10100100 00000000
211.95.164.78 11010011 01011111 10100100 01001110
255.255.254.0 11111111 11111111 111111110 00000000
与的结果是: 11010011 01011111 10100100 00000000
     可以看出,得到的结果(这个结果就是网络地址)都是一样的,因此可以判断这两个IP地址在同一个子网。

如果没有进行子网划分,A类网络的子网掩码为255.0.0.0,B类网络的子网掩码为255.255.0.0,C类网络的子网掩码为255.255.255.0,缺省情况子网掩码为255.255.255.0


实现
  
   以Java语言实现,主要针对IPv4地址。
    
 代码实现如下(包括注释):

  1. package org.slive.net;
  2.  
  3. import java.net.UnknownHostException;
  4. import java.util.regex.Pattern;
  5.  
  6. /**
  7. * <pre>
  8. * IP地址范围:
  9. * 0.0.0.0~255.255.255.255,包括了mask地址。
  10. *
  11. * IP地址划分:
  12. * A类地址:1.0.0.1~126.255.255.254
  13. * B类地址:128.0.0.1~191.255.255.254
  14. * C类地址:192.168.0.0~192.168.255.255
  15. * D类地址:224.0.0.1~239.255.255.254
  16. * E类地址:240.0.0.1~255.255.255.254
  17. *
  18. * 如何判断两个IP地址是否是同一个网段中:
  19. * 要判断两个IP地址是不是在同一个网段,就将它们的IP地址分别与子网掩码做与运算,得到的结果一网络号,如果网络号相同,就在同一子网,否则,不在同一子网。
  20. * 例:假定选择了子网掩码255.255.254.0,现在分别将上述两个IP地址分别与掩码做与运算,如下图所示:
  21. * 211.95.165.24 11010011 01011111 10100101 00011000
  22. * 255.255.254.0 11111111 11111111 111111110 00000000
  23. * 与的结果是: 11010011 01011111 10100100 00000000
  24. *
  25. * 211.95.164.78 11010011 01011111 10100100 01001110
  26. * 255.255.254.0 11111111 11111111 111111110 00000000
  27. * 与的结果是: 11010011 01011111 10100100 00000000
  28. * 可以看出,得到的结果(这个结果就是网络地址)都是一样的,因此可以判断这两个IP地址在同一个子网。
  29. *
  30. * 如果没有进行子网划分,A类网络的子网掩码为255.0.0.0,B类网络的子网掩码为255.255.0.0,C类网络的子网掩码为255.255.255.0,缺省情况子网掩码为255.255.255.0
  31. *
  32. * @author Slive
  33. */
  34. public class IpV4Util
  35. {
  36. // IpV4的正则表达式,用于判断IpV4地址是否合法
  37. private static final String IPV4_REGEX = "((\\d|[1-9]\\d|1\\d{2}|2[0-4]\\d|25[0-5])(\\.(\\d|[1-9]\\d|1\\d{2}|2[0-4]\\d|25[0-5])){3})";
  38.  
  39. // 系统子网掩码,它与ip组成一个地址
  40. private int mask;
  41.  
  42. // 1代表A类,2代表B类,3代表C类;4代表其它类型
  43. public final static int IP_A_TYPE = 1;
  44. public final static int IP_B_TYPE = 2;
  45. public final static int IP_C_TYPE = 3;
  46. public final static int IP_OTHER_TYPE = 4;
  47.  
  48. // A类地址范围:1.0.0.1---126.255.255.254
  49. private static int[] IpATypeRange;
  50. // B类地址范围:128.0.0.1---191.255.255.254
  51. private static int[] IpBTypeRange;
  52. // C类地址范围:192.168.0.0~192.168.255.255
  53. private static int[] IpCTypeRange;
  54.  
  55. // A,B,C类地址的默认mask
  56. private static int DefaultIpAMask;
  57. private static int DefaultIpBMask;
  58. private static int DefaultIpCMask;
  59.  
  60. // 初始化
  61. static
  62. {
  63. IpATypeRange = new int[2];
  64. IpATypeRange[0] = getIpV4Value("1.0.0.1");
  65. IpATypeRange[1] = getIpV4Value("126.255.255.254");
  66.  
  67. IpBTypeRange = new int[2];
  68. IpBTypeRange[0] = getIpV4Value("128.0.0.1");
  69. IpBTypeRange[1] = getIpV4Value("191.255.255.254");
  70.  
  71. IpCTypeRange = new int[2];
  72. IpCTypeRange[0] = getIpV4Value("192.168.0.0");
  73. IpCTypeRange[1] = getIpV4Value("192.168.255.255");
  74.  
  75. DefaultIpAMask = getIpV4Value("255.0.0.0");
  76. DefaultIpBMask = getIpV4Value("255.255.0.0");
  77. DefaultIpCMask = getIpV4Value("255.255.255.0");
  78. }
  79.  
  80. /**
  81. * 默认255.255.255.0
  82. */
  83. public IpV4Util()
  84. {
  85. mask = getIpV4Value("255.255.255.0");
  86. }
  87.  
  88. /**
  89. * @param mask 任意的如"255.255.254.0"等格式,如果格式不合法,抛出UnknownError异常错误
  90. */
  91. public IpV4Util(String masks)
  92. {
  93. mask = getIpV4Value(masks);
  94. if(mask == 0)
  95. {
  96. throw new UnknownError();
  97. }
  98. }
  99.  
  100. public int getMask()
  101. {
  102. return mask;
  103. }
  104.  
  105. /**
  106. * 比较两个ip地址是否在同一个网段中,如果两个都是合法地址,两个都是非法地址时,可以正常比较;
  107. * 如果有其一不是合法地址则返回false;
  108. * 注意此处的ip地址指的是如“192.168.1.1”地址,并不包括mask
  109. * @return
  110. */
  111. public boolean checkSameSegment(String ip1,String ip2)
  112. {
  113. return checkSameSegment(ip1,ip2,mask);
  114. }
  115.  
  116. /**
  117. * 比较两个ip地址是否在同一个网段中,如果两个都是合法地址,两个都是非法地址时,可以正常比较;
  118. * 如果有其一不是合法地址则返回false;
  119. * 注意此处的ip地址指的是如“192.168.1.1”地址
  120. * @return
  121. */
  122. public static boolean checkSameSegment(String ip1,String ip2, int mask)
  123. {
  124. // 判断IPV4是否合法
  125. if(!ipV4Validate(ip1))
  126. {
  127. return false;
  128. }
  129. if(!ipV4Validate(ip2))
  130. {
  131. return false;
  132. }
  133. int ipValue1 = getIpV4Value(ip1);
  134. int ipValue2 = getIpV4Value(ip2);
  135. return (mask & ipValue1) == (mask & ipValue2);
  136. }
  137.  
  138. /**
  139. * 比较两个ip地址是否在同一个网段中,如果两个都是合法地址,两个都是非法地址时,可以正常比较;
  140. * 如果有其一不是合法地址则返回false;
  141. * 注意此处的ip地址指的是如“192.168.1.1”地址
  142. * @return
  143. */
  144. public static boolean checkSameSegmentByDefault(String ip1,String ip2)
  145. {
  146. int mask = getDefaultMaskValue(ip1); // 获取默认的Mask
  147. return checkSameSegment(ip1,ip2,mask);
  148. }
  149.  
  150. /**
  151. * 获取ip值与mask值与的结果
  152. * @param ipV4
  153. * @return 32bit值
  154. */
  155. public int getSegmentValue(String ipV4)
  156. {
  157. int ipValue = getIpV4Value(ipV4);
  158. return (mask & ipValue);
  159. }
  160.  
  161. /**
  162. * 获取ip值与mask值与的结果
  163. * @param ipV4
  164. * @return 32bit值
  165. */
  166. public static int getSegmentValue(String ip, int mask)
  167. {
  168. int ipValue = getIpV4Value(ip);
  169. return (mask & ipValue);
  170. }
  171.  
  172. /**
  173. * 判断ipV4或者mask地址是否合法,通过正则表达式方式进行判断
  174. * @param ipv4
  175. */
  176. public static boolean ipV4Validate(String ipv4)
  177. {
  178. return ipv4Validate(ipv4,IPV4_REGEX);
  179. }
  180.  
  181. private static boolean ipv4Validate(String addr,String regex)
  182. {
  183. if(addr == null)
  184. {
  185. return false;
  186. }
  187. else
  188. {
  189. return Pattern.matches(regex, addr.trim());
  190. }
  191. }
  192.  
  193. /**
  194. * 比较两个ip地址,如果两个都是合法地址,则1代表ip1大于ip2,-1代表ip1小于ip2,0代表相等;
  195. * 如果有其一不是合法地址,如ip2不是合法地址,则ip1大于ip2,返回1,反之返回-1;两个都是非法地址时,则返回0;
  196. * 注意此处的ip地址指的是如“192.168.1.1”地址,并不包括mask
  197. * @return
  198. */
  199. public static int compareIpV4s(String ip1,String ip2)
  200. {
  201. int result = 0;
  202. int ipValue1 = getIpV4Value(ip1); // 获取ip1的32bit值
  203. int ipValue2 = getIpV4Value(ip2); // 获取ip2的32bit值
  204. if(ipValue1 > ipValue2)
  205. {
  206. result = -1;
  207. }
  208. else if(ipValue1 <= ipValue2)
  209. {
  210. result = 1;
  211. }
  212. return result;
  213. }
  214.  
  215. /**
  216. * 检测ipV4 的类型,包括A类,B类,C类,其它(C,D和广播)类等
  217. * @param ipV4
  218. * @return 返回1代表A类,返回2代表B类,返回3代表C类;返回4代表D类
  219. */
  220. public static int checkIpV4Type(String ipV4)
  221. {
  222. int inValue = getIpV4Value(ipV4);
  223. if(inValue >= IpCTypeRange[0] && inValue <= IpCTypeRange[1])
  224. {
  225. return IP_C_TYPE;
  226. }
  227. else if(inValue >= IpBTypeRange[0] && inValue <= IpBTypeRange[1])
  228. {
  229. return IP_B_TYPE;
  230. }
  231. else if(inValue >= IpATypeRange[0] && inValue <= IpATypeRange[1])
  232. {
  233. return IP_A_TYPE;
  234. }
  235. return IP_OTHER_TYPE;
  236. }
  237.  
  238. /**
  239. * 获取默认mask值,如果IpV4是A类地址,则返回{@linkplain #DefaultIpAMask},
  240. * 如果IpV4是B类地址,则返回{@linkplain #DefaultIpBMask},以此类推
  241. * @param anyIpV4 任何合法的IpV4
  242. * @return mask 32bit值
  243. */
  244. public static int getDefaultMaskValue(String anyIpV4)
  245. {
  246. int checkIpType = checkIpV4Type(anyIpV4);
  247. int maskValue = 0;
  248. switch (checkIpType)
  249. {
  250. case IP_C_TYPE:
  251. maskValue = DefaultIpCMask;
  252. break;
  253. case IP_B_TYPE:
  254. maskValue = DefaultIpBMask;
  255. break;
  256. case IP_A_TYPE:
  257. maskValue = DefaultIpAMask;
  258. break;
  259. default:
  260. maskValue = DefaultIpCMask;
  261. }
  262. return maskValue;
  263. }
  264.  
  265. /**
  266. * 获取默认mask地址,A类地址对应255.0.0.0,B类地址对应255.255.0.0,
  267. * C类及其它对应255.255.255.0
  268. * @param anyIp
  269. * @return mask 字符串表示
  270. */
  271. public static String getDefaultMaskStr(String anyIp)
  272. {
  273. return trans2IpStr(getDefaultMaskValue(anyIp));
  274. }
  275.  
  276. /**
  277. * 将ip 32bit值转换为如“192.168.0.1”等格式的字符串
  278. * @param ipValue 32bit值
  279. * @return
  280. */
  281. public static String trans2IpStr(int ipValue)
  282. {
  283. // 保证每一位地址都是正整数
  284. return ((ipValue >> 24) & 0xff) + "." + ((ipValue >> 16) & 0xff) + "." + ((ipValue >> 8) & 0xff) + "." + (ipValue & 0xff);
  285. }
  286.  
  287. /**
  288. * 将ip byte数组值转换为如“192.168.0.1”等格式的字符串
  289. * @param ipBytes 32bit值
  290. * @return
  291. */
  292. public static String trans2IpV4Str(byte[] ipBytes)
  293. {
  294. // 保证每一位地址都是正整数
  295. return (ipBytes[0] & 0xff) + "." + (ipBytes[1] & 0xff) + "." + (ipBytes[2] & 0xff) + "." + (ipBytes[3] & 0xff);
  296. }
  297.  
  298. public static int getIpV4Value(String ipOrMask)
  299. {
  300. byte[] addr = getIpV4Bytes(ipOrMask);
  301. int address1 = addr[3] & 0xFF;
  302. address1 |= ((addr[2] << 8) & 0xFF00);
  303. address1 |= ((addr[1] << 16) & 0xFF0000);
  304. address1 |= ((addr[0] << 24) & 0xFF000000);
  305. return address1;
  306. }
  307.  
  308. public static byte[] getIpV4Bytes(String ipOrMask)
  309. {
  310. try
  311. {
  312. String[] addrs = ipOrMask.split("\\.");
  313. int length = addrs.length;
  314. byte[] addr = new byte[length];
  315. for (int index = 0; index < length; index++)
  316. {
  317. addr[index] = (byte) (Integer.parseInt(addrs[index]) & 0xff);
  318. }
  319. return addr;
  320. }
  321. catch (Exception e)
  322. {
  323. }
  324. return new byte[4];
  325. }
  326. }
应用
  1. public static void main(String[] args) throws UnknownHostException
  2. {
  3. // 判断ip两个地址的大小关系
  4. String ip1 = "10.8.9.116";
  5. String ip2 = "10.8.9.10";
  6. System. out.println("ip1 大于 ip2? " + (compareIpV4s (ip1, ip2) > 0));
  7.  
  8. String ip3= "10.8.8.116";
  9. String ip4 = "10.10.9.10";
  10.  
  11. System. out.println("ip3 大于 ip4? " + (compareIpV4s (ip3, ip4) > 0));
  12.  
  13. // 判断ip两个地址是否是同一个网段
  14. int mask1 = getIpV4Value( "255.255.255.0");
  15. int mask2 = getIpV4Value( "255.255.0.0");
  16.  
  17. System. out.println("ip1和ip2在同一个网段中? " + (checkSameSegment(ip1, ip2, mask1)));
  18.  
  19. System. out.println("ip3和ip4在同一个网段中 ?" + (checkSameSegment(ip3, ip4, mask2)));
  20.  
  21. // 判断ip5是否在ip1和ip2范围中
  22. String ip5= "10.8.8.8";
  23. // 假设ip1和ip2在同一个网段中,并且ip1为起始地址,ip2为结束地址,ip1<=1
  24. // 比较ip1与ip5是否在同一个网段中
  25. if(checkSameSegment(ip1, ip5, mask1))
  26. {
  27. // 判断ip5是否在ip1和ip2范围中
  28. if(((compareIpV4s(ip5, ip1)) >= 0) && (compareIpV4s(ip5, ip2) <= 0))
  29. {
  30. System. out.println("ip5 在ip1-ip2范围内" );
  31. }
  32. else if ((compareIpV4s(ip5, ip1)) < 0)
  33. {
  34. System. out.println("ip5 不在ip1-ip2范围内,因为ip5小于ip1" );
  35. }
  36. else
  37. {
  38. System. out.println("ip5 不在ip1-ip2范围内,因为ip5大于ip2" );
  39. }
  40. }
  41. else
  42. {
  43. System. out.println("ip5 不在ip1-ip2范围内,因为ip5不在ip1的网段中" );
  44. }
  45. }
总结
  • 了解正在表达式,并懂得编写判断IP地址是否合法的表达式
  • 了解判断IpV4地址是否在同一个网段的原理,并能够用Java语言实现

优化与扩展
  • 另一种判断IpV4地址合法性的方法(不通过正在表达式)
  • 判断IpV4是否是A类地址,或者B类地址,又或者是广播地址等
  • 多个IpV4地址是否在同一个网段中
  • 客户端,服务器端安全策略实现——限制特点的用户IP登录系统


[置顶] 如何判断两个IP大小关系及是否在同一个网段中的更多相关文章

  1. 判断两个IP是否处于同一子网(网段)

    如何去判断A和B两个IP是否在同一网段,假如有如下两个IP地址和子网掩码,判断他们是否是同一个网段的IP地址的方法: A IP:202.194.128.9 B IP:202.194.128.14 子网 ...

  2. 如何判断两个IP地址是不是在同一个网段

     要判断两个IP地址是不是在同一个网段,就将它们的IP地址分别与子网掩码做与运算,得到的结果一网络号,如果网络号相同,就在同一子网,否则,不在同一子网. 例:假定选择了子网掩码255.255.254. ...

  3. 如何判断两个IP是否在同一网段

    下来举例说明,如何去判断A和B两个IP是否在同一网段. A IP:202.194.128.9 B IP:202.194.128.14 子网掩码:255.255.255.0 1.把A和B的地址转换为二进 ...

  4. 判断两个IP地址是不是属于同一子网的方法

    一个IP地址有三种写法: 第一种,单个IP,如192.168.55.28 第二种,IP/子网掩码,如192.168.55.28/255.255.255.0 第三种,IP/子网掩码长度,如192.168 ...

  5. 如何判断两个IP地址是不是处于同一网段?

    个人理解,欢迎指正. 一.要判断两个IP地址是不是在同一个网段,就将它们的IP地址分别与子网掩码做与运算,得到的结果-->网络号,如果网络号相同, 就在同一子网,否则,不在同一子网. 例:假定选 ...

  6. python 判断两个ip地址是否属于同一子网

    python 判断两个ip地址是否属于同一子网 """ 判断两个IP是否属于同一子网, 需要判断网络地址是否相同 网络地址:IP地址的二进制与子网掩码的二进制地址逻辑&q ...

  7. 判断两个IP是否属于同一子网

    描述 子网掩码是用来判断任意两台计算机的IP地址是否属于同一子网络的根据.子网掩码与IP地址结构相同,是32位二进制数,其中网络号部分全为“1”和主机号部分全为“0”.利用子网掩码可以判断两台主机是否 ...

  8. python 判断两个ip是不是处于同一网段

    a_ip:10.10.15.100b_ip:10.10.15.101c_ip:10.10.10.100netmask:255.255.255.0 def numtobinary(num): binar ...

  9. POJ1269:Intersecting Lines(判断两条直线的关系)

    题目:POJ1269 题意:给你两条直线的坐标,判断两条直线是否共线.平行.相交,若相交,求出交点. 思路:直线相交判断.如果相交求交点. 首先先判断是否共线,之后判断是否平行,如果都不是就直接求交点 ...

随机推荐

  1. VC编程中如何设置对话框的背景颜色和静态文本颜色

    晚上编一个小程序,涉及到如何设置对话框的背景颜色和静态文本颜色.这在VC6.0中本来是一句话就搞定的事.在应用程序类中的InitInstance()函数添加: //设置对话框背景和文本颜色 SetDi ...

  2. C++模板:ST算法

    //初始化 void init_rmq(int n){ for(int i=0;i<n;i++)d[i][0]=a[i]; for(int j=1;(1<<j)<=n;j++) ...

  3. hpu校赛--雪人的高度(离散化线段树)

    1721: 感恩节KK专场——雪人的高度 时间限制: 1 Sec  内存限制: 128 MB 提交: 81  解决: 35 [提交][状态][讨论版] 题目描述 大雪过后,KK决定在春秋大道的某些区间 ...

  4. Intellij IDEA创建Maven Web项目

    1前言 在创建项目中,IDEA提供了非常多项目模板,比方Spring MVC模板,能够直接创建一个基于Maven的Spring MVC的demo,各种配置都已经设定好了,直接编译部署就能够使用. 最開 ...

  5. JNI(2)

    JNI(2) 访问字段和方法 JNI允许本地代码访问java 对象的字段和方法. 调用需要两个步骤: 例如调用cls类的f方法, 1. 获取方法ID jmethodID mid = env->G ...

  6. python中decorator

    先讲一下python中的@符号 看下面代码 @f @f2 def fun(args, args2, args3, args4, ……): pass 上面代码相当于 def fun(args, args ...

  7. YII2 Gridview

    YII2 Gridview 部分使用规则 1.页面显示的时间戳转换 a. [ 'label'=>'创建日期', 'attribute' => 'created_at', 'filter' ...

  8. asp.net 一般处理程序小优化

    使用asp.net mvc习惯了,最近项目中又开始使用asp.net,有大量的ajax方法调用,之前有两种方法来处理: Switch case :方法少还行,如果很多,就太蛋疼了,而且方法堆在一块,也 ...

  9. QT https post请求(QNetworkRequest要设置SSL证书,而SSL证书认证有三种,实测成功)

    以VS开发为例.因为https访问需要用到SSL认证,而QT默认是不支持SSL认证,所以在使用之前必须先做一些准备工作: 需要安装OpenSSL库: 1.首先打开http://slproweb.com ...

  10. TabSpec和TabHost实例

    TabSpec与TabHost TabHost相当于浏览器中浏览器分布的集合,而Tabspec则相当于浏览器中的每一个分页面.d在Android中,每一个TabSpec分布可以是一个组件,也可以是一个 ...