Let's GO(四)
人生苦短,Let's GO
今天我学了什么?
1.panic && recover
Go的logo是一只萌萌的囊地鼠(Gopher)
当Go程序出现错误,程序将报panic(恐慌)
所以是错误代码吓到小地鼠了吗哈哈
然后需要用recover来安抚它
类似于 try-catch的语法
func b() {
defer func() { //defer must declare before panic
err:=recover() //defer + recover(),recover panic
if err !=nil {
fmt.Println("func b err")
}
}()
panic("panic in func b") //panic 触发结束程序
}
2.goroutine(协程)
非常方便的并发操作,
一个goroutine对应一个函数
func hello() {
fmt.Println("Hello World!")
wg.Done() //goroutine done,count-1
}
//替代time.Sleep(),确保goroutine都执行完毕
var wg sync.WaitGroup
func main() {
wg.Add(1) //count+1
for i := 0; i < 1000; i++ {
wg.Add(1)
go func(i int) {
fmt.Println("hello",i) //形成了闭包,i公用
wg.Done()
}(i)
} //多线程执行数不同
go hello() //open a goroutine
fmt.Println("hello world")
//time.Sleep(time.Second)
wg.Wait() //all goroutine done
}
3.锁
因为不同goroutine可能会同时读写同一个资源,所以需要给资源加锁
a. 互斥锁(sync.Mutex):控制共享资源只被一个goroutine获取
import "sync"
var (
x int
wg. sync.WaitGroup
lock sync.Mutex
)
func add() {
for i := 0; i < 5000; i++ {
lock.Lock() //将x值加锁
x ++
lock.Unlock() //执行完将x的锁取消
}
wg.Done()
}
b.读写互斥锁:适用于读多写少的情况,速度更快。
var (
xx int64
rwlock sync.RWMutex //读写互斥锁
)
func read() {
rwlock.RLock() //加读锁
time.Sleep(time.Millisecond)
rwlock.RUnlock()
wg.Done()
}
func write() {
rwlock.Lock() //写加读写锁
xx = xx + 1
time.Sleep(time.Millisecond*10)
rwlock.Unlock()
wg.Done()
}
4.channel(通道)
Go语言的并发模型是CSP(Communicating Sequential Processes),提倡通过通信共享内存而不是通过共享内存而实现通信。
一个goroutine的值通过channel发送给另一个goroutine
通道FIFO(First in,First out)
//ch1<-
func f1(ch chan<- int) { //(单向通道)chan<- 表示只能发送到通道
for i := 0; i < 100; i++ {
ch <- i
}
close(ch)
}
//( ch2 <- ch1 )^2
func f2(ch1 <-chan int,ch2 chan<- int) { //<-chan 表示只能从通道读取
for {
tmp,ok := <- ch1
if !ok {
break
}
ch2 <- tmp*tmp
}
close(ch2)
}
func main() {
ch1 := make(chan int,100) //初始化通道,100为缓冲区大小
ch2 := make(chan int,200)
go f1(ch1)
go f2(ch1,ch2)
for val := range ch2 {
fmt.Println(val)
}
}
5.网络编程(tcp,udp)
server:1.监听端口 2.建立与client的链接 3.与之交互
client:1.建立与server的链接 2.发送信息 3.关闭连接
//tcp_server_demo
func process(conn net.Conn) {
defer conn.Close() //close connection
//data
for {
reader := bufio.NewReader(conn) //read message
var buf [128]byte
n,err := reader.Read(buf[:])
if err != nil {
fmt.Println("read err,err:",err)
break
}
recv := string(buf[:n])
fmt.Printf("get:%v\n",recv) //show message
conn.Write([]byte("ok")) //reply
}
}
func main() {
//listen
listen,err := net.Listen("tcp","127.0.0.1:20000")
if err != nil {
fmt.Printf("listen port failed,err:%v\n",err)
return
}
//waiting for connection
for {
conn,err := listen.Accept()
if err != nil {
fmt.Printf("accept failed.err:%v\n",err)
continue
}
//go!
go process(conn)
}
}
//tcp_client_demo
func main() {
//conn
conn,err := net.Dial("tcp","127.0.0.1:20000")
if err != nil {
fmt.Println("dial failed,err:",err)
return
}
//send and receiver
input := bufio.NewReader(os.Stdin)
for {
s,_ := input.ReadString('\n')
s = strings.TrimSpace(s)
if strings.ToUpper(s) == "q" { //q for quit
return
}
//send message
_,err := conn.Write([]byte(s))
if err !=nil {
fmt.Printf("send failed,err:%v\n",err)
return
}
//receiver
var buf [1024]byte
n,err := conn.Read(buf[:])
if err != nil {
fmt.Printf("read failed,err:%v\n",err)
return
}
fmt.Println("get reply: ",string(buf[:n]))
}
}
客户端发送,服务端接收并发送反馈信息

