golang代码片段(摘抄)
以下是从golang并发编程实战2中摘抄过来的代码片段,主要是实现一个简单的tcp socket通讯(客户端发送一个数字,服务端计算该数字的立方根然后返回),写的不错,用到了go的并发以及看下郝林大神是如何处理socket通讯的。具体代码记录如下,多看多学习多共勉:
package main import (
"net"
"strings"
"fmt"
"time"
"io"
"bytes"
"strconv"
"math"
"math/rand"
"sync"
)
const (
SERVER_NETWORK = "tcp"
SERVER_ADDRESS = "127.0.0.1:8085"
DELIMITER = '\t'
) var wg sync.WaitGroup func main() {
wg.Add(2)
go serverGo()
time.Sleep(500 * time.Millisecond)
go clientGo(1)
wg.Wait()
} func serverGo() {
defer wg.Done()
listener, err := net.Listen(SERVER_NETWORK, SERVER_ADDRESS)
if err != nil {
printServerLog("Listen Error: %s", err)
return
}
defer listener.Close()
printServerLog("Got listener for the server.(local address: %s)", listener.Addr())
for {
conn, err := listener.Accept()
if err != nil {
printServerLog("Accept Error: %s", err)
}
printServerLog("Established a connection with a client application,(remote address: %s)", conn.RemoteAddr())
go handleConn(conn)
}
} func printLog(role string, sn int, format string, args ...interface{}) {
if !strings.HasSuffix(format, "\n") {
format += "\n"
}
fmt.Printf("%s[%d]: %s", role, sn, fmt.Sprintf(format, args...))
} func printServerLog(format string, args ...interface{}) {
printLog("Server", 0, format, args...)
} func printClientLog(sn int, format string, args ...interface{}) {
printLog("Cient", sn, format, args...)
} func handleConn(conn net.Conn) {
defer conn.Close()
for {
conn.SetDeadline(time.Now().Add(10 * time.Second)) // set read timeline 10s
strReq, err := read(conn)
if err != nil {
if err == io.EOF {
printServerLog("The connection is closed by another side.")
} else {
printServerLog("Read Error: %s", err)
}
break
}
printServerLog("Received request: %s", strReq)
intReq, err := strToInt32(strReq)
if err != nil {
n, err := write(conn, err.Error())
printServerLog("Sent error message (written %d bytes): %s", n, err)
continue
}
floatResp := cbrt(intReq)
respMsg := fmt.Sprintf("The cube root of %d is %f.", intReq, floatResp)
n, err := write(conn, respMsg)
if err != nil {
printServerLog("Sent response message (written %d bytes: %s).", n, respMsg)
}
}
} func read(conn net.Conn) (string, error){
readBytes := make([]byte, 1)
var buffer bytes.Buffer
for {
_, err := conn.Read(readBytes)
if err != nil {
return "", err
}
readByte := readBytes[0]
if readByte == DELIMITER {
break
}
buffer.WriteByte(readByte)
}
return buffer.String(), nil
} func strToInt32(str string) (int32, error){
i32, err := strconv.ParseInt(str, 10 ,32)
return int32(i32), err
} func write(conn net.Conn, content string) (int, error) {
var buffer bytes.Buffer
buffer.WriteString(content)
buffer.WriteByte(DELIMITER)
return conn.Write(buffer.Bytes())
} func cbrt(intReq int32) float64 {
return math.Cbrt(float64(intReq))
} func clientGo(id int) {
defer wg.Done()
conn, err := net.DialTimeout(SERVER_NETWORK, SERVER_ADDRESS, 2 * time.Second)
if err != nil {
printClientLog(id, "Dial Error: %s", err)
return
}
defer conn.Close()
printClientLog(id, "Connected to server.(remote address: %s,local address: %s)", conn.RemoteAddr(), conn.LocalAddr())
time.Sleep(200 * time.Millisecond) requestNumber := 5
conn.SetDeadline(time.Now().Add(5 * time.Millisecond))
for i := 0; i< requestNumber; i++ {
req := rand.Int31()
n, err := write(conn, fmt.Sprintf("%d", req))
if err != nil {
printClientLog(id, "Write Error: %s", err)
continue
}
printClientLog(id, "Sent request (written %d bytes)", n)
}
for j := 0; j< requestNumber; j++ {
strResp, err := read(conn)
if err != nil {
if err == io.EOF {
printClientLog(id, "The connection is closed by another side.")
}else{
printClientLog(id, "Read Error: %s", err)
}
break
}
printClientLog(id, "Received response: %s", strResp)
}
}
执行效果:
golang代码片段(摘抄)的更多相关文章
- Golang, 以 9 个简短代码片段,弄懂 defer 的使用特点
作者:林冠宏 / 指尖下的幽灵 掘金:https://juejin.im/user/587f0dfe128fe100570ce2d8 博客:http://www.cnblogs.com/linguan ...
- sublimetext3中保存代码片段
在日常的开发工作中,不断重复上一次敲过的代码,有时确实感到伐木累."蓝瘦"(难受)."香菇"(想哭),大概表达的也是这样的心境吧!:grinning: 所以,在 ...
- Code Snippets 代码片段
Code Snippets 代码片段 1.Title : 代码片段的标题 2.Summary : 代码片段的描述文字 3.Platform : 可以使用代码片段的平台,有IOS/OS X/ ...
- 10个 jQuery 代码片段,可以帮你快速开发。
转载自:http://mp.weixin.qq.com/s/mMstI10vqwu8PvUwlLborw 1.返回顶部按钮 你可以利用 animate 和 scrollTop 来实现返回顶部的动画,而 ...
- 代码片段添加智能提示,打造一款人见人爱的ORM框架
SqlSugar ORM优点: 1.高性能,达到原生最高水准,比SqlHelper性能要高,比Dapper快30% 比EF快50% 2.支持多种数据库 ,sql版本更新最快,其它会定期更新,可以在多种 ...
- js/jquery/html前端开发常用到代码片段
1.IE条件注释 条件注释简介 IE中的条件注释(Conditional comments)对IE的版本和IE非IE有优秀的区分能力,是WEB设计中常用的hack方法.条件注释只能用于IE5以上,IE ...
- Visual Studio 如何使用代码片段Code Snippet提高编程速度!!!
使用Code Snippet简化Coding 在开发的项目的时候,你是否经常遇到需要重复编写一些类似的代码,比如是否经常会使用 for.foreach ? 在编写这两个循环语句的时候,你是一个字符 ...
- Visual Studio 的代码片段工具
当安装完Visual Studio之后,会有附带一些原生的代码片段文件(*.snippet),对于vs2013参考目录如下: X:\Program Files (x86)\Microsoft Visu ...
- sublime代码片段
创建方法:Tools > New Snippet 这时你会看到如下示例代码: <snippet> <content><![CDATA[ Hello, ${ ...
随机推荐
- 【ocp-12c】最新Oracle OCP-071考试题库(43题)
43.(9-2)choose three Which three tasks can be performed by DDL statements? A) preventing data retrie ...
- loadrunner录制的基本知识
1.http/html录制选择web_url,如下图所示: 开始录制->Options->Recording->HTML Advanced->选择web_url->OK ...
- objectARX创建 PaletteSet 停靠面板示例
objectARX创建 PaletteSet 停靠面板示例 图文By edata ,转载注明出处 http://www.cnblogs.com/edata 部分代码参考张帆<AutoCAD Ob ...
- NOIp2018提高组游记
Day1 T1 积木大赛 NOIp2013D2T1.....看到的时候我还以为我记错了,以为原题是一次可以随便加,这题只能加一,出考场后查了下发现一模一样. #include <iostream ...
- JSP里面九个内置对象
JSP内置对象(9个常用的内置对象) 1.request对象 客户端的请求信息被封装在request对象中,通过它才能了解到客户的需求, 然后做出响应.它是HttpServletRequest类的实例 ...
- 兼容各浏览器的js回车事件
HTML代码: <input type="text" onkeydown="EnterPress(event)" /> JS代码: function ...
- 河内塔问题(C++版)
上次,我们讲了汉诺塔,今天我们来讲一讲和汉诺塔类似的题目<河内塔问题> 题目描述 Description 一位法国数学家曾编写过一个印度的古老传说:在世界中心贝拿勒斯(在印度北部)的圣庙里 ...
- Python小实验——读&写Excel文件内容
安装xlrd模块和xlwt模块 读取Excel文件了内容需要额外的模块-- \(xlrd\),在官网上可以找到下载:https://pypi.python.org/pypi/xlrd#download ...
- 2016级算法期末上机-H.难题·AlvinZH's Fight with DDLs III
1119 AlvinZH's Fight with DDLs III 思路 难题,最小点覆盖. 分析题意,某一个任务,既可以在笔记本A的 \(a\) 模式下完成,也可以在笔记本B的 \(b\) 模式下 ...
- (RaspBerry Pi) Python GPIO 基本操作
目前打算由潛入深慢慢學習RaspBerry Pi, 所以先由最容易下手的Python進入樹莓派的世界 首先要使用 GPIO 需要利用RPI.GPIO package想當然爾必須先安裝 所以先執行下列命 ...