Go语言网络库net/http

这篇具有很好参考价值的文章主要介绍了Go语言网络库net/http。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Go语言网络库net/http

Http 协议(Hyper Text Transfer Protocol,超文本传输协议)是一个简单的请求-响应协议,它通常运行在 TCP 之

上。超文本传输协议是互联网上应用最为广泛的一种网络传输协议,所有的WWW文件都必须遵守这个标准。

Http 协议是基于客户端 Cilent /服务器 Server 模式,且面向连接的。简单的来说就是客户端 Cilent 向服务器

Server 发送 http 请求 Request,服务器 Server 接收到 http 服务请求 Request 后会在 http 响应Response 中回

送所请求的数据。

Go语言内置的 net/http 包十分的优秀,提供了HTTP客户端和服务端的实现。

官网地址:https://pkg.go.dev/net/http

1、HTTP客户端

通过 Get、Head、Post 和 PostForm 函数发出 HTTP/HTTPS 请求。

resp, err := http.Get("http://example.com/")

resp, err := http.Post("http://example.com/upload", "image/jpeg", &buf)

resp, err := http.PostForm("http://example.com/form",url.Values{"key": {"Value"}, "id": {"123"}})

1.1 发送Get请求

不带请求参数:

package main

import (
	"fmt"
	"io/ioutil"
	"net/http"
)

func main() {
	resp, err := http.Get("http://www.baidu.com")
	if err != nil {
		fmt.Printf("get failed, err:%v\n", err)
		return
	}
	// 程序在使用完response后必须关闭回复的主体
	defer resp.Body.Close()
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		fmt.Printf("read from resp.Body failed, err:%v\n", err)
		return
	}
	fmt.Print(string(body))
}

带参数,参数需要使用到 net/url 来处理。

package main

import (
	"fmt"
	"io/ioutil"
	"net/http"
	"net/url"
)

func main() {
	apiUrl := "http://127.0.0.1:9090/get"
	// URL param
	data := url.Values{}
	data.Set("name", "mi")
	data.Set("age", "18")
	u, err := url.ParseRequestURI(apiUrl)
	if err != nil {
		fmt.Printf("parse url requestUrl failed, err:%v\n", err)
	}
	// URL encode
	u.RawQuery = data.Encode()
	// http://127.0.0.1:9090/get?age=18&name=mi
	fmt.Println(u.String())
	resp, err := http.Get(u.String())
	if err != nil {
		fmt.Printf("post failed, err:%v\n", err)
		return
	}
	defer resp.Body.Close()
	b, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		fmt.Printf("get resp failed, err:%v\n", err)
		return
	}
	// {"status": "ok"}
	fmt.Println(string(b))
}

对应 server 端的处理:

package main

import (
	"fmt"
	"net/http"
)

func getHandler(w http.ResponseWriter, r *http.Request) {
	defer r.Body.Close()
	data := r.URL.Query()
    // mi
	fmt.Println(data.Get("name"))
    // 18
	fmt.Println(data.Get("age"))
	answer := `{"status": "ok"}`
	w.Write([]byte(answer))
}

func main() {
	http.HandleFunc("/get", getHandler)
	err := http.ListenAndServe(":9090", nil)
	if err != nil {
		fmt.Printf("http server failed, err:%v\n", err)
		return
	}
}

带 header 头部信息:

package main

import (
	"fmt"
	"net/http"
	"net/http/httputil"
)

func main() {
	url := "http://127.0.0.1:9090/get"
	request, err := http.NewRequest(http.MethodGet, url, nil)
	if err != nil {
		panic(err)
	}
	request.Header.Add("Authorization", "jhs8723sd2dshd2")
	request.Header.Add("User-Agent", "User-Agent: Mozilla/5.0 (iPhone; CPU iPhone OS 13_2_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/13.0.3 Mobile/15E148 Safari/604.1")
	resp, err := http.DefaultClient.Do(request)
	if err != nil {
		panic(err)
	}
	defer resp.Body.Close()
	// //获取网页内容
	s, err := httputil.DumpResponse(resp, true)
	if err != nil {
		panic(err)
	}
	fmt.Printf("%s", s)
}

