/*
gotips_test.go:
Golang速学速查速用代码手册
Source: github.com/coderzh/CodeTips/blob/master/gotips_test.go
Author: coderzh(github.com/coderzh)
Blog: http://blog.coderzh.com
参考:《Go语言编程》
*/
package main
import (
"errors"
"fmt"
"github.com/stretchr/testify/assert"
"io"
"io/ioutil"
"log"
"math"
"os"
"path/filepath"
"regexp"
"strings"
"sync"
"testing"
"time"
)
// 0. 注释
/*
规范:
1. 命名:骆驼命名法(不要用下划线)
命令:
go get github.com/coderzh/xxx
go build calc
go run xxx.go
go install calc
*/
// 1. Hello World
func helloWorld() {
fmt.Println("Hello, 世界")
}
// 2.变量类型
func typeDemo() {
// 变量声明
var v1 int
var (
v2 int
v3 string
)
//var p *int // 指针类型
// 变量初始化
var v4 int = 10
// 等价于:
var v5 = 10
// 一般这样就好
v6 := 10
// 赋值,多重赋值
v1 = 10
v2, v3 = 20, "test"
// 匿名变量 _
_, v4 = v5, v6
fmt.Println(v1, v2, v3, v4)
// 常量
const Pi float64 = 3.1415926
const MaxPlayer = 10
// 枚举
const (
Sunday = iota // iota从0递增
Mondy
Tuesday
// ...
)
// 类型
// 1. 布尔
var b1 bool
b1 = true
b1 = (1 == 2)
fmt.Println(b1)
// 2. 整形
// int8 uint8 int16 uint16 int32 uint32 int64 uint64 int uint uintptr
var i32 int32
// 强制转换
i32 = int32(64)
// 运算:+, -, *, /, %(求余)
// 比较:>, <, ==, >=, <=, !=
// 位运算:x << y, x >> y, x ^ y, x & y, x | y, ^x (取反)
fmt.Println(i32)
// 3. 浮点
// float32, float64
var f1 float64 = 1.0001
var f2 float64 = 1.0002
// 浮点比较
isEqual := math.Dim(f1, f2) < 0.0001
fmt.Println(isEqual)
// 4. 字符串
var s1 string
s1 = "abc"
// 字符串连接
s1 = s1 + "ddd"
// 取长度
n := len(s1)
// 取字符
c1 := s1[0]
// 反引号,不转义,常用于正则表达式
s1 = `\w+`
fmt.Println(c1)
fmt.Println(strings.HasPrefix("prefix", "pre")) // true
fmt.Println(strings.HasSuffix("suffix", "fix")) // true
// 字节遍历
for i := 0; i < n; i++ {
ch := s1[i]
fmt.Println(ch)
}
// Unicode字符Rune遍历
for i, ch := range s1 {
fmt.Println(i, ch)
}
// 5. 数组
var arr1 [32]int
//var arr2 [3][8]int // 二维数组
// 初始化
arr1 = [32]int{0}
array := [5]int{1, 2, 3, 4, 5}
// 临时结构体数组
structArray := []struct {
name string
age int
}{{"Tim", 18}, {"Jim", 20}}
// 数组遍历
for i := 0; i < len(array); i++ {
fmt.Println(array[i])
}
for i, v := range structArray {
fmt.Println(i, v)
}
// 数组是值类型,每次参数传递都是一份拷贝
// 数组切片Slice
var mySlice []int = arr1[:2]
mySlice1 := make([]int, 5)
mySlice2 := make([]int, 5, 10)
fmt.Println("len(mySlice2:", len(mySlice2)) // 5
fmt.Println("cap(mySlice2:", cap(mySlice2)) // 10
mySlice3 := append(mySlice, 2, 3, 4)
mySlice4 := append(mySlice, mySlice1...)
copy(mySlice3, mySlice4)
// 6. Map
var m map[int]string
m[1] = "ddd"
m1 := make(map[int]string)
m2 := map[int]string{
1: "a",
2: "b",
}
delete(m2, 1)
value, ok := m1[1]
if ok {
fmt.Println(value)
}
for k, v := range m2 {
fmt.Println(k, v)
}
}
// 3. 流程控制
func flowDemo() {
// if else
a := 10
if a < 10 {
// ..
} else {
// ..
}
// switch
switch a {
case 0:
fmt.Println("0")
case 10:
fmt.Println("10")
default:
fmt.Println("default")
}
switch {
case a < 10:
fmt.Println("<10")
case a < 20:
fmt.Println("<20")
}
// 循环
for i := 0; i < 10; i++ {
}
// 无限循环
sum := 0
for {
sum++
if sum > 10 {
break
// 指定break
// break JLoop
}
}
goto JLoop
JLoop:
// break to here
}
// 4. 函数
// func 函数名(参数列表)(返回值列表) {
// }
func sum1(value1 int, value2 int) (result int, err error) {
// err = errors.New("xxxx")
return value1 + value2, nil
}
func sum2(value1, value2 int) int {
return value1 + value2
}
// 不定参数
// myFunc(1, 2, 3, 4, 5)
func myFunc(args ...int) {
for _, arg := range args {
fmt.Println(arg)
}
// 传递
// myFunc2(args...)
// myFunc2(args[1:]...)
}
// 任意类型的不定参数
func myPrintf(args ...interface{}) {
for _, arg := range args {
switch arg.(type) {
case int:
fmt.Println(arg, "is int")
case string:
fmt.Println(arg, "is string")
default:
fmt.Println(arg, "is unknown")
}
}
}
// 匿名函数
func anonymousFunc() {
f := func(a, b int) int {
return a + b
}
f(1, 2)
}
// defer
func deferDemo(path string) {
f, err := os.Open(path)
if err != nil {
return
}
defer f.Close()
// or
defer func() {
if r := recover(); r != nil {
fmt.Printf("Runtime error caught: %v", r)
}
}()
}
// 5. 结构体
type Rect struct {
// 小写为private
x, y float64
// 大写为public
Width, Height float64
}
// 大写方法为public,小写为private
func (r *Rect) Area() float64 {
return r.Width * r.Height
}
func netRect(x, y, width, height float64) *Rect {
// 实例化结构体
// rect1 := new(Rect)
// rect2 := &Rect{}
// rect3 := &Rect{Width:100, Height:200}
return &Rect{x, y, width, height}
}
// 匿名组合
type Base struct {
Name string
}
func (base *Base) Foo() {}
func (base *Base) Bar() {}
type Foo struct {
Base
*log.Logger
}
func (foo *Foo) Bar() {
foo.Base.Bar()
// ...
}
// 非侵入式接口
type IFile interface {
Read(buf []byte) (n int, err error)
Write(buf []byte) (n int, err error)
}
type File struct {
}
func (file *File) Read(buf []byte) (n int, err error) {
return 0, nil
}
func (file *File) Write(buf []byte) (n int, err error) {
return 0, nil
}
func interfaceDemo() {
// 只要实现了Read, Write方法即可
var file IFile = new(File)
// 接口查询
// 是否实现了IFile接口
if file2, ok := file.(IFile); ok {
file2.Read([]byte{})
}
// 实例类型是否是File
if file3, ok := file.(*File); ok {
file3.Read([]byte{})
}
// 类型查询
switch v := file.(type) {
}
}
// 6. 并发编程
func counting(ch chan int) {
ch <- 1
fmt.Println("counting")
}
func channelDemo() {
chs := make([]chan int, 10)
for i := 0; i < len(chs); i++ {
chs[i] = make(chan int)
// 带缓冲区大小
// c: = make(chan int, 1024)
// for i:= range c {
// }
go counting(chs[i])
}
for _, ch := range chs {
<-ch
// channel select
/*
select {
case <-ch:
// ...
case ch <- 1:
}
*/
}
// 单向Channel
var ch1 chan<- int // 只能写入int
var ch2 <-chan int // 只能读出int
// 关闭Channel
close(ch1)
_, ok := <-ch2
if !ok {
// already closed
}
}
// 锁
var m sync.Mutex
func lockDemo() {
m.Lock()
// do something
defer m.Unlock()
}
// 全局唯一操作
var once sync.Once
// once.Do(someFunction)
// 7. 网络编程
// import "net"
// net.Dial("tcp", "127.0.0.1:8080")
// 8. json处理
// import "encoding/json"
// json.Marshal(obj) 序列化
// json.Unmarshal() 反序列化
// 9. Web开发
// import "net/http"
// 模板
// import "html/template"
// 10. 常用库
// import "os"
// import "io"
// import "flag"
// import "strconv"
// import "crypto/sha1"
// import "crypto/md5"
// 11. 单元测试
// _test结尾的go文件: xxx_test.go
// 函数名以Test开头
func TestDemo(t *testing.T) {
r := sum2(2, 3)
if r != 5 {
t.Errorf("sum2(2, 3) failed. Got %d, expect 5.", r)
}
assert.Equal(t, 1, 1)
}
// 12. 性能测试
func benchmarkAdd(b *testing.B) {
b.StopTimer()
// dosometing
b.StartTimer()
}
/*
其他常用的代码片段
*/
// 1. 遍历文件 filepath.Walk
// import "path/filepath"
func doHashWalk(dirPath string) error {
fullPath, err := filepath.Abs(dirPath)
if err != nil {
return err
}
callback := func(path string, fi os.FileInfo, err error) error {
return hashFile(fullPath, path, fi, err)
}
return filepath.Walk(fullPath, callback)
}
func hashFile(root string, path string, fi os.FileInfo, err error) error {
if fi.IsDir() {
return nil
}
rel, err := filepath.Rel(root, path)
if err != nil {
return err
}
log.Println("hash rel:", rel, "abs:", path)
return nil
}
// 2. 读取文件
// import "io/ioutil"
func readFileDemo(filename string) {
content, err := ioutil.ReadFile(filename)
if err != nil {
//Do something
}
lines := strings.Split(string(content), "\n")
fmt.Println("line count:", len(lines))
}
// 判断目录或文件是否存在
func existsPathCheck(path string) (bool, error) {
// 判断不存在
if _, err := os.Stat(path); os.IsNotExist(err) {
// 不存在
}
// 判断是否存在
_, err := os.Stat(path)
if err == nil {
return true, nil
}
if os.IsNotExist(err) {
return false, nil
}
return true, err
}
// 文件目录操作
func fileDirDemo() {
// 级联创建目录
os.MkdirAll("/path/to/create", 0777)
}
// 拷贝文件
func copyFile(source string, dest string) (err error) {
sf, err := os.Open(source)
if err != nil {
return err
}
defer sf.Close()
df, err := os.Create(dest)
if err != nil {
return err
}
defer df.Close()
_, err = io.Copy(df, sf)
if err == nil {
si, err := os.Stat(source)
if err != nil {
err = os.Chmod(dest, si.Mode())
}
}
return
}
// 拷贝目录
func copyDir(source string, dest string) (err error) {
fi, err := os.Stat(source)
if err != nil {
return err
}
if !fi.IsDir() {
return errors.New(source + " is not a directory")
}
err = os.MkdirAll(dest, fi.Mode())
if err != nil {
return err
}
entries, err := ioutil.ReadDir(source)
for _, entry := range entries {
sfp := filepath.Join(source, entry.Name())
dfp := filepath.Join(dest, entry.Name())
if entry.IsDir() {
err = copyDir(sfp, dfp)
if err != nil {
fmt.Println(err)
}
} else {
err = copyFile(sfp, dfp)
if err != nil {
fmt.Println(err)
}
}
}
return nil
}
// 3. 时间处理
// import "time"
func TestTimeDemo(t *testing.T) {
// Parse
postDate, err := time.Parse("2006-01-02 15:04:05", "2015-09-30 19:19:00")
fmt.Println(postDate, err)
// Format
assert.Equal(t, "2015/Sep/30 07:19:00", postDate.Format("2006/Jan/02 03:04:05"))
assert.Equal(t, "2015-09-30T19:19:00Z", postDate.Format(time.RFC3339))
}
// 4. 正则表达式
// import "regexp"
func TestRegexp(t *testing.T) {
// 查找匹配
re := regexp.MustCompile(`(\d+)-(\d+)`)
r := re.FindAllStringSubmatch("123-666", -1)
assert.Equal(t, 1, len(r))
assert.Equal(t, "123", r[0][1])
assert.Equal(t, "666", r[0][2])
cjkRe := regexp.MustCompile(`\p{Han}|\p{Hangul}|\p{Hiragana}|\p{Katakana}`)
assert.True(t, cjkRe.MatchString("中文"))
assert.True(t, cjkRe.MatchString("도형이"))
assert.True(t, cjkRe.MatchString("カテゴリー"))
assert.False(t, cjkRe.MatchString("abc"))
}
func main() {
helloWorld()
}