如何正确理解和使用 Golang 中 nil ?

目录

指针中的 nil

切片中的 nil

map 中的 nil

通道中的 nil

函数中的 nil

接口中的 nil

避免 nil 相关问题的最佳实践

小结


在 Golang 中,nil 是一个预定义的标识符,在不同的上下文环境中有不同的含义,但通常表示“无”、“空”或“零值”。nil 可以赋值给指针、切片、map、通道、函数以及接口类型的变量。理解 nil 的含义对于编写出健壮的 Go 程序至关重要,如果不能正确地处理 nil 可能会导致意外的问题。

指针中的 nil

在 Go 中,指针是基础类型,保存了变量的内存地址。当一个指针被声明但没有被初始化时,值就是nil。示例代码如下:

package mainimport "fmt"func main() {var ptr *intfmt.Println(ptr == nil) // true
}

如果引用一个 nil 指针,会导致 panic。因此,在进行指针操作之前,一定要先判断指针是否为 nil。

切片中的 nil

切片是一个动态数组,由一个底层数组和一组描述切片属性的信息组成。当一个切片被声明但没有被初始化时,值就是 nil。示例代码如下:

package mainimport "fmt"func main() {var s []int  // 声明一个整型切片,初始值为nilfmt.Println(s == nil) // true
}

nil 切片没有指向任何有效的底层数组,长度(len)和容量(cap)都是 0。但是 nil 切片和空切片(make([]int, 0) 或 []int{})是不同的。nil 切片在没有被分配空间之前不占用内存,而空切片虽然长度为0,但是已经有了一个指向底层数组的指针,这个数组的长度为 0。

map 中的 nil

map 用于存储键值对集合,其中键是唯一的。当一个 map 被声明但没有初始化,值就是 nil。这意味着没有分配内存空间,不能被直接使用。示例代码如下:

package mainimport "fmt"func main() {var myMap map[string]intfmt.Println(myMap == nil)
}

如果往 nil map 中写入数据会导致 panic,因为 nil map 没有底层数据结构来存储数据。但是从 nil map 中读取数据不会导致错误,只是简单地返回对应类型的零值。

nil map 和没有任何键值对的 map(空 map)是不同的。nil map 不能被用来存储键值对,而空 map 是已经被初始化了但是没有元素的 map。例如:

// nil map
var nilMap map[string]int// 空 map
emptyMap := make(map[string]int)

可以对空 map 进行操作,如添加、删除键值对,但是对 nil map 进行这些操作会导致 panic。

通道中的 nil

通道是 Go 语言提供的一种同步原语,用于在 Go 协程(goroutines)之间传递消息。当一个通道被声明但没有被初始化时,值就是nil。示例代码如下:

package mainimport "fmt"func main() {var ch chan int        // 声明一个整型通道,初始值为nilfmt.Println(ch == nil) // true
}

往 nil 通道发送或接收数据都会永远阻塞,因为 nil 通道既不会被关闭,也没有其他协程来进行发送或接收操作。但是 nil 通道在 select 语句中有特殊用途,可以用于禁用 select 语句中的某个分支。

函数中的 nil

在 Go 中,函数也是一种类型,可以使用 nil 来表示一个未初始化的函数。示例代码如下:

package mainimport "fmt"func main() {var fn func(int) int   // 声明一个函数类型,初始值为nilfmt.Println(fn == nil) // true
}

调用一个 nil 函数会导致 panic。

接口中的 nil

interface 是 Go 中的一个重要特性,代表了一种抽象的数据类型。当声明一个新的 interface 变量但并未做具体的实现时,值就是 nil。例如:

package mainimport "fmt"func main() {var i interface{}fmt.Println(i == nil) // true
}

在 Go 的内部,interface{} 类型的变量由两部分组成:类型(Type)和值(Value)。当一个 interface{} 变量既没有类型也没有值时才是 nil。看如下的例子:

package mainimport "fmt"type MyInterface interface {Method()
}type MyType struct{}func (mt *MyType) Method() {}func main() {var mt *MyType = nilvar i MyInterface = mtfmt.Println(i == nil)
}

尽管 mt 是一个 nil 指针,当将其赋值给接口类型 i 时,i 仍然包含了 MyType 的类型信息,因此 i 并不是 nil。