1.2 发送POST请求

package main

import (
	"fmt"
	"io/ioutil"
	"net/http"
	"strings"
)

func main() {
	url := "http://127.0.0.1:9090/post"
	// 表单数据
	// contentType := "application/x-www-form-urlencoded"
	// data := "name=mi&age=18"
	// json
	contentType := "application/json"
	data := `{"name":"mi","age":18}`
	resp, err := http.Post(url, contentType, strings.NewReader(data))
	if err != nil {
		fmt.Printf("post failed, err:%v\n", err)
		return
	}
	defer resp.Body.Close()
	b, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		fmt.Printf("get resp failed, err:%v\n", err)
		return
	}
	// {"status": "ok"}
	fmt.Println(string(b))
}

对应 server 端:

package main

import (
	"fmt"
	"io/ioutil"
	"net/http"
)

func postHandler(w http.ResponseWriter, r *http.Request) {
	defer r.Body.Close()
	// 1. 请求类型是application/x-www-form-urlencoded时解析form数据
	r.ParseForm()
	// 打印form数据
	// map[]
	fmt.Println(r.PostForm)
	fmt.Println(r.PostForm.Get("name"), r.PostForm.Get("age"))
	// 2. 请求类型是application/json时从r.Body读取数据
	b, err := ioutil.ReadAll(r.Body)
	if err != nil {
		fmt.Printf("read request.Body failed, err:%v\n", err)
		return
	}
	// {"name":"mi","age":18}
	fmt.Println(string(b))
	answer := `{"status": "ok"}`
	w.Write([]byte(answer))
}

func main() {
	http.HandleFunc("/post", postHandler)
	err := http.ListenAndServe(":9090", nil)
	if err != nil {
		fmt.Printf("http server failed, err:%v\n", err)
		return
	}
}

1.3 PostForm方式

package main

import (
	"fmt"
	"io/ioutil"
	"net/http"
	"net/url"
)

func main() {
	resp, err := http.PostForm("http://127.0.0.1:9090/post",
		url.Values{"key": {"Value"}, "id": {"123"}})
	if err != nil {
		fmt.Println(err)
	}
	defer resp.Body.Close()
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		fmt.Println(err)
	}
	fmt.Println(string(body))
}

json:

package main

import (
	"bytes"
	"encoding/json"
	"fmt"
	"io/ioutil"
	"net/http"
)

func main() {
	data := make(map[string]interface{})
	data["site"] = "www.baidu.com"
	data["name"] = "tom"
	bytesData, _ := json.Marshal(data)
	resp, _ := http.Post("http://httpbin.org/post", "application/json", bytes.NewReader(bytesData))
	body, _ := ioutil.ReadAll(resp.Body)
	fmt.Println(string(body))
}

带有 headers 参数:

package main

import (
	"fmt"
	"io/ioutil"
	"net/http"
	"strings"
)

func main() {
	client := &http.Client{}
	req, err := http.NewRequest("POST", "http://www.01happy.com/demo/accept.php", strings.NewReader("name=cjb"))
	if err != nil {
		fmt.Println(err)
	}
	req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
	req.Header.Set("Cookie", "name=anny")
	resp, err := client.Do(req)
	defer resp.Body.Close()
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		fmt.Println(err)
	}
	fmt.Println(string(body))
}

1.4 自定义Client

要管理 HTTP 客户端的头域、重定向策略和其他设置,创建一个 Client:

package main

import (
	"fmt"
	"net/http"
	"net/http/httputil"
)

func main() {
	request, err := http.NewRequest(http.MethodGet, "http://www.baidu.com", nil)
	if err != nil {
		panic(err)
	}
	client := http.Client{
		CheckRedirect: func(req *http.Request, via []*http.Request) error {
			fmt.Println("Redirect:", req)
			return nil
		},
	}
	request.Header.Add("User-Agent", "User-Agent: Mozilla/5.0 (iPhone; CPU iPhone OS 13_2_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/13.0.3 Mobile/15E148 Safari/604.1")
	resp, err := client.Do(request)
	if err != nil {
		panic(err)
	}
	defer resp.Body.Close()
	s, err := httputil.DumpResponse(resp, true)
	fmt.Printf("%s", s)
}

