热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Go语言教程(1)——类型系统

类型系统Go语言是更好的C语言,很多思想来源于C语言,毕竟Go的设计者就是C的设计者在几十年之后再创新高。变量的声明引入和JavaScript一样的关键字var,不一样的机制,Ja
类型系统

Go 语言是更好的 C 语言,很多思想来源于 C 语言,毕竟 Go 的设计者就是 C 的设计者在几十年之后再创新高。

变量的声明引入和 Javascript 一样的关键字 var,不一样的机制,Javascript 中声明变量时没有类型,Go 语言声明的变量在后面要带上类型。

变量

var v int // 声明变量 v,自动初始化值为零值(int 类型为 0)
v = 1 // 给变量 v 赋值
v := 1 // 声明变量 v,并初始化值为 1,编译器自动推导类型为 int

上面两种方式完全等价,:= 的写法更简洁,也是官方推荐的写法,编译器可以自动从右值推导出声明的变量是哪种类型,v 必须是没被声明过的变量。

多重赋值

基本的赋值方法都是类似的,但 Go 提供了动态语言中才有的多重赋值功能

i, j = j, i 交换变量如此简单,不需要引入中间变量。

匿名变量

函数的返回值中可能只有一个感兴趣的,别的值都不需要,是否还需要定义变量去接收这几个返回值?使用匿名变量!

func GetName() (firstName, lastName, nickName string) {
return "May", "Chan", "Chibi"
}
_, _, nickName := GetName()
常量

常量值必须是编译期可确定的数字、字符串、布尔值。即常说的字符字面量(literal),根据字面量可以推测出常量的类型。

const x,y int = 1, 2 // 多常量初始化
const Pi float64 = 3.1415926
const zero = 0.0 // 类型推断
const ( // 常量组
size int64 = 1024
eof = -1 // 类型推断
Eof // 在常量组中,如果不提供类型和初始化值,那么视作与上⼀常量相同。
)

iota

iota 比较特殊,可以认为是编译器内置的一个寄存器,在每一个 const 关键字出现时被重置为 0,然后在下一个 const 出现时,每出现一次 iota,其所代表的数字就会增 1

const (
Sunday = iota // 0
Monday // 1
Tuesday // 2
)

枚举

Go 语言中不存在枚举类型,可以通过自定义类型的方式构造。


type Color int // 定义类型 Color,传递 Color 的地发不能传递 int
const (
Black Color = iota // 定义常量 Black = 0
Red
Blue
)
func test(c Color) {}
func main() {
c := Black
test(c)
x := 1
test(x) // Error: cannot use x (type int) as type Color in function argument
test(1) // 常量会被编译器自动转换
基本类型

类型⻓度默认值(零值)说明
bool1false
byte10uint8
rune40Unicode Code Point, int32
int, uint4 或 8032 或 64 位
int8, uint810-128 ~ 127, 0 ~ 255
int16, uint1620-32768 ~ 32767, 0 ~ 65535
int32, uint3240-21亿 ~ 21 亿, 0 ~ 42 亿
int64, uint6480
float3240.0
float6480.0
complex648
complex12816
uintptr4 或 8⾜以存储指针的 uint32 或 uint64 整数
array值类型
struct值类型
string“”UTF-8 字符串
slicenil引⽤类型
mapnil引⽤类型
channelnil引⽤类型
interfacenil接⼝
functionnil函数

int、uint、uintpter 类型在 32 位系统上一般是 32 位,在 64 位系统上是 64 位。

格式化打印类型信息和类型的技巧。

var (
ToBe bool = false
MaxInt uint64 = 1<<64 - 1
)
func main() {
const f = "%T(%v)\n"
fmt.Printf(f, ToBe, ToBe)
fmt.Printf(f, MaxInt, MaxInt)
}
引用类型

引用类型包括 slice、map 和 channel。它们有复杂的内部结构,除了申请内存外,还需要初始化相关的属性。

内置函数 new ,计算类型大小,为其分配零值内存,并返回指向内存的指针

内置函数 make,会被编译器翻译成具体的创建函数,由其分配内存和初始化成员结构,返回对象而非指针

a := []int{0, 0, 0} // 提供初始化表达式。
a[1] = 10
b := make([]int, 3) // 初始化长度为 3 的 slice
b[1] = 10
c := new([]int)
c[1] = 10 // Error: invalid operation: c[1] (index of type *[]int)
类型转换

不支持隐式类型转化,即便是从窄向宽转换也不行。

表达式 T(v) 将值 v 转换为类型 T

Bool

Bool 类型不接受其他类型的值,不支持强制类型转换,其他类型不能当 Bool 值使用。

Float

浮点数比较不建议使用 ==,可以使用函数判断精度。

import "math"
// p 为自定义精度
func IsEqual(f1, f2, p float64) bool {
return math.Fdim(f1, f2) }
字符串

字符串内容在初始化后不可变!

  • 默认值是空字符串 ""
  • 使用索引号访问的是某个字节 s[i]
  • 不能用序号获取字节元素指针,&s[i] 非法
  • 不可变类型,无法修改字节数组
  • 字节数组尾部不包含 NULL

struct String {
byte* str; // 指向字节数组的指针
intgo len; // 长度
}

⽀持⽤两个索引号返回⼦串。⼦串依然指向原字节数组,仅修改了指针和⻓度属性

s := "Hello, World!"
s1 := s[:5] // Hello
s2 := s[7:] // World!
s3 := s[1:5] // ello

字符串遍历

// 以字节数组遍历
str := "Hello,世界"
n := len(str)
for i := 0; i ch := str[i] // 取出下标上的字符,类型为 byte
}
// 每个中文字符在 UTF-8 占 3 个字节
// 以 Unicode 字符遍历
for i, ch := range str {
fmt.Println(i, ch) // ch 的类型为 rune
}

