当前位置: 首页 > 面试题库 >

在Golang中检查IP地址切片中IP的有效方法

沈国安
2023-03-14
问题内容

我正在用Golang开发网络应用程序。我有一个IP地址。每当出现请求时,我都会使用它net.LookupIP(host)来查找返回IP地址的主机IP地址net.IP。比较这些的最佳方法是什么?

顺便说一下,在Python中,我们有一个set数据结构,使得上述问题很容易解决,但是Go呢?


问题答案:

带“套”

建立我们的集合

SetGo中没有内置类型,但是您可以优雅地使用a map[Type]bool作为集合,例如:

// Create a set with 2 values in it: [1, 2]
m := map[int]bool{1: true, 2: true}

// Test an element:
fmt.Println(m[1]) // true
fmt.Println(m[3]) // false

// Set an element:
m[3] = true
fmt.Println(m[3]) // true

// Delete an element:
delete(m, 1)
fmt.Println(m[1]) // false

注意:我们利用了以下事实:如果键不在映射中,则对映射建立索引将导致值类型的值为零(false在的情况下)bool,可以正确地告知元素不在映射(集合)中。

在Go Playground上尝试一下。

使用net.IP在集

现在,我们只需要一个表示的类型net.IP就可以用作地图中的键类型。

不幸的是,net.IP它本身没有资格,因为它是一个片段:

type IP []byte

切片不可比。有关详细信息,请参见此问题:以键作为数组类型的哈希,以及以下内容:为什么Go中有数组?

一种简单的方法是将其转换为规范string值,我们就完成了。为此,我们可以简单地将IP的字节转换为十六进制string。但是IPv4地址可能表示为IPv6,因此我们应该首先将其转换为IPv6:

func Key(ip net.IP) string {
    return hex.EncodeToString(ip.To16())
}

注意:IP地址的字节可能不是有效的UTF-8编码string(这是Go将strings
存储在内存中的方式),但是stringGo中的值表示任意字节序列,因此以下操作也有效,简单得多并且效率更高:

func Key(ip net.IP) string {
    return string(ip.To16())  // Simple []byte => string conversion
}

我们可以使用这样的IP字符串作为键。使用IP填充地图以进行检查:

// Populate forbidden IPs:
forbIPs := map[string]bool{
    Key(ip1): true,
    Key(ip2): true,
}

// Now check a single IP:
ipToCheck := ...
if forbIPs[Key(ipToCheck)] {
    fmt.Println("Forbidden!")
} else {
    fmt.Println("Allowed.")
}

如果您要检查多个IP(由所返回net.LookupIP()),则这是一个for循环:

ips, err := net.LookupIP(host)
// Check err
for _, ip := range ips {
    if forbIPs[Key(ip)] {
        // FORBIDDEN!
    }
}

备用金钥类型

请注意,如上所述,切片是不可比较的,但数组是可比较的。因此我们也可以使用数组作为键。它看起来像这样:

func Key(ip net.IP) (a [16]byte) {
    copy(a[:], ip)
    return
}

// And the IP set:
forbIPs := map[[16]byte]bool{
    // ...
}

备择方案

排序切片

或者,我们可以简单地将禁止的IP存储在一个切片中[]net.IP,并 对其进行排序
。如果已排序,我们可以使用二进制搜索在其中找到IP(标准库sort.Search())。

是的,O(log2(n))O(1)上面的(哈希)映射解决方案相比,二进制搜索具有更高的复杂性。但是这种选择还有另一个优点:

枚举单个IP并不总是可行的。有时(通常)列出IP范围比较容易。第一个解决方案不适用于处理IP范围,但是该解决方案可能是:您还可以及时找到覆盖IP地址的范围O(log2(n))



 类似资料:
  • 问题内容: 这个问题已经在这里有了答案 : 在SQL Server中存储IP地址的数据类型 (10个答案) 3年前关闭。 在MySQL中存储和检索IP地址的最有效方法是什么?现在我正在做: 其中ip是一个字段。 有一个更好的方法吗? 问题答案: 对于IPv4地址,您可能需要将其存储为,并使用和函数从其数字值返回IP地址,反之亦然。 例:

  • 问题内容: 如何检查shell脚本中IP地址的有效性,该范围在to 的范围内? 问题答案: 如果您使用的是bash,则可以对模式进行简单的正则表达式匹配,而无需验证四边形: 如果您坚持使用POSIX Shell,则可以使用BRE而不是ERE来做基本相同的事情: 请注意,假设您的正则表达式锚定在字符串的左侧,因此不需要缩写。 如果确认每个四边形小于256很重要,那么显然您将需要更多代码: 甚至更少的

  • 我想检查IP地址是否在最小和最大IP地址的范围内。我怎样才能在德尔福做到这一点? 例如,我想做这样的事情: 127.0.0.1是范围的起始值,127.0.0.255是范围的结束值,127.0.0.15是将要检查的IP地址。

  • 我有一个IP地址,我得到了另外两个IP地址,它们一起创建了一个IP范围。我想检查第一个IP地址是否在这个范围内。我怎么能在不使用循环的情况下在Python中找到它?

  • 我有一个IP地址,另外两个IP地址一起创建了一个IP范围。我想检查第一个IP地址是否在此范围内。如何在PHP中找到它?

  • 为什么会有这种报错 为了安全,滴滴所有的接口都会校验请求IP是否在IP白名单中。 IP白名单是在滴滴企业版开放平台(http://open.es.xiaojukeji.com/) 管理中心的应用中设置的。 解决方法 获取服务器IP地址 在自己服务器上执行命令:curl http://open.es.xiaojukeji.com/open/Tools/getIp 注意: 1. 不是在自己