1.5 自定义Transport

要管理代理、TLS 配置、keep-alive、压缩和其他设置,创建一个Transport:

package main

import (
	"crypto/tls"
	"crypto/x509"
	"fmt"
	"io/ioutil"
	"net/http"
)

// 使用匹配的证书方法
func main() {
	// *.pem文件的内容
	rootCA := ""
	roots := x509.NewCertPool()
	ok := roots.AppendCertsFromPEM([]byte(rootCA))
	if !ok {
		panic("failed to parse root certificate")
	}
	client := &http.Client{
		Transport: &http.Transport{
			TLSClientConfig:    &tls.Config{RootCAs: roots},
			DisableCompression: true,
		},
	}
	resp, err := client.Get("http://baidu.com")
	if err != nil {
		panic(err)
	}
	defer resp.Body.Close()
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		panic(err)
	}
	if err != nil {
		fmt.Printf("read from resp.Body failed, err:%v\n", err)
		return
	}
	fmt.Print(string(body))
}

Client 和 Transport 类型都可以安全的被多个 goroutine 同时使用。出于效率考虑,应该一次建立、尽量重用。

package main

import (
	"crypto/tls"
	"fmt"
	"io/ioutil"
	"net/http"
)

// 跳过安全检查
func main() {
	tr := &http.Transport{
		TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
	}
	client := &http.Client{Transport: tr}
	resp, err := client.Get("https://localhost:8081")
	if err != nil {
		fmt.Println("error:", err)
		return
	}
	defer resp.Body.Close()
	body, err := ioutil.ReadAll(resp.Body)
	fmt.Println(string(body))
}

2、服务端

2.1 默认的Server

ListenAndServe 使用指定的监听地址和处理器启动一个 HTTP 服务端,处理器参数通常是 nil,这表示采用包变量

DefaultServeMux 作为处理器。

Handle 和 HandleFunc 函数可以向 DefaultServeMux 添加处理器。

package main

import (
	"log"
	"net/http"
)

type httpServer struct {
}

func (server httpServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	w.Write([]byte(r.URL.Path))
}

func main() {
	var server httpServer
	http.Handle("/foo", server)
	http.HandleFunc("/bar", func(w http.ResponseWriter, r *http.Request) {
		w.Write([]byte(r.URL.Path))
	})
	log.Fatal(http.ListenAndServe(":9090", nil))
}

事件处理器的 Handler 接口定义如下:

type Handler interface {
	ServeHTTP(ResponseWriter, *Request)
}

只要实现了这个接口,就可以实现自己的 handler 处理器。

如果 ListenAndServe() 传入的第一个参数地址为空,则服务器在启动后默认使用 http://127.0.0.1:8080 地址

进行访问。

如果这个函数传入的第二个参数为 nil,则服务器在启动后将使用默认的多路复用器DefaultServeMux

2.2 默认的Server示例

使用 Go 语言中的 net/http 包来编写一个简单的接收 HTTP 请求的 Server 端示例,net/http 包是对 net 包的

进一步封装,专门用来处理 HTTP 协议的数据。具体的代码如下:

package main

import (
	"fmt"
	"net/http"
)

func sayHello(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintln(w, "Hello shanghai!")
}

func main() {
	http.HandleFunc("/", sayHello)
	err := http.ListenAndServe(":9090", nil)
	if err != nil {
		fmt.Printf("http server failed, err:%v\n", err)
		return
	}
}

将上面的代码编译之后执行,打开你电脑上的浏览器在地址栏输入127.0.0.1:9090回车:

# 输出
Hello shanghai!
package main

import (
	"encoding/json"
	_ "encoding/json"
	"fmt"
	"io/ioutil"
	"net/http"
	"net/url"
	"strings"
)

var (
	api   = "http://127.0.0.1:9090/"
	token = "9adf92861"
)

type Result struct {
	Code int    `json:"code"`
	Msg  string `json:"msg"`
}

