当前位置: 首页 > news >正文

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 mainimport ("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 mainimport ("fmt""io/ioutil""net/http""net/url"
)func main() {apiUrl := "http://127.0.0.1:9090/get"// URL paramdata := 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 encodeu.RawQuery = data.Encode()// http://127.0.0.1:9090/get?age=18&name=mifmt.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 mainimport ("fmt""net/http"
)func getHandler(w http.ResponseWriter, r *http.Request) {defer r.Body.Close()data := r.URL.Query()// mifmt.Println(data.Get("name"))// 18fmt.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 mainimport ("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 mainimport ("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"// jsoncontentType := "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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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 httpServerhttp.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 mainimport ("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 mainimport ("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 stringvalues := 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 Resultvar ress Respw.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)// 处理器参数一般为nilerr := http.ListenAndServe(":9090", nil)if err != nil {fmt.Printf("Http server failed. err: %v\n", err)return}}

2.3 自定义Server

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

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

package mainimport ("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 mainimport ("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表明服务器使用默认的多路复用器DefaultServeMuxs := &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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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.96log.Println("devops.miliantech.com 对应地址ip地址---->", addr)// 查找DNS A记录iprecords, _ := net.LookupIP("devops.miliantech.com")for _, ip := range iprecords {// LookupIP -----> 172.21.6.96fmt.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-EcyWBgnEtxt: google-site-verification=GHb98-6msqyx_qqjGl5eRatD3QTHyVB6-xQ3gJB5UwMtxt: 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 mainimport ("bufio""io""log""net"
)func pes(conn net.Conn) {defer conn.Close()read := bufio.NewReader(conn)for {var p [512]byten, 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 mainimport ("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]byten, 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 粘包问题

一次只能读取固定长度的数据,解决方案:

package mainimport ("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 int32err = 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 mainimport ("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]byten, 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 mainimport ("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

Go语言网络库net/http Http 协议(Hyper Text Transfer Protocol&#xff0c;超文本传输协议)是一个简单的请求-响应协议&#xff0c;它通常运行在 TCP 之 上。超文本传输协议是互联网上应用最为广泛的一种网络传输协议&#xff0c;所有的WWW文件都必须遵守这个标准。 Http 协…...

Acwing.91 最短Hamilton路径(动态规划)

题目 给定一张n个点的带权无向图&#xff0c;点从0~n-1标号&#xff0c;求起点0到终点n-1的最短Hamilton路径。Hamilton路径的定义是从0到n-1不重不漏地经过每个点恰好一次。 输入格式 第—行输入整数n。 接下来n行每行n个整数&#xff0c;其中第i行第j个整数表示点i到j的距…...

[hfut] [important] v4l2 vedio使用总结/opevx/ffpeg/v4l2/opencv/cuda

(158条消息) linux驱动camera//test ok_感知算法工程师的博客-CSDN博客 (158条消息) linux V4L2子系统——v4l2架构&#xff08;1&#xff09;之整体架构_感知算法工程师的博客-CSDN博客 (158条消息) linux V4L2子系统——v4l2的结构体&#xff08;2&#xff09;之video_devi…...

2023年深圳杯数学建模A题影响城市居民身体健康的因素分析

2023年深圳杯数学建模 A题 影响城市居民身体健康的因素分析 原题再现&#xff1a; 以心脑血管疾病、糖尿病、恶性肿瘤以及慢性阻塞性肺病为代表的慢性非传染性疾病&#xff08;以下简称慢性病&#xff09;已经成为影响我国居民身体健康的重要问题。随着人们生活方式的改变&am…...

指令调度(Instruction Scheduling)

指令调度&#xff08;Instruction Scheduling&#xff09; 指令调度的约束基本机器模型基本块调度全局调度 指令调度是为了提高指令级并行&#xff08;ILP&#xff09;&#xff0c;对于超长指令字&#xff08;VLIW, Very Long Instruction Word&#xff09;和多发射系统&#x…...

【运维】Linux 跨服务器复制文件文件夹

【运维】Linux 跨服务器复制文件文件夹 如果是云服务 建议用内网ip scp是secure copy的简写&#xff0c;用于在Linux下进行远程拷贝文件的命令&#xff0c;和它类似的命令有cp&#xff0c;不过cp只是在本机进行拷贝不能跨服务器&#xff0c;而且scp传输是加密的。可能会稍微影…...

k8s apiserver如何支持http访问?

原本是可以通过设置api-server的--insecure-port来实现&#xff0c;但是这个参数已经被废弃了&#xff0c;更好的方法则是使用proxy来实现&#xff1a; 在集群任意一个节点上起一个proxy服务&#xff0c;并设置允许所有host访问&#xff1a; kubectl proxy --address0.0.0.0 …...

Safetensors,高效安全易用的深度学习新工具

大家好&#xff0c;本文将介绍一种为深度学习应用提供速度、效率、跨平台兼容性、用户友好性和安全性的新工具。 Safetensors简介 Hugging Face开发了一种名为Safetensors的新序列化格式&#xff0c;旨在简化和精简大型复杂张量的存储和加载。张量是深度学习中使用的主要数据…...

Unity 工具之 NuGetForUnity 包管理器,方便在 Unity 中的进行包管理的简单使用

Unity 工具之 NuGetForUnity 包管理器&#xff0c;方便在 Unity 中的进行包管理的简单使用 目录 Unity 工具之 NuGetForUnity 包管理器&#xff0c;方便在 Unity 中的进行包管理的简单使用 一、简单介绍 二、NuGetForUnity 的下载导入 Unity 三、NuGetForUnity 在 Unity 的…...

运算放大器(二):恒流源

一、实现原理 恒流源的输出电流能够在一定范围内保持稳定&#xff0c;不会随负载的变化而变化。 通过运放&#xff0c;将输入的电压信号转换成满足一定关系的电流信号&#xff0c;转换后的电流相当一个输出可调的简易恒流源。 二、电路结构 常用的恒流源电路如…...

企业选择租用CRM还是一次性买断CRM?分别有哪些优势?

CRM是企业管理客户关系&#xff0c;提升销售业绩&#xff0c;实现业务增长的重要工具。市场上的CRM系统销售方式主要有两种——租用型和买断型。那么&#xff0c;租用CRM好还是一次性买断CRM好&#xff1f;本文将从以下几个方面进行分析&#xff1a; 1、什么是租用型CRM和买断…...

VBA_MF系列技术资料1-133

MF系列VBA技术资料 为了让广大学员在实际VBA编程中有切实可行的思路及有效的提高自己的编程技巧&#xff0c;我参考大量的资料&#xff0c;并结合自己的经验总结了这份MF系列VBA技术综合资料&#xff0c;而且开放源码&#xff08;MF04除外&#xff09;&#xff0c;其中MF01-04属…...

Android 项目架构

🔥 什么是架构 🔥 在维基百科里是这样定义的: 软件架构是一个系统的轮廓 . 软件架构描述的对象是直接构成系统的抽象组件. 各个组件之间的连接则明确和相对细致地描述组件之间的通讯 . 在实现阶段, 这些抽象组件被细化为实际组件 , 比如具体某个类或者对象 . 面试的过程中…...

【Linux】进程通信 — 管道

文章目录 &#x1f4d6; 前言1. 通信背景1.1 进程通信的目的&#xff1a;1.2 管道的引入&#xff1a; 2. 匿名管道2.1 匿名管道的原理&#xff1a;2.2 匿名管道的创建&#xff1a;2.3 父子进程通信&#xff1a;2.3.1 read()阻塞等待 2.4 父进程给子进程派发任务&#xff1a;2.5…...

ROS 2 — 托管(生命周期)节点简介

一、说明 这篇文章是关于理解ROS 2中托管&#xff08;生命周期&#xff09;节点的概念。我们描述了概念性的想法以及我们为什么需要它。所以让我们开始吧&#xff01; 二、托管式节点 — 什么和为什么&#xff1f; 为了理解托管式节点&#xff0c;让我们从一个简单的问题陈述开…...

vue2企业级项目(六)

vue2企业级项目&#xff08;六&#xff09; 自定义指令 创建src/directive/index.js const directives require.context("./modules", true, /\.js$/);export default {install: (Vue) > {directives.keys().forEach((key) > {let directive directives(key…...

OSPF的选路原则

域内 --- 1类&#xff0c;2类LSA 域间 --- 3类LSA 域外 --- 5类&#xff0c;7类LSA --- 根据开销值的计算规则不同&#xff0c;还分为类型1和类型2 1.如果学到的路由都是通过1类&#xff0c;2类LSA获取的域内路由 --- 这种情况直接比较开销值&#xff0c;优先选择开销值小的路…...

4.操作元素属性

4.1操作元素常用属性 ●通过 JS 设置/修改 标签元素属性&#xff0c;比如通过src更换图片 ●最常见的属性比如&#xff1a;href、 title、 src 等 ●语法: 对象.属性 值【示例】 // 1.获取元素 const pic document.querySelector( img ) // 2.操作元素 pic.src ./images/b…...

uniapp 微信小程序:v-model双向绑定问题(自定义 props 名无效)

uniapp 微信小程序&#xff1a;v-model双向绑定问题&#xff08;自定义 props 名无效&#xff09; 前言问题双向绑定示例使用 v-model使用 v-bind v-on使用 sync 修饰符 参考资料 前言 VUE中父子组件传递数据的基本套路&#xff1a; 父传子 props子传父 this.$emit(事件名, …...

【Lua学习笔记】Lua进阶——Table(3) 元表

接上文 文章目录 元表__tostring__call__index__newindex运算符元方法其它元表操作 元表 Q&#xff1a;为什么要使用元表&#xff1f; A&#xff1a;在Lua中&#xff0c;常常会需要表与表之间的操作。元表中提供了一些元方法&#xff0c;通过自定义元方法可以实现想要的功能&…...

AI编程常用工具 Jupyter Notebook

点击上方蓝色字体&#xff0c;选择“设为星标” 回复”云原生“获取基础架构实践 深度学习编程常用工具 我们先来看 4 个常用的编程工具&#xff1a;Sublime Text、Vim、Jupyter。虽然我介绍的是 Jupyter&#xff0c;但并不是要求你必须使用它&#xff0c;你也可以根据自己的喜…...

RocketMQ重复消费的解决方案::分布式锁直击面试!

文章目录 场景分析方法的幂等分布式锁Redis实现分布式锁抢锁的设计思路 分布式锁案例 直击面试rocketmq什么时候重复消费消息丢失的问题消息在哪里丢失发送端确保发送成功并且配合失败的业务处理消费端确保消息不丢失rocketmq 主从同步刷盘 场景分析 分布式系统架构中,队列是分…...

如何降低TCP在局域网环境下的数据传输延迟

以Ping为例。本案例是一个测试题目&#xff0c;只有现象展示&#xff0c;不含解决方案。 ROS_Kinetic_26 使用rosserial_windows实现windows与ROS master发送与接收消息_windows 接收ros1 消息 什么是ping&#xff1f; AI&#xff1a; ping是互联网控制消息协议&#xff08;…...

【LeetCode】78.子集

题目 给你一个整数数组 nums &#xff0c;数组中的元素 互不相同 。返回该数组所有可能的子集&#xff08;幂集&#xff09;。 解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3] 输出&#xff1a;[[],[1],[2],[1…...

认可功能介绍 - 技术声誉靠认可

需求 大家在学习和工作中&#xff0c; 经常碰到一些热心帮助自己的人&#xff0c; 我们怎么向他们表示感谢呢&#xff1f; 各位博主在 CSDN 也做了很多贡献&#xff0c;也有不少用户在做各种各样的社区活动&#xff0c;这些活动给我们的领军人物什么回馈呢&#xff1f; 这些…...

EtherNet/IP转CAN网关can协议标准

生产管理设备中&#xff0c;会有设备与其他设备的协议不同&#xff0c;数据无法互通&#xff0c;让你的工作陷入困境。这时&#xff0c;一款神奇的产品出现了——远创智控YC-EIP-CAN通讯网关&#xff01; 1, 这款通讯网关采用ETHERNET/IP从站功能&#xff0c;可以将各种CAN总线…...

解决代理IP负载均衡与性能优化的双重挑战

在当今数字化时代&#xff0c;代理IP的应用范围日益广泛&#xff0c;它不仅在数据爬取、网络抓取等领域发挥着重要作用&#xff0c;也成为网络安全和隐私保护的有力工具。然而&#xff0c;面对庞大的数据流量和复杂的网络环境&#xff0c;如何实现代理IP的负载均衡和性能优化成…...

深度探索 Elasticsearch 8.X:function_score 参数解读与实战案例分析

在 Elasticsearch 中&#xff0c;function_score 可以让我们在查询的同时对搜索结果进行自定义评分。 function_score 提供了一系列的参数和函数让我们可以根据需求灵活地进行设置。 近期有同学反馈&#xff0c;function_score 的相关参数不好理解&#xff0c;本文将深入探讨 f…...

测牛学堂:软件测试之andorid app性能测试面试知识点总结(二)

APP性能测试指标之FPS 如果经常玩游戏的同学应该听过FPS。 FPS本来是图像领域中的概念&#xff0c;是指画面每秒传输的帧数。每秒钟帧数越多&#xff0c;所显示的动作就会越流畅。 但是因为功耗的限制&#xff0c;一般60fps就是跑满的效果了。 我们测试的话&#xff0c;一般…...

尚医通06:数据字典+EasyExcel+mongodb

内容介绍 1、数据字典列表前端 2、EasyExcel介绍、实例 3、数据字典导出接口、前端 4、数据字典导入接口、前端 5、数据字典添加redis缓存 6、MongoDB简介 7、MongoDB安装 8、MongoDB基本概念 数据字典列表前端 1、测试问题 &#xff08;1&#xff09;报错日志 &am…...