这里要先运行server端,不然client端找不到端口会恐慌的
6. 单元测试
执行go test来测试功能是否符合预期
func TestSplit(t *testing.T) { //测试哪个函数就叫Testxxx()
type test struct {
input string
sep string
want []string
}
tests := map[string]test { //给出几组样例,OJ或者leetcode不会就是这样的吧..
"simple":test {"ab c"," ",[]string{"ab","c"}}, //给测试数据命令,可以针对这组数据进行测试
"multi sep":test{"hello,World",",",[]string{"hello","World"}},
}
for name,tc := range tests {
t.Run(name, func(t *testing.T) {
got := Split(tc.input,tc.sep)
if !reflect.DeepEqual(got,tc.want) {
t.Errorf("name:%s want:%v got:%v\n",name,tc.want,got)
}
})
}
}
通过测试:

未能通过测试:

性能测试,将跑够足够的量来测试
//性能基准测试
func BenchmarkSplit(b *testing.B) {
//b.N 由go test决定是否继续加大测试量
for i := 0; i < b.N; i++ {
Split("a,v,c",",")
}
}
将给出详细的测试结果:

总结
Go的基础语法到这里就粗略的过一遍了,如果要打磨自己的基础的,可以从学校OJ一直到acm题多加练习。代码量上去了,理解也会水涨船高。
我可能没有将Go基础语法全部内容写出来,而且我可以说只是复制粘贴稍微整理了一下代码,
更加详细的教学请移步大佬的博客李文周的博客
再次感谢前辈的教学内容。
挺看好Go的前景的,能从Go的各方各面感觉到这个语言的蓬勃生气,
但可能因为还是初学者,眼界不够,暂时也说不出个子丑寅卯来,继续学吧。
人生苦短,GO!GO!GO!
Let's GO(四)的更多相关文章
- 构建一个基本的前端自动化开发环境 —— 基于 Gulp 的前端集成解决方案(四)
通过前面几节的准备工作,对于 npm / node / gulp 应该已经有了基本的认识,本节主要介绍如何构建一个基本的前端自动化开发环境. 下面将逐步构建一个可以自动编译 sass 文件.压缩 ja ...
- 《Django By Example》第四章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:祝大家新年快乐,这次带来<D ...
- 如何一步一步用DDD设计一个电商网站(四)—— 把商品卖给用户
阅读目录 前言 怎么卖 领域服务的使用 回到现实 结语 一.前言 上篇中我们讲述了“把商品卖给用户”中的商品和用户的初步设计.现在把剩余的“卖”这个动作给做了.这里提醒一下,正常情况下,我们的每一步业 ...
- 从0开始搭建SQL Server AlwaysOn 第四篇(配置异地机房节点)
从0开始搭建SQL Server AlwaysOn 第四篇(配置异地机房节点) 第一篇http://www.cnblogs.com/lyhabc/p/4678330.html第二篇http://www ...
- MVVM设计模式和WPF中的实现(四)事件绑定
MVVM设计模式和在WPF中的实现(四) 事件绑定 系列目录: MVVM模式解析和在WPF中的实现(一)MVVM模式简介 MVVM模式解析和在WPF中的实现(二)数据绑定 MVVM模式解析和在WPF中 ...
- “四核”驱动的“三维”导航 -- 淘宝新UI(需求分析篇)
前言 孔子说:"软件是对客观世界的抽象". 首先声明,这里的"三维导航"和地图没一毛钱关系,"四核驱动"和硬件也没关系,而是为了复杂的应用而 ...
- 【翻译】MongoDB指南/CRUD操作(四)
[原文地址]https://docs.mongodb.com/manual/ CRUD操作(四) 1 查询方案(Query Plans) MongoDB 查询优化程序处理查询并且针对给定可利用的索引选 ...
- HTML 事件(四) 模拟事件操作
本篇主要介绍HTML DOM中事件的模拟操作. 其他事件文章 1. HTML 事件(一) 事件的介绍 2. HTML 事件(二) 事件的注册与注销 3. HTML 事件(三) 事件流与事件委托 4. ...
- 【原】AFNetworking源码阅读(四)
[原]AFNetworking源码阅读(四) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇还遗留了很多问题,包括AFURLSessionManagerTaskDe ...
- 带你实现开发者头条APP(四)---首页优化(加入design包)
title: 带你实现开发者头条APP(四)---首页优化(加入design包) tags: design,Toolbar,TabLayout,RecyclerView grammar_cjkRuby ...
随机推荐
- TB6612FNG电机驱动模块
TB6612FNG电机驱动模块 模块原理图 模块的使用 TB6612是双驱动,也就是可以驱动两个电机 下面分别是控制两个电机的IO口 STBY口接单片机的IO口清零电机全部停止, 置1通过AIN1 A ...
- Python流程控制语句详解
1.程序结构 计算机在解决问题时,分别是顺序执行所有语句.选择执行部分语句.循环执行部分语句,分别是:顺序结构.选择结构.循环结构.如下图: 2.选择语句 2.1最简单的if语句 Python使用保留 ...
- 用python复制文件夹
用python复制文件 1. 根据文件夹的名称复制 需要复制的文件夹编号文件中,每一行表示一个编号,如下所示: > cat id.txt 1 2 3 ... > 目标文件的目录结构树如下所 ...
- Linux中GitLab的部署
1.版本控制介绍 版本控制最主要的功能就是追踪文件的变更.它将什么时候.什么人更改了文件的什么内容等信息忠实地了记录下来.每一次文件的改变,文件的版本号都将增加.除了记录版本变更外,版本控制的另一 ...
- numpy.stack和numpy.concatenate的区别
在使用numpy进行矩阵运算的时候踩到的坑,原因是不能正确区分numpy.concatenate和numpy.stack在功能上的差异. 先说numpy.concatenate,直接看文档: nump ...
- SpringMVC整合mybaitis
目录 一.新建一个基于Maven的Web项目 二.创建数据库与表 三.添加依赖包 四.新建POJO实体层 五.新建MyBatis SQL映射层 六.完成Spring整合MyBatis配置 七.创建服务 ...
- Flask框架基础功能
引言 本文简单汇总Flask框架几大基础功能,包括: 路由系统 模板 数据库 几种常用Flask库 一个简单的Flask事例 Flask是一个基于Python,依赖Jinja2模板和WSGI服务的框架 ...
- 【vue】axios二次封装,更好的管理api接口和使用
在现在的前端开发中,前后端分离开发比较主流,所以在封装方法和模块化上也是非常需要掌握的一门技巧.而axios的封装也是非常的多,下面的封装其实跟百度上搜出来的axios封装或者axios二次封装区别不 ...
- 微信小程序-创建小程序页面
QQ讨论群:785071190 创建页面 创建小程序页面非常简单,鼠标在需要创建页面的目录右击,可看到下图菜单,选择"Page"即可创建出一个页面. 输入页面名称,回车就可以创建出 ...
- SpringBoot读取application.properties中文乱码
[本文版权归微信公众号"代码艺术"(ID:onblog)所有,若是转载请务必保留本段原创声明,违者必究.若是文章有不足之处,欢迎关注微信公众号私信与我进行交流!] 解决方案 在ID ...