type Resp struct {
	Code string `json:"code"`
	Msg  string `json:"msg"`
}

func wechatUrlHandle(w http.ResponseWriter, r *http.Request) {
	// 获取链接地址
	var wechatUrl string
	values := r.URL.Query()
	wechatUrl = values.Get("url")
	// 验证URL的正确性
	targetUrl, err := url.ParseRequestURI(wechatUrl)
	if err != nil {
		fmt.Printf("parse url failed, err:%v\n", err)
	}
	fmt.Printf(fmt.Sprintf("url: %v", targetUrl))
	// 拼接参数发起请求
	data := fmt.Sprintf("token=%s&url=%s", token, targetUrl.String())
	headers := "application/x-www-form-urlencoded"
	res, err := http.Post(api, headers, strings.NewReader(data))
	if err != nil {
		fmt.Printf("post failed, err: %v\n", err)
		return
	}
	// 关闭请求资源
	defer res.Body.Close()
	body, err := ioutil.ReadAll(res.Body)
	if err != nil {
		fmt.Fprintf(w, fmt.Sprintf("get resp failed. err: %v\n", err))
		return
	}
	var info Result
	var ress Resp
	w.Header().Set("Content-Type", "application/json")
	if err := json.Unmarshal(body, &info); err == nil {
		fmt.Println(info.Code, info.Msg)
		ress.Code = "200"
		ress.Msg = "Success"
		t, _ := json.Marshal(ress)
		w.Write(t)
	} else {
		ress.Code = "200"
		ress.Msg = "failure"
		t, _ := json.Marshal(ress)
		fmt.Println("err: ", err)
		w.Write(t)
	}
}

func main() {
	http.HandleFunc("/wechat/url", wechatUrlHandle)
	// 处理器参数一般为nil
	err := http.ListenAndServe(":9090", nil)
	if err != nil {
		fmt.Printf("Http server failed. err: %v\n", err)
		return
	}

}

2.3 自定义Server

要管理服务端的行为,可以创建一个自定义的 Server。

用户可以通过 Server 结构体对服务器进行更详细的配置,包括设置地址,为请求读取操作设置超过时间等等。

package main

import (
	"log"
	"net/http"
	"time"
)

type httpServer struct {
}

func (server httpServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	w.Write([]byte(r.URL.Path))
}

func main() {
	s := &http.Server{
		Addr:           ":9090",
		Handler:        httpServer{},
		ReadTimeout:    10 * time.Second,
		WriteTimeout:   10 * time.Second,
		MaxHeaderBytes: 1 << 20,
	}
	log.Fatal(s.ListenAndServe())
}

2.4 多路复用器

多路复用器的基本原理:根据请求的 URL 地址找到对应的处理器,调用处理器对应的 ServeHTTP() 方法处理请

求。

DefaultServeMux 是 net/http 包的默认多路复用器,其实就是 ServeMux 的一个实例。

HandleFunc() 函数用于为指定的 URL 注册一个处理器,HandleFunc() 处理器函数会在内部调用

DefaultServeMux 对象对应的 HandleFunc 方法。

2.4.1 ServeMux与DefaultServeMux

我们可以使用默认多路复用器注册多个处理器,达到与处理器一样的作用。

下面演示如何通过默认多路复用器创建自己的服务器。

package main

import (
	"fmt"
	"net/http"
)

//定义多个处理器
type handle1 struct{}

func (h1 *handle1) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "handle one")
}

type handle2 struct{}

func (h2 *handle2) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "handle two")
}

func main() {
	handle1 := handle1{}
	handle2 := handle2{}
	// Handler:nil表明服务器使用默认的多路复用器DefaultServeMux
	s := &http.Server{
		Addr:    "127.0.0.1:8080",
		Handler: nil,
	}
	// Handle函数调用的是多路复用器DefaultServeMux.Handle方法
	http.Handle("/handle1", &handle1)
	http.Handle("/handle2", &handle2)
	s.ListenAndServe()
}

我们通过使用自己的 handle1 和 handle2 来指定两个处理器,http.Handle() 函数可以调用

DefaultServeMux.Handle() 方法来处理请求。