修改字符串

单引号字符常量表⽰ Unicode Code Point,⽀持 \uFFFF、\U7FFFFFFF、\xFF 格式。
对应 rune 类型(int32 的别名,4 字节表示),UCS-4

func main() {
fmt.Printf("%T\n", 'a') var c1, c2 rune = '\u6211', '们'
println(c1 == '我', string(c2) == "\xe4\xbb\xac")
}

输出

int32 // rune 是 int32 的别名
true true

修改字符串,可先将其转换成 []rune[]byte,完成后再转换为 string。⽆论哪种转
换,都会重新分配内存,并复制字节数组

func main() {
s := "abcd"
bs := []byte(s)
bs[1] = 'B'
println(string(bs)) // aBcd u := "电脑"
us := []rune(u)
us[1] = '话'
println(string(us)) // 电话
}
指针

指针类型 *T 是指向类型 T 的指针,零值为 nil

& 取址运算符。

* 间接引用,访问对象。

指针和 C 都是一样的,不同的是 Go 没有指针运算。

可以通过 unsafe.Pointer 和任意类型指针间进⾏转换。

func main() {
x := 0x12345678
p := unsafe.Pointer(&x) // *int -> Pointer
n := (*[4]byte)(p) // Pointer -> *[4]byte 转换成数组指针 // 78 45 34 12
for i := 0; i fmt.Printf("%X ", n[i])
}
}

局部变量的指针

返回局部变量的指针是安全的,编译器会根据需要将其分配在 GC Heap 上。

func test() *int {
x := 100
return &x // 在堆上分配 x 内存。但在内联时,也可能直接分配在⺫标栈。
}

变相实现指针运算

Pointer 转换成 uintptr,可变相实现指针运算。

func main() {
d := struct {
s string
x int
}{"abc", 100} p := uintptr(unsafe.Pointer(&d)) // *struct -> Pointer -> uintptr
p += unsafe.Offsetof(d.x) // uintptr + offset
p2 := unsafe.Pointer(p) // uintptr -> Pointer
px := (*int)(p2) // Pointer -> *int
*px = 200 // d.x = 200 fmt.Printf("%#v\n", d)
}

输出

struct { s string; x int }{s:"abc", x:200}

注意:GC 把 uintptr 当成普通整数对象,它⽆法阻⽌ &#8220;关联&#8221; 对象被回收。

自定义类型

可将类型分为命名和非命名两大类。

命名类型包括 bool、int、string 等。

非命令类型则是 array、slice、map 等和具体元素类型、长度等有关。

具有相同声明的未命名类型被视为同一类型

  • 具有相同基类型指针
  • 具有相同元素类型和长度的 array
  • 具有相同元素类型的 slice
  • 具有相同键值类型的 map
  • 具有相同元素类型和传送方向的 channel
  • 具体相同字段序列(字段名、类型、标签、顺序)的匿名 struct
  • 签名相同的(参数和返回值,不包括参数名称) function
  • 方法集相同的(方法名、方法签名相同,和次序无关) interface

var a struct { x int `a` }
var b struct { x int `ab` }
// cannot use a (type struct { x int "a" }) as type struct { x int "ab" } in assignment
// 结构体标签不同
b = a

Type

可以使用 type 在全局或函数内定义新类型。

func main() {
type bigint int64
var x bigint = 100
println(x)
}

新类型不是原类型的别名,除拥有相同数据存储结构外,它们之间没有任何关系,不会持有原类型任何信息。

除⾮目标类型是未命名类型,否则必须显式转换。

x := 1234
var b bigint = bigint(x) // 必须显式转换,除⾮是常量。
var b2 int64 = int64(b)
// slice 只要存储相同类型的值
var s myslice = []int{1, 2, 3} // 未命名类型,隐式转换。
var s2 []int = s

推荐阅读
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文由编程笔记#小编为大家整理,主要介绍了logistic回归(线性和非线性)相关的知识,包括线性logistic回归的代码和数据集的分布情况。希望对你有一定的参考价值。 ... [详细]
  • 电话号码的字母组合解题思路和代码示例
    本文介绍了力扣题目《电话号码的字母组合》的解题思路和代码示例。通过使用哈希表和递归求解的方法,可以将给定的电话号码转换为对应的字母组合。详细的解题思路和代码示例可以帮助读者更好地理解和实现该题目。 ... [详细]
  • IhaveconfiguredanactionforaremotenotificationwhenitarrivestomyiOsapp.Iwanttwodiff ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 不同优化算法的比较分析及实验验证
    本文介绍了神经网络优化中常用的优化方法,包括学习率调整和梯度估计修正,并通过实验验证了不同优化算法的效果。实验结果表明,Adam算法在综合考虑学习率调整和梯度估计修正方面表现较好。该研究对于优化神经网络的训练过程具有指导意义。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
  • 本文介绍了南邮ctf-web的writeup,包括签到题和md5 collision。在CTF比赛和渗透测试中,可以通过查看源代码、代码注释、页面隐藏元素、超链接和HTTP响应头部来寻找flag或提示信息。利用PHP弱类型,可以发现md5('QNKCDZO')='0e830400451993494058024219903391'和md5('240610708')='0e462097431906509019562988736854'。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 怎么在PHP项目中实现一个HTTP断点续传功能发布时间:2021-01-1916:26:06来源:亿速云阅读:96作者:Le ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 本文介绍了在wepy中运用小顺序页面受权的计划,包含了用户点击作废后的从新受权计划。 ... [详细]
author-avatar
周茜闹心_325
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有