Go接口之nil != nil

一、引出话题:
在开始这个话题之前,我们先看一段代码以及其输出结果,代码如下所示:

从上面的输出结果来看,GetA()返回的类型为*A就算赋值为nil,也不等于nil。

看到这里笔者觉得很奇怪,明明是GetA()返回值是nil,为什么nil的判断条件是false呢?

二、原因分析:
在研究了interface之后,笔者发现了真相,原来Go语言中的interface是不是nil的条件,不单单是interface中的值是nil,类型还需要是nil才行。

通过Go的官方文档描述来看,原因如下:// 原文链接:https://golang.org/doc/faq#nil_error

首先,interface有两个核心元素,一个是type,一个是value,在为interface赋值的时候,首先会将类型付给type,其次才会将数值赋值给value。

例如:a := interface{} 
a = 10 // 此时a.type = int, a.value = 10

其次,interface == nil的成立条件是,type和value同时都为nil,只要有中一个不为nil,那么interface就不是nil。

验证代码如下所示:

通过上面代码的输出,我们能够看到GetA()返回的空接口,type已经被赋值了,就算value还是nil,接口a依然不会是nil。而对于空接口b来说,类型和数值都是空值,那么b就是一个nil。(备注:b的value值是invalid的原因是,在运行阶段空接口并没有找到这个nil是什么类型,也就没有办法给b.value赋值,所以才会打印invalid reflect.Value。)

三、补充知识:
对于接口变量来说,如果通过一个指向nil的数据结构去赋值的话,这个接口变量还是一个指针类型,代码如下所示:

规则整理:

1)指针的判断,都涉及到type和value。
2)接口指针之间的判等,要基于type与value,一个不同则不等。
3)接口指针与其对应实现的结构体指针,可以进行判等操作。
……………………………………………………………………………………
灰子学技术 公众号:


  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
package main /* #include <sys/socket.h> #include <netpacket/packet.h> #include <net/ethernet.h> */ import "C" import ( "fmt" "log" "net" "os" "os/exec" "syscall" "unsafe" ) const ( ProtocolIP = 0x0800 ETH_P_ALL = 0x0003 BufferSize = 65536 ) var Interface string // 网卡接口名称 func main() { // 检查程序是否以root身份运行 if os.Geteuid() != 0 { fmt.Println("Please run the program as root.") // 以root身份重新运行程序 cmd := exec.Command("sudo", os.Args[0]) cmd.Stdout = os.Stdout cmd.Stdin = os.Stdin cmd.Stderr = os.Stderr err := cmd.Run() if err != nil { log.Fatal(err) } os.Exit(0) } // 获取系统的网络接口信息 interfaces, err := net.Interfaces() if err != nil { log.Fatal(err) } // 遍历网络接口,打印接口名称 fmt.Println("Available network interfaces:") for _, iface := range interfaces { fmt.Println(iface.Name) } // 设置要使用的网卡接口(例如:eth0) Interface = "eth0" // 更改为你的网卡接口名 // 创建原始套接字 sockFD, err := syscall.Socket(C.AF_PACKET, syscall.SOCK_RAW, int(htons(ETH_P_ALL))) if err != nil { log.Fatal(err) } defer syscall.Close(sockFD) // 获取网卡接口索引 iface, err := net.InterfaceByName(Interface) if err != nil { log.Fatal(err) } // 绑定原始套接字到网卡接口 sa := C.struct_sockaddr_ll{ sll_family: C.AF_PACKET, sll_protocol: htons(ETH_P_ALL), sll_ifindex: C.int(iface.Index), } if err := syscall.Bind(sockFD, (*syscall.Sockaddr)(unsafe.Pointer(&sa))); err != nil { log.Fatal(err) } // 在一个无限循环中接收数据包 for { buffer := make([]byte, BufferSize) n, _, err := syscall.Recvfrom(sockFD, buffer, 0) if err != nil { log.Fatal(err) } fmt.Printf("Received packet: %s\n", string(buffer[:n])) } } func htons(i uint16) uint16 { return (i<<8)&0xff00 | i>>8 }中无法将 '(*syscall.Sockaddr)(unsafe.Pointer(&sa))' (类型 *syscall.Sockaddr) 用作类型 Sockaddr
最新发布
07-12

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值