Handle: nil 对应的是处理器是DefaultServeMux。

在ServeMux对象的ServeHTTP()方法中,根据URL查找我们注册的服务器然后请求交给它处理。

虽然默认的多路复用器很好用,但仍然不推荐使用,因为它是一个全局变量,所有的代码都可以修改它。有些第三

方库中可能与默认复用器产生冲突。所以推荐的做法是自定义。

2.4.2 自定义多路复用器
package main

import (
	"fmt"
	"net/http"
)

func newservemux(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "NewServeMux")
}

func main() {
	mux := http.NewServeMux()
	mux.HandleFunc("/", newservemux)
	s := &http.Server{
		Addr:    ":8081",
		Handler: mux,
	}
	s.ListenAndServe()
}

NewServeMux 实质上还是 ServeMux。

2.4.3 ServeMux的路由匹配

我们现在需要绑定三个URL分别为/,/happy,/bad。

package main

import (
	"fmt"
	"net/http"
)

func newservemux(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "NewServeMux")
}

func newservemuxhappy(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "Newservemuxhappy")
}

func newservemuxbad(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "NewServeMuxbad")
}
func main() {
	mux := http.NewServeMux()
	mux.HandleFunc("/", newservemux)
	mux.HandleFunc("/happy", newservemuxhappy)
	mux.HandleFunc("/bad", newservemuxbad)
	s := &http.Server{
		Addr:    ":8080",
		Handler: mux,
	}

	s.ListenAndServe()
}

2.5 HttpRouter

ServeMux 的一个缺陷是无法使用变量实现 URL 模式匹配,而 HttpRouter 可以,HttpRouter 是一个高性能的第

三方 HTTP 路由包,弥补了 net/http 包中的路由不足问题。

$ go get github.com/julienschmidt/httprouter

httprouter 的使用首先得使用 New() 函数,生成一个 *Router 路由对象,然后使用 GET(),方法去注册匹配的函

数,最后再将这个参数传入 http.ListenAndServe 函数就可以监听服务。

package main

import (
	"net/http"

	"github.com/julienschmidt/httprouter"
)

func Hello(w http.ResponseWriter, r *http.Request, _ httprouter.Params) {
	w.Write([]byte("Hello,httprouter!"))
}

func main() {
	router := httprouter.New()
	router.GET("/", Hello)
	http.ListenAndServe(":8080", router)
}