避免 nil 相关问题的最佳实践

  • 在使用指针、切片、map、通道和函数类型的变量之前,先检查是否为 nil。
  • 理解零值和 nil 的区别,对于某些类型(如切片、map、通道和接口),nil 代表它们的零值。但是一个类型的零值不一定是 nil(例如数值类型和结构体类型)。
  • 如果函数返回一个接口类型,避免返回具体类型的 nil 指针,可能会导致接口的值不是 nil 而引起混淆。
  • 当函数返回错误时,如果没有错误发生,应该返回 nil 而不是错误类型的 nil 实例。
  • 关闭文件、数据库连接等资源之前,检查是否为 nil,以避免 nil 指针解引用。

小结

nil 在 Golang 中是一个非常重要的概念,深入理解 nil 在 Go 语言中的应用方法,对于编写高质量的 Go 代码非常重要。希望本文能够帮助你更好地掌握 nil 的相关知识。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/225240.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LeetCode 2415. 反转二叉树的奇数层:深度优先搜索(DFS)

【LetMeFly】2415.反转二叉树的奇数层:深度优先搜索(DFS) 力扣题目链接:https://leetcode.cn/problems/reverse-odd-levels-of-binary-tree/ 给你一棵 完美 二叉树的根节点 root ,请你反转这棵树中每个 奇数 层的节点值。 例如&#xff0c…

部署LVS的NET模式

实验准备 #负载调度器# 192.168.116.40 #内网 12.0.0.100 #外网 先添加双网卡 #web服务器# 192.168.116.20 #web1 192.168.116.30 #web2 #nfs共享服务# 192.168.116.10 #nfs systemctl stop firewalld setenforce 0 1.nfs共享文件 1…

Axure元件库的介绍以及个人简介和登录界面案例展示

目录 一. 元件介绍 二. 基本元件的使用 2.1 形状元件 2.2 图片元件 2.3 占位符 2.4 文本 2.5 线段元件 2.6 热区文件 三. 表单元件的使用 3.1 文本框 3.2 文本域 3.3 下拉列表 3.4 列表框 3.5 复选框 3.6 单选按钮 四. 菜单与表格元件的使用 4.1 树 4.2 表格…

Cmake基础(1)

什么是cmake 众所周知,对于C/C,不同的IDE的project文件是不同的,VS中叫做vcproject。在linux中make工具叫做makefile,codeblock中叫做cpb。而cmake是一个通用的project组织方式,cmake的项目文件cmakelists.txt可以转成…

持续集成交付CICD:GitLabCI操作Harbor仓库

目录 一、实验 1.GitLabCI操作Harbor仓库 二、问题 1.gitlab-runner连接docker daemon报错 一、实验 1.GitLabCI操作Harbor仓库 (1)修改GitLabCI共享库代码并提交到mater CI.yaml .pipelineInit:tags:- buildstage: .prevariables:GIT_CHECKOUT: …

Qt图像处理-基于OpenCv的图像二值化处理

本文讲解Qt图像处理-基于OpenCv的图像二值化处理 一、概述 图像二值化原理 图像二值化就是将图像上的像素点的灰度值设置为0或255,也就是将整个图像呈现出明显的黑白效果的过程。图像的二值化使图像中数据量大为减少,从而能凸显出目标的轮廓。 要得到二值化图像,首先要把…

Apache Avro编程快速入门

Maven配置 添加Avro依赖 <dependency><groupId>org.apache.avro</groupId><artifactId>avro</artifactId>

金蝶云星空协同开发环境应用内执行单据类型脚本

文章目录 金蝶云星空协同开发环境应用内执行单据类型脚本业务界面查询单据类型表数据导出数据执行数据库脚本单据类型xml检验是否执行成功检查数据库检查业务数据 金蝶云星空协同开发环境应用内执行单据类型脚本 业务界面 查询单据类型表数据 先使用类型中文在单据类型多语言…

std::iota 函数简单使用

std::iota 是 C 标准库中的一个算法&#xff0c;位于 <numeric> 头文件中。它的作用是用一个连续的范围内的递增序列填充容器。 函数签名如下&#xff1a; template< class ForwardIt, class T > void iota( ForwardIt first, ForwardIt last, T value ); 其中&…

