网络编程
socket,http,rpc,json处理
前端是Web+移动,后端是Linux(命令行centos)+开源
http编程
服务器和客户端(浏览器也是客户端)
socket编程
rpc编程
------------------------------------------------------------------------------server.go
package main
//服务端开发,特别是Web开发,基本上全是处理HTTP请求的处理。
//根据具体用途分为两种:Web页面开发和API接口开发。
//Web页面开发也完全可以看成是API接口开发,只是它的两个主要部分,
//页面和ajax请求,一个是返回html,另外一个可以返回html,也可以返回其他格式的而已。
//API接口开发是针对有客户端产品而言的。可能是移动设备,可能是PC应用等。
import (
"fmt"
"io"
"net/http"
)
func handler(w http.ResponseWriter, r *http.Request) {
http.SetCookie(w, &http.Cookie{Name: "pa", Value: "dps"})
r.Header.Add("server", "go/1.0")
io.WriteString(w, "myserver is running!")
fmt.Printf("r.header=%+v,r.usr.string=%+v,r.url.path=%+v\n", r.Header, r.URL.String(), r.URL.Path)
r.ParseForm()
d := r.Form
fmt.Println(d)
}
func main() {
http.HandleFunc("/agent", handler)
http.ListenAndServe(":8080", nil)
}
------------------------------------------------------------------------------------get.go
自己写客户端,get请求
代替浏览器的地址栏里输入的所有信息
func main() {
resp, err := http.Get("http://localhost:8080/agent?name=幻刺&password=123456#stuff")
if err != nil {
fmt.Println("resp err=", err)
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Println("body err=", err)
}
fmt.Println(string(body))
fmt.Println(resp.StatusCode)
fmt.Println(resp.Proto, resp.ProtoMajor, resp.ProtoMinor)
}
---------------------------------------------------------------------------------post.go
自己写客户端 post请求
post适合提交有密码的信息,因为参数都存储在body中,设置对应的bodytype,对应在浏览器地址栏不会显示
浏览器默认是get请求,向服务器索取数据的一种请求
post是想服务器提交数据的一种请求,要提交的信息位于信息头后面的实体中(数据体),在地址栏看不到。
HTTP协议是以ASCII码传输,建立在TCP/IP协议之上的应用层规范,规范http请求格式
method request-URL version
headers
空一行
entity-body
POST提交的数据必须放在消息主体entity-body中,但没有规定编码方式,但要服务端解析成功才有意义。服务器一般根据 请求头中 Content-Type字段获得body的编码方式
POST(URL,bodytype,[]byte)
(1)application/x-www-form-urlencoded 最常见的POST提交数据的方式,浏览器的原生form表单。后面可以跟charset=utf-8
(2)multipart/form-data
(3)application/json
(4)text/xml XML-RPC远程调用
//第三个参数的简单形式 (url.Values{"title":{"sss"},"content":{"article body"}})
func main() {
v := url.Values{}
v.Add("name", "幻刺")
v.Set("password", "123456")
u := ioutil.NopCloser(strings.NewReader(v.Encode()))
r, err := http.Post("http://localhost:8080/agent", "application/x-www-form-urlencoded", u)
if err != nil {
fmt.Println("http post err:", err)
}
defer r.Body.Close()
fmt.Println(r.StatusCode)
body, err := ioutil.ReadAll(r.Body)
if err != nil {
fmt.Println("http body err:", err)
}
fmt.Println(string(body))
}
---------------------------------------------------------------------------------client.go
请求头的单词一定不能拼错,都是对应服务端语言去解析的
func main() {
client := &http.Client{}
req, err := http.NewRequest("POST", "http://localhost:8080/agent", strings.NewReader("name=敌法,password=32984"))
if err != nil {
fmt.Println("newRequest err:", err)
}
req.Header.Set("Connection", "keep-alive")
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
req.Header.Set("Cookie", "幻刺")
req.Header.Set("User-Agent", "幽鬼")
resp, err := client.Do(req)
if err != nil {
fmt.Println("client do err:", err)
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Println("body read err:", err)
}
fmt.Println(string(body))
}
------------------------------------------------socket编程
tcp tcp4 tcp6 udp udp4 udp6 ip ip4 ip6 简洁的抽象了网络层和传输层
conn,err:=net.Dial("tcp","192.168.0,10:8080)
conn,err:=net.Dial("udp","192.168.0,10:8080)
conn,err:=net.Dial("ip4:icmp","192.168.0,10:8080)
conn,err:=net.Dial("ip4:1","192.168.0,10:8080)
用tcp写个聊天测试,分客户端和服务端
------------------------------------Dial()
server.go
func main() {
addr, err := net.ResolveTCPAddr("tcp", ":8080")
checkErr(err)
listen, err := net.ListenTCP("tcp", addr)
checkErr(err)
fmt.Println("Start server...")
for {
conn, err := listen.Accept()
checkErr(err)
go Handle(conn) // 每次建立一个连接就放到单独的线程内做处理
}
}
const BufLength = 128
var users map[string]net.Conn = make(map[string]net.Conn, 10)
func Handle(conn net.Conn) {
conn.Write([]byte("欢迎加入2B聊天组~"))
for {
data := make([]byte, 0) //此处做一个输入缓冲以免数据过长读取到不完整的数据
buf := make([]byte, BufLength)
for {
n, err := conn.Read(buf)
if err != nil && err != io.EOF {
checkErr(err)
}
data = append(data, buf[:n]...)
if n != BufLength {
break
}
}
cmd := strings.Split(string(data), "|")
fmt.Println("命令:", cmd)
switch cmd[0] {
case "nick":
fmt.Println("注册名称:" + cmd[1])
users[cmd[1]] = conn
case "say":
for k, v := range users {
if k != cmd[1] {
fmt.Println("给" + k + "发送消息:" + cmd[2])
v.Write([]byte(cmd[1] + ":[" + cmd[2] + "]"))
}
}
}
}
}
func checkErr(err error) {
if err != nil {
fmt.Println(err)
os.Exit(-1)
}
}
------------------------------------client.go
var nick string = ""
func main() {
addr, err := net.ResolveTCPAddr("tcp", ":8080")
checkErr(err)
conn, err := net.DialTCP("tcp", nil, addr)
checkErr(err)
// 读取提示
data := make([]byte, 1024)
conn.Read(data)
fmt.Println(string(data))
// 输入昵称
fmt.Print("输入昵称:")
fmt.Scanf("%v", &nick)
fmt.Println("Hello " + nick)
conn.Write([]byte("nick|" + nick))
go Handle(conn)
for {
someTex := ""
fmt.Scanf("%v", &someTex)
conn.Write([]byte("say|" + nick + "|" + someTex))
}
}
const BufLength = 128
func Handle(conn net.Conn) {
for {
data := make([]byte, 1024)
buf := make([]byte, BufLength)
for {
n, err := conn.Read(buf)
if err != nil && err != io.EOF {
checkErr(err)
}
data = append(data, buf[:n]...)
if n != BufLength {
break
}
}
fmt.Println(string(data))
}
}
func checkErr(err error) {
if err != nil {
fmt.Println(err)
os.Exit(-1)
}
}
----------------------------------------------------------------------rpc编程
RPC--romote procedure call protocol远程调用,类似调用本地函数一样调用服务器的函数,不需要了解底层网络细节的应用程序通信协议。RPC构建与TCP或UDP,或者是HTTP之上。采用客户端/服务器的工作模式,
当执行一个远程过程调用时,客户端首先发送一个带有参数的调用信息到服务端,等待服务器的响应。在服务端,服务进程保存睡眠状态直到客户端的调用信息到达为止。组合客户端接收来自服务端的应答信息,获得进程结果
必须格式:
func (t *T) MethodName(argType T1,replyType *T2)error{}
第一个参数是RPC客户端要传入的参数;第二个参数是要返回给RPC客户端的结果。
服务端一般 通过TCP或HTTP在某个网络地址上进行监听来创建该服务
net/rpc rpc.Dail() 和 rpc.DialHTTP() 方法来与指定的RPC服务端建立连接。
调用RPC客户端的 Call()方法进行同步处理 Go()进行异步处理,即rpc客户端不用等待服务端的结果即可执行后面的程序
若在rpc传输过程中不指定编码解码器,默认使用go标准库的 encoding/gob
-----------------------------------------rpcserver.go
package server
type Args struct {
A, B int
}
type Quotient struct {
Quo, Rem int
}
type Arith int
func (t *Arith) Multiply(args *Args, reply *int) error {
*reply = args.A * args.B
return nil
}
func (t *Arith) Divide(args *Args, quo *Quotient) error {
if args.B == 0 {
return errors.New("divide by zero")
}
quo.Quo = args.A / args.B
quo.Rem = args.A % args.B
return nil
}
//注册服务对象并开启该 RPC 服务的代码如下:
arith := new(Arith)
rpc.Register(arith)
rpc.HandleHTTP()
l, e := net.Listen("tcp", ":1234")
if e != nil {
log.Fatal("listen error:", e)
}
go http.Serve(l, nil)
此时, RPC 服务端注册了一个Arith类型的对象及其公开方法Arith.Multiply()和
Arith.Divide()供 RPC 客户端调用。 RPC 在调用服务端提供的方法之前,必须先与 RPC 服务
端建立连接,如下列代码所示:
client, err := rpc.DialHTTP("tcp", serverAddress + ":1234")
if err != nil {
log.Fatal("dialing:", err)
}
在建立连接之后, RPC 客户端可以调用服务端提供的方法。首先,我们来看同步调用程序顺
序执行的方式:
args := &server.Args{7,8}
var reply int
err = client.Call("Arith.Multiply", args, &reply)
if err != nil {
log.Fatal("arith error:", err)
}
fmt.Printf("Arith: %d*%d=%d", args.A, args.B, reply)
此外,还可以以异步方式进行调用,具体代码如下:
quotient := new(Quotient)
divCall := client.Go("Arith.Divide", args, "ient, nil)
replyCall := <-divCall.Done
---------------------------------
package main
import (
"fmt"
"log"
"net"
"net/http"
"net/rpc"
)
type Echo int
//被注册的对象至少要有一个方法满足下面这个特征,才会被导出到rpc服务接口
func (t *Echo) Hi(args string, reply *string) error {
*reply = "echo" + args
return nil
}
func main() {
//注册rpc服务,默认名字为对象的类型Echo,rpc.RegisterName指定特殊名字
rpc.Register(new(Echo))
//指定rpc的传输协议,这里用http作为rpc调用的载体。也可用rpc.ServeConn接口,定制自己的传输协议
rpc.HandleHTTP()
l, e := net.Listen("tcp", ":1234")
if e != nil {
log.Fatal("listen error:", err)
}
http.Serve(l, nil)
}
/*
客户端的调用方式
func main(){
client,_:=rpc.DialHTTP("tcp", "127.0.0.1:1234")
var args="hello rpc"
var reply string
client.Call("Echo.Hi", args, &reply)
fmt.Printf("arith:%d*%d=%d\n", args.A,args.B,reply)
}
*/
----------------------------------------
type Watcher int
func (w *Watcher) GetInfo(arg int, result *int) error {
*result = 1
return nil
}
func main() {
http.HandleFunc("/login", Handler)
watcher := new(Watcher)
rpc.Register(watcher)
rpc.HandleHTTP()
l, err := net.Listen("tcp", ":1234")
if err != nil {
fmt.Println("监听失败,端口可能已经被占用")
}
fmt.Println("正在监听1234端口")
http.Serve(l, nil)
}
func Handler(w http.ResponseWriter, r *http.Request) {
io.WriteString(w, "传输的html文档")
}
---------------
func main() {
client, err := rpc.DialHTTP("tcp", "127.0.0.1:1234")
if err != nil {
fmt.Println("链接rpc服务器失败:", err)
}
var reply int
err = client.Call("Watcher.GetInfo", 1, &reply)
if err != nil {
fmt.Println("调用远程服务失败", err)
}
fmt.Println("远程服务返回结果:", reply)
}
web服务器开发,处理客户端请求
(1)表单
在Javascript 中,页面上的每一对