更为重要的是,它为 URL 提供了两种匹配模式:

  • /user/:pac:精准匹配 /user/pac
  • /user/*pac:匹配所有模式 /user/hello
Pattern: /user/:user

 /user/gordon              match
 /user/you                 match
 /user/gordon/profile      no match
 /user/                    no match
Pattern: /src/*filepath

 /src/                     match
 /src/somefile.go          match
 /src/subdir/somefile.go   match
package main

import (
	"fmt"
	"log"
	"net/http"

	"github.com/julienschmidt/httprouter"
)

func Index(w http.ResponseWriter, r *http.Request, _ httprouter.Params) {
	fmt.Fprint(w, "Welcome!\n")
}

func Hello(w http.ResponseWriter, r *http.Request, ps httprouter.Params) {
	fmt.Fprintf(w, "hello, %s!\n", ps.ByName("name"))
}

func main() {
	router := httprouter.New()
	router.GET("/", Index)
	router.GET("/hello/:name", Hello)
	log.Fatal(http.ListenAndServe(":8080", router))
}

当然这里还有 POST(),DELETE() 函数的详情,就不一一介绍了。

3、其它应用

3.1 DNS

package main

import (
	"fmt"
	"log"
	"net"
)

func main() {
	// 根据域名返回ip地址
	addr, err := net.ResolveIPAddr("ip", "devops.miliantech.com")
	if err != nil {
		return
	}
	// 2023/07/24 20:45:43 devops.miliantech.com 对应地址ip地址----> 172.21.6.96
	log.Println("devops.miliantech.com 对应地址ip地址---->", addr)
	// 查找DNS A记录
	iprecords, _ := net.LookupIP("devops.miliantech.com")
	for _, ip := range iprecords {
		// LookupIP -----> 172.21.6.96
		fmt.Println("LookupIP ----->", ip)
	}
	//查找DNS CNAME记录
	canme, _ := net.LookupCNAME("devops.miliantech.com")
	// LookupCNAME -----> devops.miliantech.com.
	fmt.Println("LookupCNAME ----->", canme)
	//查找DNS PTR记录
	ptr, e := net.LookupAddr("172.21.6.96")
	if e != nil {
		// lookup 172.21.6.96: dnsquery: DNS name does not exist.
		fmt.Println(e)
	} else {
		fmt.Println(ptr)
	}
	for _, ptrval := range ptr {
		fmt.Println(ptrval)
	}
	//查找DNS NS记录 名字服务器记录
	nameserver, _ := net.LookupNS("baidu.com")
	for _, ns := range nameserver {
		/*
			ns记录 &{ns3.baidu.com.}
			ns记录 &{ns2.baidu.com.}
			ns记录 &{ns7.baidu.com.}
			ns记录 &{ns4.baidu.com.}
			ns记录 &{dns.baidu.com.}
		*/
		fmt.Println("ns记录", ns)
	}
	//查找DNS MX记录 邮件服务器记录
	mxrecods, _ := net.LookupMX("google.com")
	for _, mx := range mxrecods {
		fmt.Println("mx:", mx)
	}
	//查找DNS TXT记录 域名对应的文本信息
	txtrecords, _ := net.LookupTXT("baidu.com")

	for _, txt := range txtrecords {
		/*
			txt: _globalsign-domain-verification=qjb28W2jJSrWj04NHpB0CvgK9tle5JkOq-EcyWBgnE
			txt: google-site-verification=GHb98-6msqyx_qqjGl5eRatD3QTHyVB6-xQ3gJB5UwM
			txt: v=spf1 include:spf1.baidu.com include:spf2.baidu.com include:spf3.baidu.com include:spf4.baidu.com mx ptr -all
		*/
		fmt.Println("txt:", txt)
	}

	//查看本地IP地址
	addrs, err := net.InterfaceAddrs()
	if err != nil {
		return
	}
	for _, address := range addrs {
		// 检查ip地址判断是否回环地址
		if ipnet, ok := address.(*net.IPNet); ok && !ipnet.IP.IsLoopback() {
			if ipnet.IP.To4() != nil {
				/*
					本地地址 -- > 169.254.124.27
					本地地址 -- > 169.254.27.66
					本地地址 -- > 169.254.158.118
					本地地址 -- > 192.168.223.1
					本地地址 -- > 192.168.132.1
					本地地址 -- > 192.168.226.185
				*/
				fmt.Println("本地地址 -- >", ipnet.IP.String())
			}
		}
	}
}

3.2 简单的tcp请求实现

3.2.1 服务端
package main

import (
	"bufio"
	"io"
	"log"
	"net"
)

func pes(conn net.Conn) {
	defer conn.Close()
	read := bufio.NewReader(conn)
	for {
		var p [512]byte
		n, err := read.Read(p[:])
		if err != nil {
			if err == io.EOF {
				log.Println("客户端关闭链接", conn.RemoteAddr().String())
				return
			}
			log.Println("读取失败", err.Error())
			return
		}
		log.Println("客户端发送消息为", string(p[:n]))
		conn.Write([]byte("ok"))
	}
}

func main() {
	lis, err := net.Listen("tcp", "127.0.0.1:20003")
	if err != nil {
		log.Println("listen ,err ", err.Error())
		return
	}

	for {
		conn, err := lis.Accept()
		if err != nil {
			log.Println("建立链接异常", err.Error())
		}
		go pes(conn)
	}
}
3.2.2 客户端
package main

import (
	"bufio"
	"io"
	"log"
	"net"
	"os"
	"strings"
)

