作者:ReganYueGo语言学习查缺补漏ing Day3
来源:恒生LIGHT云社区
零、前言 因为笔者基础不牢,在使用Go语言的时候经常遇到很多摸不着头脑的问题,所以笔者下定决心好好对Go语言进行查漏补缺,本【Go语言查缺补漏ing】系列主要是帮助新手Gopher更好的了解Go语言的易错点、重难点。希望各位看官能够喜欢,点点赞、关注一下呗!
一、结构体的比较问题 我们先来看一段关于结构体的比较的代码:
package mainimport "fmt"func main() {
struct1 := struct {
ageint
name string
sex bool
}{age: 18, name: "搏达", sex:false}struct2 := struct {
ageint
name string
sex bool
}{age: 21, name: "Regan Yue", sex:true}if struct1 == struct2 {
fmt.Println("struct1 == struct2")
}struct3 := struct {
age int
people map[string]bool
}{age: 31, people: map[string]bool{"搏达": false}}struct4 := struct {
age int
people map[string]bool
}{age: 21, people: map[string]bool{"ReganYue": false}}if struct3 == struct4{
fmt.Println("struct3 == struct4")
}
}
你觉得编译时会报错吗?
文章图片
文章图片
啊这,出大问题。它说包含map的结构体不能被比较。
是的,只有包含bool类型、数值型、字符串、指针、数组等类型的结构体才能比较,而包含slice、map、函数其中任一项的结构体均不能做比较。
还有几点值得注意的是:
- 结构体之间只能比较它们是否相等,而不能比较它们的大小。
- 只有所有属性都相等而属性顺序都一致的结构体才能进行比较。
package mainimport "fmt"func main() {
s1 := struct {
name string
age int64
}{name:"Regan",age:21}
s2 := &s1
fmt.Println(s2.name)
fmt.Println((*s2).name)
fmt.Println((&s2).name)
}
你觉得哪里会报错?
文章图片
哦!是(&s2).name有点问题.....
因为&是取地址符,*是指针解引用符。所以将指针解引用才能访问到成员变量。
你会好奇为什么指针不解引用也能获取成员变量吗?其实是进行解引用了,这就是我们GO语言的强大之处了,它能够自动进行解引用,但是遗憾的是,只能解引用乙级指针,如果是多级指针,就得靠你自己了。
三、类型别名和类型定义的区别
package mainimport "fmt"type Int1 int
type Int2 = intfunc main() {
var i = 0
var i1 Int1 = i
var i2 Int2= i
fmt.Println(i, i1, i2)
}
是不是一眼就看出来问题所在了?
不过他们放在一起,你能马上分别哪个是类型别名,哪个是类型定义吗?
type Int1 int
是定义了基于int类型创建了一个新类型,而
type Int2 = int
【Go语言学习查缺补漏ing Day3】是将Int2作为int类型的别名。
所以将会报如下错误:
文章图片
这是因为Go是强类型语言,所以我们需要将int强制转换Int1类型就可以通过编译了。
四、切片的一个注意事项
package mainimport "fmt"func main() {
a := []int{7, 8, 9}
fmt.Printf("%+v\n", a)
fun1(a)
fmt.Printf("%+v\n", a)
fun2(a)
fmt.Printf("%+v\n", a)
}
func fun1(a []int) {
a = append(a, 5)
}
func fun2(a []int) {
a[0] = 8
}
看一看运行结果:
[7 8 9]
[7 8 9]
[8 8 9]
什么?!! 没有append进去?这是因为append可能会导致内存的重新分配,我们就不能再确定append之前的切片与之后的切片是不是引用的相同的内存空间。
所以说这个执行fun1来append之后的切片a与外面的切片a不是同一个,所以查看外面的a,可以发现外面切片的内容没有改变。
五、Golang字符串连接的几种方式
package mainimport (
"bytes"
"fmt"
"strings"
)func main() {
str1 := "abc"
str2 := "def"
strs := []string{"Regan","Yue"}
fmt.Println(str1+str2)
sprintf := fmt.Sprintf("abc%s", "accc")
fmt.Println(sprintf)
join := strings.Join(strs, "-")
fmt.Println(join)var buffer bytes.Buffer
buffer.WriteString("hello")
buffer.WriteString("world")
fmt.Println(buffer.String())
}
这里提到了四种连接字符串的方式,第一种利用+号直接连接,第二种利用 fmt.Sprintf(),第三种是利用strings.Join(),第四种是利用buffer.WriteString()来进行字符串连接。