2 Go 基础
定义变量:
var varname type
var name1, name2, name3 type
var name type = value
var a,b,c = v1, v2, v3
a,b,c := v1, v2 ,v3
_, b = 1, 2
常量:
const name = value
const Pi float32 = 3.14
const i = 100
内置基础类型:
// Boolean
var isActive = bool // default false
var enabled, disabled = true, false
valid := false
// 数值类型, rune(int32), int8, int16, int32, int64, byte(uint8), uint8, uint16, uint32, uint64
// float32, float64(默认)
// complex64
//字符串, go 字符串都是采用UTF8,用 双引号或者反引号括起来定义,类型是 string
var hello string
var emptyString = ""
s := "hello" // 字符串不可变
c := []byte(s) // 把字符串转换为 []byte 类型
c[0] = 'c'
s2 := string(c) // 转为 string
fmt.Printf("%s\n", s2)
m := `hello
world` // ` 括号来的是Raw string
错误类型:
// go有一个error 类型,package.errors
err:=errors.New("emit macho swarf: elf header corrupted")
if err != nil {
fmt.Print(err)
}
分组声明:
import (
"fmt"
"os"
)
const (
i = 100
pi = 3.14
)
var(
i int
pi float32
)
iota 枚举:
const (
x = iota //x==0
y = iota //y==1
)
设计规则:
- 大写开头的变量可以导出(共有变量)
- 大写字母开头的函数是共有函数
array, slice, map
// array 数组,数组长度也是类型的一部分,[3]int, [4]int 是不同类型
var arr [n]type
var arr [10]int
arr[0] = 42
arr[1] = 13
fmt.Printf("first %d\n", arr[0])
a := [3]int{1, 2, 3}
b := [10]int{1, 2, 3} //其余元素0
c := [...]int{1,2,3} //自动计算个数
doubleArray := [2][4]int{[4]int{1,2,3,4}, [4]int{5,6,7,8}}
// slice 动态『数组』,引用类型,指向一个底层 array。实际场景中使用 slice 要比 array 多
var fslice []int //声明同数组,只是没有长度
slice := []byte {'a','b'}
a := [10]int{1,2,3,4,5,6,7,8,9,10}
var slice []int
slice = a[:]
/* slice 内置函数
len 获取长度
cap 获取最大容量
append 追加一个或多个元素,返回一个 和 slice 类型一样的slice
copy 从 源 slice src 中复制元素到目标 dst,返回复制的元素个数
*/
// map,字典的概念。 map[keyType]valueType
// map 无序,长度不固定,**引用类型**,len(map)返回 key 数量,map 值可以修改。
// map和其他基本类型不同,不是 thread-safe,多个 go-routine 存取用 mutex lock
var numers map[string]int
numbers := make(map[string]int)
numbers["one"] = 1
numbers["ten"] = 10
fmt.Println("one is ", numbers["three"])
rating := map[string]float32{"c": 5, "Go": 4.5}
goRaking, ok := rating["Go"]
if ok {
fmt.Println("go rating is ", goRaking)
} else {
fmt.Println("no go rating")
}
delete(rating, "C")
make, new 操作:
make 只能用于 内建类型(map/slice/chanell)的内存分配, new 用于各种类型,new 返回指针。 make 返回初始化后的(非零)值。
零值(zero value): 变量未填充之前的默认值,通常是 0
int,int8,int32,int64 0
uint 0x0
rune 0
byte 0x0
float32, float64 0
bool false
string ""
流程控制:
// if(不用括号)
if x > 10 {
//...
} else {
//...
}
if x := computeValue(); x > 10 { //允许声明一个变量,但是只在该条件内作用域
//...
} else {
//...
}
if x == 3 {
//...
} else if x < 3 {
//
} else {
}
// goto,还是少用不用吧
// for
sum := 0
for idx := 0; index < 10; index++ {
sum += idx
}
sum := 1
for ; sum < 1000; {
sum += sum
}
// 实现 while 功能
sum := 1
for sum < 1000 {
sum += sum
}
// break/continue
// for 配合 range 读取 slice 和 map 数据
for k,v := range map {
fmt.Println("key, value", k, v)
}
// switch
i := 6
switch i {
case 4:
//
fallthrough // 强制执行后边的代码
}
函数: func 声明
func funcName(input1 type1, input2 type2) (output1 type1, output2 type2) {
//
return value1, value2
}
func max(a, b int) int {
if a > b {
return a
}
return b
}
func SumAndProduct(A, B int) (add int, Multiplied int) {
add = A + B
Multiplied = A * B
return
}
变参:
func myfunc(arg ...int) {
for _, n := range arg {
fmt.Printf("Add the number is :%d\n", n)
}
} // arg 是一个 int 的 slice
传值和传指针:默认传值,无法修改传入的值。除非使用指针
func add(a int) int {
a = a+1
return a
}
// 如果需要修改值,我们可以传递指针
func add_p(a *int) int {
*a = *a + 1
return *a
} // x :=
/*
传递指针使得多个函数能够操作同一个对象
传指针比较轻量级(8bytes),只是传内存地址,可以用指针传递大结构体。
GO中 channel, slice,map 实现机制类似指针,可以直接传递,而不用取地址。(如果需要修改 slice 长度让需要取地址)
*/
defer: 延迟语句,函数执行到最后时,这些 defer 语句会逆序执行,最后函数返回。用来处理错误,关闭资源
func ReadWrite() bool {
file.Open("file")
defer file.Close()
if failureX {
return false
}
if failureY {
return false
}
return true
}
函数作为值、类型:go 中函数也是一种变量,可以用 type 定义,类型就是所有拥有相同的参数,相同返回值的一种类型
// type typeName func(input1 inputType1, intpu2 inputType2 [, ...]) (result1 resultType1 [, ...])
// 可以把这个类型的函数当做值来传递
package main
import "fmt"
type testInt func(int) bool // 声明一个函数类型
func isOdd(i int) bool {
if i%2 == 0 {
return false
}
return true
}
func filter(slice []int, f testInt) []int {
var result []int
for _, value := range slice {
if f(value) {
result = append(result, value)
}
}
return result
}
func main() {
slice := []int{1, 2, 3, 4}
odd := filter(slice, isOdd)
fmt.Println("Odd:", odd)
}
Panic/Recover:
Panic 是一个内建函数,可以中断原有的控制流程,进入一个令人恐慌的流程中。
当 F 函数调用 panic,F 的执行逻辑被中断,但是 F 中的延迟函数会正常执行,然后 F 返回调用它的地方。
在调用的地方,F 的行为就像调用了 panic,继续向上直到发生 panic 的 goroutine 中所有调用的函数返回,此时程序退出。
panic 可以手动或者运行时错误产生(越界数组)
recover:内建函数,可以让进入恐慌的流程中的 goroutine 回复过来。recover 仅在延迟函数中有效。
正常逻辑调用 recover 返回 nil。当前的 goroutine 先入恐慌可以用 recover 捕获 panic 的输入值,并且回复正常的执行。
main init 函数:
import: 导入包文件
// 点操作,表示你调用的时候可以省略前缀,比如 fmt.Println("hello") 可以写 Println("Hello")
import (
. "fmt"
)
// 别名操作
import (
f "fmt" //别名,简化调用 f.Println
"os"
)
// _ 操作
import (
_ "github.com/ziutek/mymysql/godrv" //_引入该包,而不直接使用包里的函数,而是调用包里的 init 函数
)
Struct 类型:属性或者字段容器
package main
import (
"fmt"
)
type Person struct {
name string
age int
}
func Older(p1, p2 Person) (Person, int) {
if p1.age > p2.age {
return p1, p1.age - p2.age
}
return p2, p2.age - p1.age
}
func main() {
// 初始化方式
var P Person
P.name = "Astaxie"
P.age = 23
p1 := Person{"Tom", 12}
p2 := Person{age: 10, name: "Tom"}
p := new(Person) // p 是个指针
fmt.Printf("name is ", P.name)
}
2.5 面向对象
method: 带有接受者的函数。method 附属在一个给定的类型上,语法和函数的声明语法几乎一样,只是在函数后加上一个 receiver
A method is a funciton with am inplicit first argument, called a reciver - Rob Pike
/*
定义: func (r ReceiverType) funcName(parameters) (results)
虽然 method 名字一样,但是接受者不一样,method 就不一样
method 里可以访问接受者的字段
调用 method 通过 . 访问就像 stuct 里访问字段一样
*/
package main
import (
"fmt"
"math"
)
type Rectangle struct {
width, height float64
}
type Circle struct {
radius float64
}
func (r Rectangle) area() float64 {
return r.width * r.height
}
func (c Circle) area() float64 {
return c.radius * c.radius * math.Pi
}
func main() {
r1 := Rectangle{12, 1}
c1 := Circle{10}
fmt.Println("r1 area", r1.area())
fmt.Println("r1 area", c2.area())
}
指针作为 receiver:
func (r *Rectangle) setHeight(height float64) {
r.height = height //NOTE:注意这里不必非要用 *r.hight = hight,Go 发现参数指针以后帮你自动处理
}
method 继承:如果匿名字段实现了一个 method,那么包含这个匿名字段的 struct 也能调用该 method
method 重写:和匿名字段冲突一样的道理,重写就实现了隐藏匿名字段的 method
2.6 接口 interface
interface 是一组 method 签名的组合,通过 interface 来定义对象的一组行为,通过 interface 定义对象的一组行为。 interface 类型定义了一组方法,如果某个对象实现了某个接口的所有方法,则此对象实现了这个接口(隐式)。
package main
import (
"fmt"
)
type Human struct {
name string
age int
phone string
}
type Student struct {
Human
school string
loan float32
}
type Employee struct {
Human
company string
money float32
}
func (h *Human) SayHi() {
fmt.Printf("Hi %s %s", h.name, h.phone)
}
func (h *Human) Sing(lyrics string) {
fmt.Println("la la", lyrics)
}
func (h *Human) Guzzle(beerStein string) {
fmt.Println("Guzzle", beerStein)
}
func (e *Employee) SayHi() {
fmt.Printf("hi I am %s work at %s call me %s\n", e.name, e.company, e.phone)
}
func (s *Student) BorrowMoney(amount float32) {
s.loan += amount
}
func (e *Employee) SpendSalary(amount float32) {
e.money -= amount
}
type Men interface {
SayHi()
Sing(lyrics string)
Guzzle(beerStein string)
}
type YoungChap interface {
SayHi()
Sing(song string)
BorrowMoney(amount float32)
}
type ElderyGent interface {
SayHi()
Sing(song string)
SpendSalary(amount float32)
}
interface 可以被任意对象实现,一个对象也可以实现任意多个 interface。 任意类型都是现了空 interface (interface{}),也就是包含了0个 method 的interface。
interface 值:如果定义一个interface的变量,这个变量里可以存实现这个 interface 的任意类型的对象。 比如定义了一个 Men interface 类型的变量 m,m 可以存 Human, Student, Employee。 因为 m 可以持有三种对象,可以定一个包含 Men 类型的 slice,它可以倍赋予实现了 Men 接口的任意结构的对象。 通过这种方式实现了鸭子类型。
空 interface: 可以存储任意类型数值,类似于 c void*。一个函数把 interface{} 作为参数,可以接受任意类型。同理也可以返回任意类型。
var a interface{}
var i int = 5
s := "hello"
a = i
a = s
interface 的函数参数:interface的变量可以持有任意实现该interface 类型的对象。只要实现接口,我们就可以传入
interface变量存储的类型:如何知道变量里实际存储的类型呢?
- comma-ok
- switch 测试
package main
import (
"fmt"
"strconv"
)
type Element interface{}
type List []Element
type Person struct {
name string
age int
}
func (p Person) String() string {
return "(name:" + p.name + "-age:" + strconv.Itoa(p.age) + "years)"
}
func main() {
list := make(List, 3)
list[0] = 1
list[1] = "hello"
list[3] = Person{"Deniss", 30}
for index, element := range list {
switch value := element.(type) { // element.(type) 只能在 switch 里边使用,外边还是要用 comma-ok
case int:
fmt.Printf("int %d %d", index, value)
case string:
fmt.Printf("str %d %d", index, value)
case Person:
fmt.Printf("person %d %d", index, value)
default:
fmt.Printf("list different type %d", index)
}
}
}
嵌入 interface:如果一个 interface1 作为另一个interface2 的嵌入字段,那么interface2 隐式包含了interface1 里的 method
反射: 检查程序运行时状态。reflect
t := reflect.TypeOf(i)
v := reflect.ValueOf(i)
并发
goroutine: 通过 go 的 runtime 管理的一个线程管理器,通过 go 关键字实现。其实就是一个普通的函数
go hello(a,b,c)
通过 go 关键字启动了一个 goroutine
package main
import (
"fmt"
"runtime"
)
func say(s string) {
for i := 0; i < 5; i++ {
runtime.Gosched() //让 cpu 让出时间片,下次某个时间继续恢复执行该 goroutine
fmt.Println(s)
}
}
func main() {
go say("world")
say("hello")
}
多个 goroutine 运行在同一个进程里边,共享内存数据。 不过设计上需要遵循:不要通过共享来通信,通过通信来共享。
channels:goroutine 共享相同的地址空间,访问共享内存需要做好同步。必须用 make 创建
ci := make(chan int)
cs := make(chan string)
cf := make(chan interface{})
ch <- v //发送数据 v 到 channel ch
v:= <-ch //从 ch 中接受数据,并赋值给v
默认channel接收和发送数据都是阻塞的,除非另一端准备好,这样使得 goroutine 同步变得简单,无需显示 lock。
buffered channels: 允许指定 channel 缓冲区大小。ch := make(chan type, value)
当 value =0,channel 是无缓冲区阻塞读写的,当 value>0,channel 是有缓冲、非阻塞的,知道写满 value 个才阻塞写入。
package main
import "fmt"
func sum(a []int, c chan int) {
total := 0
for _, v := range a {
total += v
}
c <- total
}
func main() {
a := []int{3, 1, 3, 5, 6, 7}
c := make(chan int)
go sum(a[:len(a)/2], c)
go sum(a[len(a)/2:], c)
x, y := <-c, <-c // 从 c 接收
fmt.Println(x, y, x+y)
}
Range 和 Close: 通过 range,像操作slice 或者 map 一样操作缓存类型的 channel
package main
import "fmt"
func fibonacci(n int, c chan int) {
x, y := 1, 1
for i := 0; i < n; i++ {
c <- x
x, y = y, x+y
}
close(c) // 应该在生产者里关闭,防止 panic。当你没有任何数据要发送,或者想显示结束 range 调用 close
}
func main() {
c := make(chan int, 10)
go fibonacci(cap(c), c)
for i := range c { //不断读取知道 channel 被显示关闭
fmt.Println(i)
}
}
Select: 多个 channel 操作。Go提供了select监听 channel 数据流动 Select: 多个 channel。select 可以监听 channel 上的数据流动。select 默认阻塞,只有当监听的 channel 中有发送或接收可以 进行时才会运行,当多个 channel 都准备好的时候随机选择一个执行。
package main
import "fmt"
func fibonacci(c, quit chan int) {
x, y := 1, 1
for {
select {
case c <- x:
x, y = y, x+y
case <-quit:
fmt.Println("quit")
return
default:
// 当监听的 channel 没有准备好的时候默认执行(select不再阻塞等待channel)
}
}
}
func main() {
c := make(chan int)
quit := make(chan int)
go func() {
for i := 0; i < 10; i++ {
fmt.Println(<-c)
}
quit <- 0
}()
fibonacci(c, quit)
}
有时候会出现 goroutine 阻塞的情况。还可以设置 select 设置超时。
3 Web基础
用 go 实现一个 http server 真滴很简单,支持高并发,内部实现是一个单独的 goroutine 处理请求。
package main
import (
"fmt"
"log"
"net/http"
"strings"
)
func sayhelloName(w http.ResponseWriter, r *http.Request) {
r.ParseForm()
fmt.Println(r.Form)
fmt.Println("path", r.URL.Path)
for k, v := range r.Form {
fmt.Println("key:", k)
fmt.Println("val:", strings.Join(v, ""))
}
fmt.Fprintf(w, "hello") //输出到客户端
}
func main() {
http.HandleFunc("/", sayhelloName)
err := http.ListenAndServe(":9090", nil)
if err != nil {
log.Fatal("ListenAndServe: ", err)
}
}
4 表单
5 访问数据库
5.1 database/sql 接口
sql.Register
driver.Driver // 返回的数据库conn 只能用在一个 goroutine
driver.Conn
driver.Stmt // Stmt 是一种准备好的状态,只能用在一个goroutine中
driver.Tx // 事务提交和回滚
driver.Execer /// Conn 选择实现的接口
driver.Result // 执行 update/insert 等操作返回的结果接口定义
driver.Rows // 执行查询返回的结果集接口定义
driver.RowsAffected // int64别名
driver.Value //空接口,容纳任何数据
driver.ValueConverter // 把一个普通值转换成 driver.Value
driver.Valuer // 定义了返回一个 driver.Value 的方式
database/sql
5.2 使用 Mysql
使用 "github.com/go-sql-driver/mysql" 演示数据库增删改查
package main
import (
"database/sql"
"fmt"
_ "github.com/go-sql-driver/mysql"
)
func main() {
// Open 打开一个注册过的数据库驱动
db, err := sql.Open("mysql", "root:@/test?charset=utf8")
checkErr(err)
// prepare 返回准备要执行的 sql,然后返回准备完毕的执行状态
stmt, err := db.Prepare("INSERT userinfo SET username=?,departname=?,created=?")
checkErr(err)
//exec 执行 stmt 准备好的 sql 语句
res, err := stmt.Exec("wnn", "研发", "2012-12-10")
checkErr(err)
id, err := res.LastInsertId()
checkErr(err)
fmt.Println(id)
stmt, err = db.Prepare("update userinfo set username=? where uid=?")
checkErr(err)
res, err = stmt.Exec("wnnupdate", id)
checkErr(err)
affect, err := res.RowsAffected()
checkErr(err)
fmt.Println(affect)
//query 直接执行 sql 返回 rows 结果
rows, err := db.Query("SELECT * from userinfo")
checkErr(err)
for rows.Next() {
var uid int
var username, departname, created string
err = rows.Scan(&uid, &username, &departname, &created)
checkErr(err)
fmt.Println(uid)
fmt.Println(username)
fmt.Println(departname)
fmt.Println(created)
}
stmt, err = db.Prepare("delete from userinfo where uid=?")
checkErr(err)
res, err = stmt.Exec(id)
checkErr(err)
affect, err = res.RowsAffected()
checkErr(err)
fmt.Println(affect)
db.Close()
}
func checkErr(err error) {
if err != nil {
panic(err)
}
}
5.5 使用 Beego orm
package main
import (
"fmt"
"github.com/astaxie/beego/orm"
_ "github.com/go-sql-driver/mysql"
)
// Model Struct
type User struct {
Id int
Name string `orm:"size(100)"`
}
func init() {
orm.RegisterDataBase("default", "mysql", "root:@/test?charset=utf8", 30)
orm.RegisterModel(new(User))
orm.RunSyncdb("default", false, true)
orm.RegisterDriver("mysql", orm.DRMySQL)
}
func main() {
o := orm.NewOrm()
user := User{Name: "wnn"}
id, err := o.Insert(&user)
user2 := User{Name: "wnn2"}
id, err = o.Insert(&user2)
fmt.Printf("id:%d, err:%v\n", id, err)
user.Name = "wnnupdate"
num, err := o.Update(&user)
fmt.Printf("Num:%d, err:%v\n", num, err)
u := User{Id: user.Id}
err = o.Read(&u)
fmt.Printf("err: %v\n", err)
num, err = o.Delete(&u)
fmt.Printf("num %d, err:%v\n", num, err)
}