func main() {
	conn, err := net.Dial("tcp", "127.0.0.1:20003")
	if err != nil {
		log.Println("conn err", err.Error())
		return
	}
	defer conn.Close()
	inputreader := bufio.NewReader(os.Stdin)
	for {
		input, _ := inputreader.ReadString('\n')

		input = strings.Trim(input, "\r\n")

		if input == "q" || input == "Q" {
			log.Println("退出链接")
			return
		}

		conn.Write([]byte(input))
		var p [512]byte
		n, err := conn.Read(p[:])
		if err != nil {
			if err == io.EOF {
				log.Println("服务端关闭链接", conn.RemoteAddr().String())
				return
			}
			log.Println("读取失败", err.Error())
			return
		}
		log.Println("服务端发送消息为", string(p[:n]))
	}

}

3.3 粘包问题

一次只能读取固定长度的数据,解决方案:文章来源地址https://www.toymoban.com/news/detail-606203.html

package main

import (
	"bufio"
	"bytes"
	"encoding/binary"
	"fmt"
)

func Encode(massge string) ([]byte, error) {
	len_for_msaaget := int32(len(massge))
	pkg := new(bytes.Buffer)
	err := binary.Write(pkg, binary.LittleEndian, len_for_msaaget)
	if err != nil {
		return nil, err
	}
	err = binary.Write(pkg, binary.LittleEndian, []byte(massge))
	if err != nil {
		return nil, err
	}
	return pkg.Bytes(), nil
}

func Decode(reader *bufio.Reader) (string, error) {
	peek, err := reader.Peek(4)
	if err != nil {
		return "", err
	}
	lengthBuff := bytes.NewBuffer(peek)

	var length int32
	err = binary.Read(lengthBuff, binary.LittleEndian, &length)

	if err != nil {
		return "", err
	}
	if int32(reader.Buffered()) <= 4+length {

		return "", fmt.Errorf("err")
	}
	p := make([]byte, int(4+length))
	_, err = reader.Read(p)
	if err != nil {
		return "", err
	}
	return string(p[4:]), nil
}

3.4 简单的udp请求实现

3.4.1 服务端
package main

import (
	"log"
	"net"
)

func main() {
	udp, err := net.ListenUDP("udp", &net.UDPAddr{
		IP:   net.IPv4(0, 0, 0, 0),
		Port: 30000,
	})
	if err != nil {
		return
	}

	defer udp.Close()
	for {
		var p [1024]byte
		n, addr, err := udp.ReadFromUDP(p[:])
		if err != nil {
			log.Println("udp 请求处理异常 ", err.Error())
			return
		}

		log.Println("请求信息", string(p[:n]), addr.String())

		_, err = udp.WriteToUDP([]byte("get"), addr)
		if err != nil {
			log.Println("返回信息失败", err.Error())
			return
		}

	}
}
3.4.2 客户端
package main

import (
	"log"
	"net"
)

func main() {
	udp, err := net.DialUDP("udp", nil, &net.UDPAddr{
		IP:   net.IPv4(0, 0, 0, 0),
		Port: 30000,
	})
	if err != nil {
		log.Println(err.Error())
		return
	}

	defer udp.Close()
	_, err = udp.Write([]byte("我是服务端, hello server"))
	if err != nil {
		log.Println(err.Error())
		return
	}

	data := make([]byte, 4096)
	fromUDP, u, err := udp.ReadFromUDP(data)
	if err != nil {
		log.Println("接收失败", err)
		return
	}

	log.Println("接收信息返回成功 ", string([]byte(data[:fromUDP])), u)

}

