Go语言数据类型详细介绍
IT行业相对于一般传统行业,发展更新速度更快,一旦停止了学习,很快就会被行业所淘汰。所以我们需要踏踏实实的不断学习,精进自己的技术,尤其是初学者。今天golang学习网给大家整理了《Go语言数据类型详细介绍》,聊聊类型、数据,我们一起来看看吧!
一、Go 语言两大数据类型
Go 语言数据类型包含基础类型和复合类型两大类。
基础类型包括:
- 布尔类型 bool。
- 数值类型 int,int8,int16,int32,int64,float32,float64。
- uint8,uint16,uint32,uint64。
- 字符串 string,byte,rune。
复合类型包括:
- 指针、数组、切片、字典、通道、结构体、接口。
下面用代码加注释的方式说明:
二、基础类型
声明一个bool类型的变量,只有true和false两个值,初始值为false
var isLogin bool // 声明一个int类型的变量,初始值为0 //(int8,int16,int32 uint8,uint16....类似,只是有符号和无符号的区别) // 最常用的就是int,如果需要更大的范围可以用int64或者uint64 var count int // 声明一个string类型的变量,初始值为"" var s string //声明一个byte类型的变量 var b byte //声明一个rune类型的变量 //int32的别名,表示单个Unicode字符 var r rune
三、复合类型
指针
go的指针和c语言的指针类型,都是表示一个变量的地址,不同的是,go的指针要比c的指针简单的多,老规矩,代码注释,如下:
package main
import "fmt"
func main() {
var count = 100 //定义变量count
var ptr *int //定义一个指针ptr,此指针可以保存int类型变量的地址
ptr = &count //ptr保存的是变量count的地址, & 符号是取变量地址的符号
fmt.Println("count=",count) //打印count的值
fmt.Println("ptr=", *ptr) //打印ptr指向的变量的值,此句打印100
}
运行结果如下:
count= 100
ptr= 100
四、数组(array)
数组为一组相同数据类型数据的集合,大小固定,不能更改,每个元素称为element,声明的数组元素默认值都是对应类型的0值。而且数组在Go语言中是一个值类型(value type)所有值类型变量在赋值和作为参数传递时都会产生一次复制动作,即对原值的拷贝
package main
import "fmt"
func main() {
// 1.声明后赋值 (var [])
var arr [2]int // 数组元素的默认值都是 0
fmt.Println(arr) // 输出:[0 0]
arr[0] = 1
arr[1] = 2
fmt.Println(arr) // 输出:[1 2]
// 2.声明并赋值 (var = []{元素1,元素2,...})
var intArr = [2]int{1, 2}
strArr := [3]string{`aa`, `bb`, `cc`}
fmt.Println(intArr) // 输出:[1 2]
fmt.Println(strArr) // 输出:[aa bb cc]
// 3.声明时不设定大小,赋值后语言本身会计算数组大小
// var [] = [...]{元素1,元素2,...}
var arr1 = [...]int{1, 2}
arr2 := [...]int{1, 2, 3}
fmt.Println(arr1) // 输出:[1 2]
fmt.Println(arr2) // 输出:[1 2 3]
//arr1[2] = 3 // 编译报错,数组大小已设定为2
// 4.声明时不设定大小,赋值时指定索引
// var [] = [...]{索引1:元素1,索引2:元素2,...}
var arr3 = [...]int{1: 22, 0: 11, 2: 33}
arr4 := [...]string{2: "cc", 1: "bb", 0: "aa"}
fmt.Println(arr3) // 输出:[11 22 33]
fmt.Println(arr4) // 输出:[aa bb cc]
// 遍历数组
for i := 0; i
<h2>五、切片(slice)</h2>
<p>因为数组的长度定义后不可修改,所以需要切片来处理可变长数组数据。切片可以看作是一个可变长的数组,是一个引用类型。</p>
<p><strong>它包含三个数据:</strong></p>
- 指向原生数组的指针
- 切片中的元素个数
- 切片已分配的存储空间大小
注:了解c++和java的同学,可以参考vector和List,切片就是类似这两个数据结构,直接上代码:
package main
import "fmt"
func main() {
var sl []int // 声明一个切片
sl = append(sl, 1, 2, 3) // 往切片中追加值
fmt.Println(sl) // 输出:[1 2 3]
var arr = [5]int{1, 2, 3, 4, 5} // 初始化一个数组
var sl1 = arr[0:2] // 冒号:左边为起始位(包含起始位数据),右边为结束位(不包含结束位数据);不填则默认为头或尾
var sl2 = arr[3:]
var sl3 = arr[:5]
fmt.Println(sl1) // 输出:[1 2]
fmt.Println(sl2) // 输出:[4 5]
fmt.Println(sl3) // 输出:[1 2 3 4 5]
sl1 = append(sl1, 11, 22) // 追加元素
fmt.Println(sl1) // 输出:[1 2 11 22]
}
使用make关键字直接创建切片,
语法:make([]类型, 大小,预留空间大小),make() 函数用于声明slice切片、map字典、channel通道。
如下:
package main
import "fmt"
func main() {
var s1 = make([]int, 5) // 定义元素个数为5的切片
s2 := make([]int, 5, 10) // 定义元素个数5的切片,并预留10个元素的存储空间(预留空间不知道有什么用?)
s3 := []string{`aa`, `bb`, `cc`} // 直接创建并初始化包含3个元素的数组切片
fmt.Println(s1, len(s1)) // 输出:[0 0 0 0 0] 5
fmt.Println(s2, len(s2)) // 输出:[0 0 0 0 0] 5
fmt.Println(s3, len(s3)) // [aa bb cc] 3
s1[1] = 1 // 声明或初始化大小中的数据,可以指定赋值
s1[4] = 4
//s1[5] = 5 // 编译报错,超出定义大小
s1 = append(s1, 5) // 可以追加元素
fmt.Println(s1, len(s1)) // 输出:[0 1 0 0 4 5] 6
s2[1] = 1
s2 = append(s2, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11)
fmt.Println(s2, len(s2)) // 输出:[0 1 0 0 0 1 2 3 4 5 6 7 8 9 10 11] 16
// 遍历切片
for i := 0; i
<h2>六、字典/映射(map)</h2>
<p><code>map</code>是一种键值对的无序集合,与 <code>slice </code>类似也是一个引用类型。map 本身其实是个指针,指向内存中的某个空间。</p>
<p>声明方式与数组类似,声明方式:var 变量名 map[key类型值类型 或直接使用 make 函数初始化:make(map[key类型]值类型, 初始空间大小)。</p>
<p>其中key值可以是任何可以用==判断的值类型,对应的值类型没有要求。</p>
<p><strong>直接上代码,如下:</strong></p>
<pre class="brush:plain;">package main
import (
"fmt"
"unsafe"
)
func main() {
// 声明后赋值
var m map[int]string
fmt.Println(m) // 输出空的map:map[]
//m[1] = `aa` // 向未初始化的map中赋值报错:panic: assignment to entry in nil map
// 声明并初始化,初始化使用{} 或 make 函数(创建类型并分配空间)
var m1 = map[string]int{}
var m2 = make(map[string]int)
m1[`a`] = 11
m2[`b`] = 22
fmt.Println(m1) // 输出:map[a:11]
fmt.Println(m2) // 输出:map[b:22]
// 初始化多个值
var m3 = map[string]string{"a": "aaa", "b": "bbb"}
m3["c"] = "ccc"
fmt.Println(m3) // 输出:map[a:aaa b:bbb c:ccc]
// 删除 map 中的值
delete(m3, "a") // 删除键 a 对应的值
fmt.Println(m3) // 输出:map[b:bbb c:ccc]
// 查找 map 中的元素
v, ok := m3["b"]
if ok {
fmt.Println(ok)
fmt.Println("m3中b的值为:", v) // 输出:m3中b的值为: bbb
}
// 或者
if v, ok := m3["b"]; ok { // 流程处理后面讲
fmt.Println("m3中b的值为:", v) // 输出:m3中b的值为: bbb
}
fmt.Println(m3["c"]) // 直接取值,输出:ccc
// map 中的值可以是任意类型
m4 := make(map[string][5]int)
m4["a"] = [5]int{1, 2, 3, 4, 5}
m4["b"] = [5]int{11, 22, 33}
fmt.Println(m4) // 输出:map[a:[1 2 3 4 5] b:[11 22 33 0 0]]
fmt.Println(unsafe.Sizeof(m4)) // 输出:8,为8个字节,map其实是个指针,指向某个内存空间
}
七、通道(channel)
说到通道 channel,则必须先了解下 Go 语言的 goroutine 协程(轻量级线程)。channel就是为 goroutine 间通信提供通道。goroutine是 Go 语言提供的语言级的协程,是对 CPU 线程和调度器的一套封装。
channel 也是类型相关的,一个 channel 只能传递一种类型的值。
声明:var 通道名 chan 通道传递值类型 或 make 函数初始化:make(chan 值类型, 初始存储空间大小)
说白了,通道类似消息队列,主要应用在并发编程里面比较多,
直接上代码,如下:
package main
import (
"fmt"
"time"
)
func main() {
var ch1 chan int // 声明一个通道
ch1 = make(chan int) // 未初始化的通道不能存储数据,初始化一个通道
ch2 := make(chan string, 2) // 声明并初始化一个带缓冲空间的通道
// 通过匿名函数向通道中写入数据,通过
<p><strong>运行结果如下:</strong></p>
<blockquote><p>// ------ 输出:--------<br>1<br>a<br>写入: 1<br>写入: 2<br>写入: 3<br>读取: 1<br>读取: 2<br>读取: 3<br>写入: 4<br>写入: 5<br>写入: 6<br>读取: 4<br>读取: 5<br>读取: 6<br>写入: 7<br>写入: 8<br>写入: 9<br>读取: 7<br>读取: 8<br>读取: 9</p></blockquote>
<p><code>goroutine </code>和 <code>channel</code> 的详细用法会有相应的博客专门来讲这一章节,具体可在我的个人主页里面,找一下相关的博客参考。</p>
<h2>八、结构体(struct)</h2>
<p>结构体是一种聚合的数据类型,是由零个或多个任意类型的值聚合成的实体。每个值称为结构体的成员,和java中的class是一个意思:</p>
<pre class="brush:plain;">package main
import "fmt"
// 定义一个结构体 person
type person struct {
name string
age int
}
func main() {
var p person // 声明一个 person 类型变量 p
p.name = "max" // 赋值
p.age = 12
fmt.Println(p) // 输出:{max 12}
p1 := person{name: "mike", age: 10} // 直接初始化一个 person
fmt.Println(p1.name) // 输出:mike
p2 := new(person) // new函数分配一个指针,指向 person 类型数据
p2.name = `张三`
p2.age = 15
fmt.Println(*p2) // 输出:{张三 15}
}
九、接口(interface)
接口用来定义行为。Go 语言不同于面向对象语言,没有类的概念,也没有传统意义上的继承。Go 语言中的接口,用来定义一个或一组行为,某些对象实现了接口定义的行为,则称这些对象实现了(implement)该接口,类型即为该接口类型。
定义接口也是使用 type 关键字,格式为:
// 定义一个接口
type InterfaceName interface {
FuncName1(paramList) returnType
FuncName2(paramList) returnType
...
}
实列:
package main
import (
"fmt"
"strconv"
)
// 定义一个 Person 接口
type Person interface {
Say(s string) string
Walk(s string) string
}
// 定义一个 Man 结构体
type Man struct {
Name string
Age int
}
// Man 实现 Say 方法
func (m Man) Say(s string) string {
return s + ", my name is " + m.Name
}
// Man 实现 Walk 方法,strconv.Itoa() 数字转字符串
func (m Man) Walk(s string) string {
return "Age: " + strconv.Itoa(m.Age) + " and " + s
}
func main() {
var m Man // 声明一个类型为 Man 的变量
m.Name = "Mike" // 赋值
m.Age = 30
fmt.Println(m.Say("hello")) // 输出:hello, my name is Mike
fmt.Println(m.Walk("go work")) // 输出:Age: 30 and go work
jack := Man{Name: "jack", Age: 25} // 初始化一个 Man 类型数据
fmt.Println(jack.Age)
fmt.Println(jack.Say("hi")) // 输出:hi, my name is jack
}
十、错误(error)
error 类型本身是 Go 语言内部定义好的一个接口,接口里定义了一个 Error() 打印错误信息的方法,源码如下:
type error interface {
Error() string
}
自定义错误信息:
package main
import (
"errors"
"fmt"
)
func main() {
// 使用 errors 定制错误信息
var e error
e = errors.New("This is a test error")
fmt.Println(e.Error()) // 输出:This is a test error
// 使用 fmt.Errorf() 定制错误信息
err := fmt.Errorf("This is another error")
fmt.Println(err) // 输出:This is another test error
}
文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Go语言数据类型详细介绍》文章吧,也可关注golang学习网公众号了解相关技术文章。
Go语言流程控制详情
- 上一篇
- Go语言流程控制详情
- 下一篇
- Jaeger Client Go入门并实现链路追踪
-
- Golang · Go教程 | 3小时前 | 格式化输出 printf fmt库 格式化动词 Stringer接口
- Golangfmt库用法与格式化技巧解析
- 140浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang配置Protobuf安装教程
- 147浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang中介者模式实现与通信解耦技巧
- 378浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang多协程通信技巧分享
- 255浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang如何判断变量类型?
- 393浏览 收藏
-
- Golang · Go教程 | 3小时前 |
- Golang云原生微服务实战教程
- 310浏览 收藏
-
- Golang · Go教程 | 4小时前 |
- Golang迭代器与懒加载结合应用
- 110浏览 收藏
-
- Golang · Go教程 | 4小时前 | 性能优化 并发安全 Golangslicemap 预设容量 指针拷贝
- Golangslicemap优化技巧分享
- 412浏览 收藏
-
- Golang · Go教程 | 4小时前 |
- Golang代理模式与访问控制实现解析
- 423浏览 收藏
-
- Golang · Go教程 | 5小时前 |
- Golang事件管理模块实现教程
- 274浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3164次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3376次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3405次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4509次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3785次使用
-
- MySQL事务处理特性的实现原理
- 2023-01-20 138浏览
-
- 一文带你了解Go语言中的类型断言和类型转换
- 2022-12-28 196浏览
-
- Golang中map数据类型的使用方法
- 2022-12-30 443浏览
-
- 一文详解Golang中的切片数据类型
- 2022-12-31 171浏览
-
- MySql恢复数据方法梳理讲解
- 2023-01-07 424浏览

