網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
使用Go語(yǔ)言寫(xiě)一個(gè)Http?Server的實(shí)現(xiàn)_Golang
作者:虎鯨不是魚(yú) ? 更新時(shí)間: 2022-06-24 編程語(yǔ)言Http Server 代碼
go.mod:
module goStudy1 go 1.17
main.go:
package main import ( "fmt" "os" "strconv" //"github.com/thinkeridea/go-extend/exnet" "io" "log" "net/http" "strings" ) /* 編寫(xiě)一個(gè) HTTP 服務(wù)器,4個(gè)功能: 1,接收客戶端 request,并將 request 中帶的 header 寫(xiě)入 response header 2,讀取當(dāng)前系統(tǒng)的環(huán)境變量中的 VERSION 配置,并寫(xiě)入 response header 3,Server 端記錄訪問(wèn)日志包括客戶端 IP,HTTP 返回碼,輸出到 server 端的標(biāo)準(zhǔn)輸出 4,當(dāng)訪問(wèn) localhost/healthz 時(shí),應(yīng)返回 200 */ // Main方法入口 func main() { println("環(huán)境正常") // 功能1 http.HandleFunc("/requestAndResponse", requestAndResponse) // 功能2 http.HandleFunc("/getVersion", getVersion) // 功能3 http.HandleFunc("/ipAndStatus", ipAndStatus) //注冊(cè)接口句柄 // 功能4 http.HandleFunc("/healthz", healthz) //注冊(cè)接口句柄 err := http.ListenAndServe(":81", nil) //監(jiān)聽(tīng)空句柄,80端口被占用,使用81端口 if nil != err { log.Fatal(err) //顯示錯(cuò)誤日志 } } // 功能1,接收請(qǐng)求及響應(yīng) func requestAndResponse(response http.ResponseWriter, request *http.Request) { println("調(diào)用requestAndResponse接口") headers := request.Header //header是Map類型的數(shù)據(jù) println("傳入的hander:") for header := range headers { //value是[]string //println("header的key:" + header) values := headers[header] for index, _ := range values { values[index] = strings.TrimSpace(values[index]) //println("index=" + strconv.Itoa(index)) //println("header的value:" + values[index]) } //valueString := strings.Join(values, "") //println("header的value:" + valueString) println(header + "=" + strings.Join(values, ",")) //打印request的header的k=v response.Header().Set(header, strings.Join(values, ",")) // 遍歷寫(xiě)入response的Header //println() } fmt.Fprintln(response, "Header全部數(shù)據(jù):", headers) io.WriteString(response, "succeed") } // 功能2,獲取環(huán)境變量的version func getVersion(response http.ResponseWriter, request *http.Request) { println("調(diào)用getVersion接口") envStr := os.Getenv("VERSION") //envStr := os.Getenv("HADOOP_HOME") //println("系統(tǒng)環(huán)境變量:" + envStr) //可以看到 C:\soft\hadoop-3.3.1 Win10需要重啟電腦才能生效 response.Header().Set("VERSION", envStr) io.WriteString(response, "succeed") } // 功能3,輸出IP與返回碼 func ipAndStatus(response http.ResponseWriter, request *http.Request) { println("調(diào)用ipAndStatus接口") form := request.RemoteAddr println("Client->ip:port=" + form) //虛擬機(jī)是橋接模式。使用postman返回的全部是127.0.0.1 用手機(jī)打開(kāi)網(wǎng)站192.168.1.139:81/ipAndStatus可以看到新IP ipStr := strings.Split(form, ":") println("Client->ip=" + ipStr[0]) //打印ip // 獲取http響應(yīng)碼 //response.WriteHeader(301) //手動(dòng)設(shè)置響應(yīng)碼,默認(rèn)200 //response.WriteHeader(http.StatusOK)//由于默認(rèn)是調(diào)用這個(gè),∴返回碼都是這個(gè)200【server.go有源碼】 println("Client->response code=" + strconv.Itoa(http.StatusOK)) //println("response code->:" + code) io.WriteString(response, "succeed") } // 功能4,連通性測(cè)試接口 func healthz(response http.ResponseWriter, request *http.Request) { println("調(diào)用healthz接口") response.WriteHeader(200) //設(shè)置返回碼200 //response.WriteHeader(http.StatusOK)//默認(rèn)會(huì)調(diào)用這個(gè)方法,默認(rèn)就是200【server.go有源碼】 io.WriteString(response, "succeed") }
由于80端口被占用,使用了81端口。
調(diào)試
由于Linux虛擬機(jī)沒(méi)有安裝go環(huán)境,只有windows有g(shù)o環(huán)境,使用goland開(kāi)發(fā)后,用postman調(diào)試。
功能1
網(wǎng)站:http://127.0.0.1:81/requestAndResponse
POST的request中額外配置了 k2=v1 。Send后可以看到:
Response中出現(xiàn)了手動(dòng)新增的請(qǐng)求頭及其它默認(rèn)的請(qǐng)求頭。【原始的response只有3對(duì)kv結(jié)果,已經(jīng)遍歷添加成功】。說(shuō)明成功寫(xiě)入。
功能2
由于Windows需要重啟才能刷新環(huán)境變量,故:
//envStr := os.Getenv("VERSION") envStr := os.Getenv("HADOOP_HOME")
測(cè)試時(shí),此處讀取已經(jīng)存在的環(huán)境變量,原理是一致的。
網(wǎng)站:http://127.0.0.1:81/getVersion
說(shuō)明Go可以讀取到環(huán)境變量的值,并且寫(xiě)入response的headers。
功能3
網(wǎng)站:http://127.0.0.1:81/ipAndStatus
分別用postman、手機(jī)請(qǐng)求這個(gè)網(wǎng)站【手機(jī)請(qǐng)求時(shí)需要和PC在同一個(gè)路由,將網(wǎng)站IP更換為PC的IP才可以訪問(wèn)】,goland中顯示:
環(huán)境正常
調(diào)用getVersion接口
調(diào)用ipAndStatus接口
Client->ip:port=127.0.0.1:59595
Client->ip=127.0.0.1
Client->response code=200
調(diào)用ipAndStatus接口
Client->ip:port=192.168.1.138:37548
Client->ip=192.168.1.138
Client->response code=200
顯然讀取到了client的IP。由于server.go中有寫(xiě):
// WriteHeader sends an HTTP response header with the provided // status code. // // If WriteHeader is not called explicitly, the first call to Write // will trigger an implicit WriteHeader(http.StatusOK). // Thus explicit calls to WriteHeader are mainly used to // send error codes. // // The provided code must be a valid HTTP 1xx-5xx status code. // Only one header may be written. Go does not currently // support sending user-defined 1xx informational headers, // with the exception of 100-continue response header that the // Server sends automatically when the Request.Body is read. WriteHeader(statusCode int)
默認(rèn)的響應(yīng)頭就是取返回值為200,不設(shè)置就是按照默認(rèn)的200來(lái)返回,故此處的響應(yīng)碼為200。
由于響應(yīng)體引用的請(qǐng)求體并不包含返回碼,如果直接從響應(yīng)體的請(qǐng)求中拿返回碼【request.Response.StatusCode】,會(huì)報(bào)內(nèi)存錯(cuò)誤及空指針的panic。
功能4
網(wǎng)址:http://127.0.0.1:81/healthz
使用postman調(diào)用接口,可以看到:
默認(rèn)的響應(yīng)體的響應(yīng)頭的返回碼就是200。且返回值3個(gè)。
可以看出,Go相比Java還是很簡(jiǎn)潔的。
原文鏈接:https://blog.csdn.net/qq_41990268/article/details/124334227
相關(guān)推薦
- 2022-11-14 CSS樣式中選擇器+盒子模型+定位+浮動(dòng)
- 2024-04-08 SpringBoot使用Mybatis-Plus中分頁(yè)出現(xiàn)total=0的情況解決
- 2022-01-21 Flink中window 窗口和時(shí)間以及watermark水印
- 2023-11-11 Flask 表單form.validate_on_submit()什么情況下會(huì)是false——解決辦
- 2022-06-16 C語(yǔ)言從猜數(shù)字游戲中理解數(shù)據(jù)結(jié)構(gòu)_C 語(yǔ)言
- 2022-05-26 Nginx多個(gè)前端服務(wù)配置方式詳解_nginx
- 2022-08-14 Gradle?Build?Cache引發(fā)的Task緩存編譯問(wèn)題_Android
- 2023-07-22 linux自動(dòng)化運(yùn)維之linux腳本自動(dòng)操作redis數(shù)據(jù)
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過(guò)濾器
- Spring Security概述快速入門(mén)
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支