知道了不行,熟悉也不够,要真正进入潜意识,成为条件反射才行!
多读书,多看报,少吃零食,多睡觉。
golang基本语法中比较有含金量的技能点 1、锁 2、反射 3、并发编程
数据与切片区别
1、数组 值传递
2、切片 引用传递 使用append添加数据
3、map 引用传递
4、指针 引用传递
var声明变量 自动推导可以定义局部变量
go语言和python语言一样支持多返回值,也就是说函数可以一次性返回多个值,这和其他语言有着显著的不同。
go语言中数组是值传递,这和其他的语言有着很大的不同,切片是引用传递。
Go 是静态类型语⾔,不能在运⾏期改变变量类型。使⽤关键字 var 定义变量,⾃动初始化为零值。如果提供初始化值,可省略变量类型,由
编译器⾃动推断。
var x int
var f float32 = 1.6
var s = "abc"
在函数内部,可⽤更简略的 “:=” ⽅式定义变量。
func main() {
x := 123
// 注意检查,是定义新局部变量,还是修改全局变量。该⽅式容易造成错误。
}
可⼀次定义多个变量。
var x, y, z int
var s, n = "abc", 123
var (
a int
b float32
)
func main() {
n, s := 0x1234, "Hello, World!"
println(x, s, n)
}
多变量赋值时,先计算所有相关值,然后再从左到右依次赋值。
data, i := [3]int{0, 1, 2}, 0
i, data[i] = 2, 100+l
// (i = 0) -> (i = 2), (data[0] = 100)
特殊只写变量 “_”,⽤于忽略值占位。
func test() (int, string) {
return 1, "abc"
}
func main() {
_, s := test()
println(s)
}
编译器会将未使⽤的局部变量当做错误。
var s string
func main() {
i := 0
// 全局变量没问题。
// Error: i declared and not used。(可使⽤ "_ = i" 规避)
}
注意重新赋值与定义新同名变量的区别。
引⽤类型
引⽤类型包括 slice、map 和 channel。它们有复杂的内部结构,除了申请内存外,还需
要初始化相关属性。
内置函数 new 计算类型⼤⼩,为其分配零值内存,返回指针。⽽ make 会被编译器翻译成具体的创建函数,由其分配内存和初始化成员结构,返回对象⽽⾮指针。
a := []int{0, 0, 0}
a[1] = 10
b := make([]int, 3)
b[1] = 10
c := new([]int)
c[1] = 10
// 提供初始化表达式。
// makeslice
// Error: invalid operation: c[1] (index of type *[]int)
类型转换
不⽀持隐式类型转换,即便是从窄向宽转换也不⾏。
var b byte = 100
// var n int = b
var n int = int(b)
// Error: cannot use b (type byte) as type int in assignment
// 显式转换
使⽤括号避免优先级错误。
*Point(p)
(*Point)(p)
<-chan int(c)
(<-chan int)(c)
// 相当于 *(Point(p))
// 相当于 <-(chan int(c))
同样不能将其他类型当 bool 值使⽤。
a := 100
if a {
println("true")
// Error: non-bool a (type int) used as if condition
}
字符串
字符串是不可变值类型,内部⽤指针指向 UTF-8 字节数组。
• 默认值是空字符串 “”。
• ⽤索引号访问某字节,如 s[i]。
• 不能⽤序号获取字节元素指针,&s[i] ⾮法。
• 不可变类型,⽆法修改字节数组。
• 字节数组尾部不包含 NULL。
struct String
{
byte*
intgo
str;
len;
};
使⽤索引号问字符 (byte)。
s := "abc"
println(s[0] == '\x61', s[1] == 'b', s[2] == 0x63)
输出:
true true true
使⽤ “`” 定义不做转义处理的原始字符串,⽀持跨⾏。
s := `a
b\r\n\x00
c`
println(s)
输出:
a
b\r\n\x00
c
连接跨⾏字符串时,"+" 必须在上⼀⾏末尾,否则导致编译错误。
s := "Hello, " +
"World!"
s2 := "Hello, "
+ "World!"
// Error: invalid operation: + untyped string
⽀持⽤两个索引号返回⼦串。⼦串依然指向原字节数组,仅修改了指针和⻓度属性。
s := "Hello, World!"
s1 := s[:5]
// Hello
s2 := s[7:]
s3 := s[1:5]
// Hello
// World!
// ello
golang中http的函数原型
func HandleFunc(pattern string, handler func(ResponseWriter, *Request))
func ListenAndServe(addr string, handler Handler) error
tcp函数原型
服务端
func Listen(network, address string) (Listener, error)
listener.Accept() Accept() (Conn, error)
客户端
func Dial(network, address string) (Conn, error)
Write(b []byte) (n int, err error)
获取到Conn后就可以从中进行读写
简单tcp服务器
package main
import (
"net"
"fmt"
"io"
)
func main() {
listener, e := net.Listen("tcp", "192.168.28.30:9999")
if e!=nil {
fmt.Println(e)
}
defer listener.Close()
i:=1
for ; ; {
conn, err := listener.Accept()
if err!=nil {
fmt.Println(err)
}
go func() {
buf := make([]byte, 2048)
for ; ; {
n, err2 := conn.Read(buf)
if err2 != nil {
fmt.Println(err2)
}
if err2==io.EOF {
break
}
i++
fmt.Println("==========",i,"===========")
fmt.Println(string(buf[:n]))
}
}()
}
}