[C#]判断一个IP是否在某个IP段内

关于IP地址

IPv4地址是由4段0-255的数字组成的,例如:a.b.c.d(0≤a,b,c,d≤255),IPv4也叫32位地址,为什么是32位呢,因为把每一段转换成二进制后,它的取值范围就是00000000-11111111,所以加起来就是32位。
那么IP地址的本质是一个32位的二进制数,我们便可以将这个32位的二进制再转换成十进制,进行比较不就可以了吗?

IP地址转十进制

以4.3.2.1为例,对应的十进制是多少呢?

  • 1.转换成二进制,即00000100.00000011.00000010.00000001;
  • 2.去掉“.”,得到完整的二进制数:00000100000000110000001000000001;
  • 3.再转回十进制:00000100000000110000001000000001(2)=67305985(10);
    步骤比较繁琐,有什么更快的方法来计算这个十进制数呢?
    肯定是有的,就是移位操作,67305985=4<<24+3<<16+2<<8+1。这样就实现了IP地址转十进制的操作,即使是128位的IPv6,也是如法炮制。

判断一个IP是否在某个IP段

所以判断一个IP是否在某个IP段就直接将要对比的IP地址分别转换成十进制比大小就好了,也不用担心性能问题,因为移位操作是CPU最擅长的事情,比起加减乘除还要来的快,因此,我们将IP地址转换成十进制的操作封装成一个方法,方便直接调用:

        /// <summary>
        /// IP地址转换成数字
        /// </summary>
        /// <param name="addr">IP地址</param>
        /// <returns>数字,输入无效IP地址返回0</returns>
        private static uint IPToID(string addr)
        {
            if (!IPAddress.TryParse(addr, out var ip))
            {
                return 0;
            }

            byte[] bInt = ip.GetAddressBytes();
            if (BitConverter.IsLittleEndian)
            {
                Array.Reverse(bInt);
            }

            return BitConverter.ToUInt32(bInt, 0);
        }
       /// <summary>
        /// 判断IP地址在不在某个IP地址段
        /// </summary>
        /// <param name="input">需要判断的IP地址</param>
        /// <param name="begin">起始地址</param>
        /// <param name="ends">结束地址</param>
        /// <returns></returns>
         public static bool IpAddressInRange( string input,string begin,string ends)
        {
            uint current = IPToID(input);
            return current>=IPToID(begin)&&current<=IPToID(ends);
        }

判断一个是否在某个IP段其实就这么简单!

参考代码:
https://github.com/ldqk/Masuit.Tools/blob/f4299bf368e103c5d0e6c1c682f2efaae687752c/Masuit.Tools/Extensions.cs#L926

posted @ 2023-01-29 10:39  肥皂泡泡oO  阅读(138)  评论(0编辑  收藏  举报