C++使用回调函数的两种方式

一.函数指针 #include <iostream>typedef void (*callback)(int ,int); class MyTest { public:void setCallback(callback cb){m_callback = cb;}void add(int a, int b){m_callback(a, b);}private:callback m_callback; };void onCallback(int a, int b) {std::cout …

[C++]——学习模板

了解模板——初阶 前言&#xff1a;一、模板1.1 什么是模板1.2 模板的概念1.3 模板可以做什么1.4 泛型模板 二、函数模板2.1 函数模板概念和格式2.2 函数模板原理2.3 函数模板实例化2.3.1 隐式实例化2.3.2 显式实例化 2.4 模板参数的匹配原则2.5 函数模板声明定义分离 三、类模…

若依框架springboot——修改前端图片上传样式

简述 使用过若依框架的&#xff0c;一定知道若依前端框架上传图片的样式&#xff0c;是一个正方形加号图片&#xff0c;但是如果你要使用自定义样式呢。 比如将下面这个图进行修改呢 修改后的样式 你可以直接找到element-ui 修改上传图片的组件&#xff0c;也可以加入新的组…

二进制to十六进制

输入小于等于十六位的二进制数据&#xff0c;输出十六进制数据&#xff1b; #include <stdio.h> #include <stdlib.h> #include <math.h>int main(void) {char arr[16] { 0 }; int array[16] { 0 }; int hex[4] { 0 };int i 0; int num 0;scanf("…

AcWing 1250. 格子游戏(并查集)

题目链接 活动 - AcWing本课程系统讲解常用算法与数据结构的应用方式与技巧。https://www.acwing.com/problem/content/1252/ 题解 当两个点已经是在同一个连通块中&#xff0c;再连一条边&#xff0c;就围成一个封闭的圈。一般用x * n y的形式将&#xff08;x, y&#xff0…

CentOS 防火墙管理及使用的redis基本常用命令

文章目录 防火墙管理使用systemctl管理防火墙启动、关闭使用firewalld-cmd配置访问防火墙策略firewalld配置文件修改限制来源IP docker使用 redis 防火墙管理 需要关闭防火墙或者开启对应端口 使用systemctl管理防火墙启动、关闭 启动防火墙&#xff1a; systemctl start fi…

设计模式——桥接模式(结构型)

引言 桥接模式是一种结构型设计模式&#xff0c; 可将一个大类或一系列紧密相关的类拆分为抽象和实现两个独立的层次结构&#xff0c; 从而能在开发时分别使用。 问题 抽象&#xff1f; 实现&#xff1f; 听上去挺吓人&#xff1f; 让我们慢慢来&#xff0c; 先考虑一个简单的…

物联网智能仓库解决方案

物联网智能仓库解决方案是一种基于物联网技术的仓库管理系统&#xff0c;通过自动化设备、智能化管理系统和大数据分析等技术&#xff0c;实现仓库的智能化运营和管理。 物联网智能仓库解决方案包括&#xff1a; 仓库设备自动化&#xff1a;通过自动化设备和技术&#xff0c;实…

18--日志(了解)

1、日志 1.1 什么是日志 日志用来记录用户操作、系统运行状态等&#xff0c;是一个系统的重要组成部分。对于一些简单的小程序&#xff0c;可能并不需要在如何记录日志的问题上花费太多精力。但是对于作为基础平台为很多产品提供服务的后端程序&#xff0c;就必须要考虑如何依…

KVM虚拟机console使用

注意这些设置都在你要进入虚拟机里设置&#xff0c;不是在你的物理机设置 首先debian12 需要设置 grep ttyS0 /etc/securetty #没有则加上 echo ttyS0 >> /etc/securetty #启动 systemctl start serial-gettyttyS0 systemctl enable serial-gettyttyS0#CentOS Stream …

Springboot+vue的公寓报修管理系统(有报告)。Javaee项目,springboot vue前后端分离项目

演示视频&#xff1a; Springbootvue的公寓报修管理系统&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot vue前后端分离项目 项目介绍&#xff1a; 本文设计了一个基于Springbootvue的前后端分离的公寓报修管理系统&#xff0c;采用M&#xff08;model&…