到了这里,关于Go语言网络库net/http的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Go语言网络编程(socket编程)http编程

    Web服务器的工作原理可以简单地归纳为 客户机通过TCP/IP协议建立到服务器的TCP连接 客户端向服务器发送HTTP协议请求包,请求服务器里的资源文档 服务器向客户机发送HTTP协议应答包,如果请求的资源包含有动态语言的内容,那么服务器会调用动态语言的解释引擎负责处理“

    2024年02月09日
    浏览(64)
  • Go语言的TCP和HTTP网络服务基础

    目录 【TCP Socket 编程模型】 Socket读操作 【HTTP网络服务】 HTTP客户端 HTTP服务端 TCP/IP 网络模型实现了两种传输层协议:TCP 和 UDP,其中TCP 是面向连接的流协议,为通信的两端提供稳定可靠的数据传输服务;UDP 提供了一种无需建立连接就可以发送数据包的方法。实现网络编程,

    2024年02月12日
    浏览(47)
  • Go语言网络编程:HTTP服务端之底层原理与源码分析——http.HandleFunc()、http.ListenAndServe()

    在 Golang只需要几行代码便能启动一个 http 服务,在上述代码中,完成了两件事: 调用 http.HandleFunc 方法,注册了对应于请求路径 /ping 的 handler 函数 调用 http.ListenAndServe,启动了一个端口为 8999 的 http 服务 2.1 server 结构 Addr :表示服务器监听的地址。如\\\":8080\\\"表示服务器在本地

    2024年02月08日
    浏览(58)
  • Go语言网络编程入门:TCP、HTTP、JSON序列化、Gin、WebSocket、RPC、gRPC示例

    在本文中,我们将介绍Go语言中的网络编程的不同方式,包括TCP、HTTP、Gin框架、WebSocket、RPC、gRPC的介绍与连接实例,并对所有示例代码都给出了详细的注释,最后对每种模式进行了总结。 TCP(传输控制协议)是一种面向连接的、可靠的、基于字节流的传输层通信协议,提供

    2024年02月16日
    浏览(56)
  • 7天玩转 Golang 标准库之 http/net

    在构建web应用时,我们经常需要处理HTTP请求、做网页抓取或者搭建web服务器等任务,而Go语言在这方面为我们提供了强大的内置工具:net/http标准库,它为我们操作和处理HTTP协议提供了便利。 首先,我们来看看如何使用net/http标准库发送一个HTTP请求。net/http库中的 http.Get 函数

    2024年02月04日
    浏览(46)
  • Go For Web:Golang http 包详解(源码剖析)

    本文作为解决如何通过 Golang 来编写 Web 应用这个问题的前瞻,对 Golang 中的 Web 基础部分进行一个简单的介绍。目前 Go 拥有成熟的 Http 处理包,所以我们去编写一个做任何事情的动态 Web 程序应该是很轻松的,接下来我们就去学习了解一些关于 Web 的相关基础,了解一些概念,

    2023年04月14日
    浏览(34)
  • Go net http包

    在Go中,搭建一个HTTP server简单到令人难以置信。只需要引入net/http包,写几行代码,一个HTTP服务器就可以正常运行并接受访问请求。 下面就是Go最简单的HTTP服务器: 编译运行程序,然后打开浏览器访问 http://localhost:8080/ , 我们可以看到网页输出\\\"hi\\\" ! 就这么简单,我们实现了

    2024年02月08日
    浏览(42)
  • Golang:Go语言结构

    在我们开始学习 Go 编程语言的基础构建模块前,让我们先来了解 Go 语言最简单程序的结构。 Go 语言的基础组成有以下几个部分: 包声明 引入包 函数 变量 语句 表达式 注释 接下来让我们来看下简单的代码,该代码输出了\\\"Hello World!\\\": 让我们来看下以上程序的各个部分: 第一

    2024年02月10日
    浏览(56)
  • golang 发起 http 请求,获取访问域名的 ip 地址(net, httptrace)

    今天碰到了个需求,我要知道程序对外访问的 http 请求域名的 ip 地址。 直接查看 golang 的 net/http 包,发现 Response 中并没有我想要的 ip 信息。 考虑到在 OSI 七层模型中, ip 是网络层协议,而 http 是应用层协议。去翻了下 net 包,发现了基础用法。 先提一下,域名访问服务器

    2024年02月02日
    浏览(50)
  • 【Golang 接口自动化02】使用标准库net/http发送Post请求

    目录 写在前面 发送Post请求 示例代码 源码分析 Post请求参数解析 响应数据解析 验证 发送Json/XMl Json请求示例代码 xml请求示例代码 总结 资料获取方法 上一篇我们介绍了使用  net/http  发送get请求,因为考虑到篇幅问题,把Post单独拎了出来,我们在这一篇一起从源码来了解一

    2024年02月14日
    浏览(59)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包