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"
)
/*
编写一个 HTTP 服务器,4个功能:
1,接收客户端 request,并将 request 中带的 header 写入 response header
2,读取当前系统的环境变量中的 VERSION 配置,并写入 response header
3,Server 端记录访问日志包括客户端 IP,HTTP 返回码,输出到 server 端的标准输出
4,当访问 localhost/healthz 时,应返回 200
*/
// Main方法入口
func main() {
println("环境正常")
// 功能1
http.HandleFunc("/requestAndResponse", requestAndResponse)
// 功能2
http.HandleFunc("/getVersion", getVersion)
// 功能3
http.HandleFunc("/ipAndStatus", ipAndStatus) //注册接口句柄
// 功能4
http.HandleFunc("/healthz", healthz) //注册接口句柄
err := http.ListenAndServe(":81", nil) //监听空句柄,80端口被占用,使用81端口
if nil != err {
log.Fatal(err) //显示错误日志
}
}
// 功能1,接收请求及响应
func requestAndResponse(response http.ResponseWriter, request *http.Request) {
println("调用requestAndResponse接口")
headers := request.Header //header是Map类型的数据
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, ",")) // 遍历写入response的Header
//println()
}
fmt.Fprintln(response, "Header全部数据:", headers)
io.WriteString(response, "succeed")
}
// 功能2,获取环境变量的version
func getVersion(response http.ResponseWriter, request *http.Request) {
println("调用getVersion接口")
envStr := os.Getenv("VERSION")
//envStr := os.Getenv("HADOOP_HOME")
//println("系统环境变量:" + 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("调用ipAndStatus接口")
form := request.RemoteAddr
println("Client->ip:port=" + form) //虚拟机是桥接模式。使用postman返回的全部是127.0.0.1 用手机打开网站192.168.1.139:81/ipAndStatus可以看到新IP
ipStr := strings.Split(form, ":")
println("Client->ip=" + ipStr[0]) //打印ip
// 获取http响应码
//response.WriteHeader(301) //手动设置响应码,默认200
//response.WriteHeader(http.StatusOK)//由于默认是调用这个,∴返回码都是这个200【server.go有源码】
println("Client->response code=" + strconv.Itoa(http.StatusOK))
//println("response code->:" + code)
io.WriteString(response, "succeed")
}
// 功能4,连通性测试接口
func healthz(response http.ResponseWriter, request *http.Request) {
println("调用healthz接口")
response.WriteHeader(200) //设置返回码200
//response.WriteHeader(http.StatusOK)//默认会调用这个方法,默认就是200【server.go有源码】
io.WriteString(response, "succeed")
}
由于80端口被占用,使用了81端口。
调试由于Linux虚拟机没有安装go环境,只有windows有go环境,使用goland开发后,用postman调试。
功能1网站:http://127.0.0.1:81/requestAndResponse
POST的request中额外配置了 k2=v1 。Send后可以看到:
Response中出现了手动新增的请求头及其它默认的请求头。【原始的response只有3对kv结果,已经遍历添加成功】。说明成功写入。
功能2由于Windows需要重启才能刷新环境变量,故:
//envStr := os.Getenv("VERSION")
envStr := os.Getenv("HADOOP_HOME")
测试时,此处读取已经存在的环境变量,原理是一致的。
网站:http://127.0.0.1:81/getVersion
说明Go可以读取到环境变量的值,并且写入response的headers。
功能3网站:http://127.0.0.1:81/ipAndStatus
分别用postman、手机请求这个网站【手机请求时需要和PC在同一个路由,将网站IP更换为PC的IP才可以访问】,goland中显示:
环境正常
调用getVersion接口
调用ipAndStatus接口
Client->ip:port=127.0.0.1:59595
Client->ip=127.0.0.1
Client->response code=200
调用ipAndStatus接口
Client->ip:port=192.168.1.138:37548
Client->ip=192.168.1.138
Client->response code=200
显然读取到了client的IP。由于server.go中有写:
// 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)
默认的响应头就是取返回值为200,不设置就是按照默认的200来返回,故此处的响应码为200。
由于响应体引用的请求体并不包含返回码,如果直接从响应体的请求中拿返回码【request.Response.StatusCode】,会报内存错误及空指针的panic。
功能4网址:http://127.0.0.1:81/healthz
使用postman调用接口,可以看到:
默认的响应体的响应头的返回码就是200。且返回值3个。
可以看出,Go相比Java还是很简洁的。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)