0、前言

Go语言中内置net/http包提供了HTTP客户端和服务端的实现

1、HTTP服务端

模拟一个HTTP服务端。

package main

import (
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
) // 定义客户端提交的post请求的json数据内容
type Auth struct {
Username string `json: username`
Password string `json: password`
} // 定义服务端返回json数据给客户端的内容
type Resp struct {
Code string `json: code`
Msg string `json: msg`
} func f1(w http.ResponseWriter,r *http.Request){
str := `from home`
w.Write([]byte(str))
} func f2(w http.ResponseWriter,r *http.Request){
b,err := ioutil.ReadFile("./html/index.html") // 读取到html文件(byte类型切片)
if err != nil {
w.Write([]byte(fmt.Sprintf("%v",err)))
}
w.Write(b) // 返回响应数据(必须传入byte类型切片)
} func f3(w http.ResponseWriter,r *http.Request){
// 对于GET请求,参数都放在URL上(query param),请求体中是没有数据的
queryParam := r.URL.Query() // 自动帮我们识别URL中的urlParam
query := queryParam.Get("query")
page := queryParam.Get("page")
fmt.Println(query,page)
fmt.Println(r.URL) // 查看请求url
fmt.Println(r.Method) // 查看请求方法
fmt.Println(ioutil.ReadAll(r.Body)) // 查看请求的body
w.Write([]byte("ok"))
} // post接口接收json数据
func f4(w http.ResponseWriter,r *http.Request){ // 检查是否为POST请求
if r.Method != "POST"{
w.WriteHeader(405) // 返回错误代码
return
}
body,_ := ioutil.ReadAll(r.Body)
//body_str := string(body)
//fmt.Println(body_str) var auth Auth
var result Resp
if err := json.Unmarshal(body,&auth);err == nil {
// 拿到json数据
fmt.Printf("用户名:%v 密码:%v",auth.Username,auth.Password) result.Code = "200"
result.Msg = "Success"
// 将返回的数据转化成json格式
ret,_ := json.Marshal(result)
w.Write(ret)
}else{
result.Code = "500"
result.Msg = "Failed"
ret,_ := json.Marshal(result)
w.Write(ret)
}
} func main(){
http.HandleFunc("/home",f1)
http.HandleFunc("/index",f2)
http.HandleFunc("/xxx",f3)
http.HandleFunc("/login",f4) // 启动HTTP服务(监听地址和端口)
http.ListenAndServe("0.0.0.0:9090",nil)
}

2、HTTP客户端

HTTP客户端能够发送HTTP请求,如:GET/POST

2.1、GET请求示例

package main

import (
"fmt"
"io/ioutil"
"net/http"
"net/url"
) // net/http client func main(){
resp,err := http.Get("http://127.0.0.1:9090/index")
if err != nil {
fmt.Printf("get url failed,err:%v\n",err)
return
}
body,err := ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Printf("read resp.Body failed,err:%v",err)
}
fmt.Println(string(body)) }

2.2、GET请求URL带参数示例

我们可以在发送Get请求的时候在url上携带参数,例如:http://xx/xx?query=xx&page=xx

package main

import (
"fmt"
"io/ioutil"
"net/http"
"net/url"
) // net/http client func main(){
//resp,err := http.Get("http://127.0.0.1:9090/xxx?query=jack&page=1")
data := url.Values{} // url encode(携带get请求参数)
urlObj,_ := url.Parse("http://127.0.0.1:9090/xxx")
data.Set("query","jack")
data.Set("page","1")
queryStr := data.Encode() // url encode之后的地址
fmt.Println(queryStr)
urlObj.RawPath = queryStr // 添加url
req,err := http.NewRequest("GET",urlObj.String(),nil) // 发送请求
resp,err := http.DefaultClient.Do(req) if err != nil {
fmt.Printf("get url failed,err:%v\n",err)
return
}
defer resp.Body.Close() // 关闭连接 body,err := ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Printf("read resp.Body failed,err:%v",err)
}
fmt.Println(string(body))
}

短连接

默认情况下浏览器开启了长连接,如果请求频繁的话,可能会存在长连接还没有关闭,又启动了新的连接,一直这样循环下去,就会导致连接超额,每个连接都会占用资源/网络IO,那么其实可以通过关闭长连接的方式来避免这个问题

// 禁用KeepAlive的client
tr := &http.Transport{
DisableKeepAlives: true,
}
client := http.Client{
Transport: tr,
}
client.Do(req)

2.3、POST请求携带Json数据示例1

很多时候,我们在实现POST请求都需要携带对应规范的json格式数据,例如

{
"username":"admin",
"password":"123456"
}

实现上面的规范来提交json数据

package main

import (
"bytes"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
) type auth struct {
Username string `json: username`
Password string `json: password`
} func main(){
// post请求
auths := auth{"admin","123456"}
bs,_ := json.Marshal(auths) // 将结构体数据转换成json格式
resp,_ := http.Post("http://127.0.0.1:9090/login","application/json", bytes.NewBuffer([]byte(bs)))
body, _ := ioutil.ReadAll(resp.Body)
fmt.Printf("Post request with json result: %s\n", string(body))
}

2.4、POST请求携带Json数据示例1

很多时候,我们在实现POST请求都需要携带对应规范的json格式数据,例如

{
"username":"admin",
"password":"123456"
}

实现上面的规范来提交json数据

package main

import (
"bytes"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
) // 我们需要在结构体中添加注解来映射对应的key
type auth struct {
Username string `json: username`
Password string `json: password`
} func main(){
// post请求
auths := auth{"admin","123456"}
bs,_ := json.Marshal(auths) // 将结构体数据转换成json格式
resp,_ := http.Post("http://127.0.0.1:9090/login","application/json", bytes.NewBuffer([]byte(bs)))
body, _ := ioutil.ReadAll(resp.Body)
fmt.Printf("Post request with json result: %s\n", string(body))
}

2.5、POST请求携带Json数据示例2

很多时候,我们在实现POST请求都需要携带对应规范的json格式数据,例如

{
"username":"admin",
"password":"123456"
}

实现上面的规范来提交json数据

package main

import (
"bytes"
"encoding/json"
"fmt"
"go_dev/Project/EyeSkyAgent/conf"
"io/ioutil"
"net/http"
) type auth struct {
Username string `json: username`
Password string `json: password`
} func main(){
// post请求
var data auth
data.Username = "Jack"
data.Password = "Jack123"
bs, err := json.Marshal(data) reader := bytes.NewReader(bs)
request, err := http.NewRequest("POST", "http://127.0.0.1:9090/login", reader)
if err != nil{
conf.Logger.Error("请求server端失败...")
}
// 携带头部
request.Header.Set("Content-Type", "application/json;charset=UTF-8")
client := http.Client{}
// 返回服务端的响应数据
resp, err := client.Do(request)
if err != nil {
fmt.Println("请求获取响应失败")
}
body, _ := ioutil.ReadAll(resp.Body)
fmt.Printf("Post request with json result: %s\n", string(body))
}

Golang模块之HTTP的更多相关文章

  1. Golang 模块(Module)官方手册

    官方原文: https://github.com/golang/go/wiki/Modules Go 1.11包括此处建议的对版本模块的初步支持.模块是Go 1.11中的实验性加入功能,并计划纳入反馈 ...

  2. golang模块viper读取配置文件

    一.介绍 Viper是一个方便Go语言应用程序处理配置信息的库.它可以处理多种格式的配置.它支持的特性: 设置默认值 从JSON.TOML.YAML.HCL和Java properties文件中读取配 ...

  3. golang常用模块介绍

    golang模块 一.命令行库Cobra Cobra提供简单的接口来创建强大的现代化CLI接口,比如git与go工具.Cobra同时也是一个程序, 用于创建CLI程序 https://www.jian ...

  4. 使用 Elastic Stack 来监控和调优 Golang 应用程序

    Golang 因为其语法简单,上手快且方便部署正被越来越多的开发者所青睐,一个 Golang 程序开发好了之后,势必要关心其运行情况,今天在这里就给大家介绍一下如果使用 Elastic Stack 来 ...

  5. Golang的模块管理Module

    Golang 1.11版本终于支持了官方的模块依赖管理功能,1.11以前想要实现依赖管理只能够通过借助第三方库来实现,1.11以前的版本Golang项目必须依赖以GOPATH,从当前版本开始Golan ...

  6. 用Golang为Python编写模块

    Go里面需要显示的引入C模块, 让编译器支持生成动态链接库, 并且在代码中可以使用C语言的数据类型,这个至关重要. Calling Go code from Python code 摘取一个最简单例子 ...

  7. 设计自用的golang日志模块

    设计自用的golang日志模块 golang的原生日志模块不能满足需求,而开源的第三方包,也不完全够用.用户较多的logrus,却没有rotate功能,这已经是众所周知的.对于运维来说,当然是希望日志 ...

  8. 治理Go模块 服务治理 中台业务 Golang的net.Conn接口,double close

    小结: 1.中台业务 前台业务 快车.专车.顺风车,在滴滴这些业务线叫做前台服务,他们有一些共同的特性,都有司机信息,订单的状态,收银,账号等等这些业务逻辑,我们会把专门的业务逻辑集合起来,形成专职的 ...

  9. Golang中基础的命令行模块urfave/cli

    前言相信只要部署过线上服务,都知道启动参数一定是必不可少的,当你在不同的网络.硬件.软件环境下去启动一个服务的时候,总会有一些启动参数是不确定的,这时候就需要通过命令行模块去解析这些参数,urfave ...

随机推荐

  1. 关于STM32串口接收中断中只能接收一个字节

    最近调试STM32的串口接收时发现例程中只能接收一个字节 例程如下: //初始化串口1 void uart_init(u32 bound){ //GPIO端口设置 GPIO_InitTypeDef G ...

  2. Solaris磁盘镜像恢复

    注:此文章笔者实验记录,欢迎大家指正 Solaris磁盘镜像恢复方法一: 系统启动,开机提示子镜像需要维护: 查看磁盘镜像信息 进入系统后,metastat -pc 和metadb #查看镜像状态与m ...

  3. ATL的GUI程序设计(3)

    第三章 ATL的窗口类 CWindowImpl.CWindow.CWinTraits,ATL窗口类的奥秘尽在此三者之中.在本章里,李马将为你详细解说它们的使用方法.另外,本章的内容也可以算是本书的核心 ...

  4. [C/C++]_[Unicode转Utf8,Ansi转Unicode,Ansi文件转Utf8文件]

    http://blog.csdn.net/infoworld/article/details/15337665 场景: 1.也就只有windows需要那么麻烦,还搞一个ANSI编码.学学mac os ...

  5. CCF_ 201403-4_无线网络

    分散点的bfs,先建立一个互相是否可达的二维数组,vis[i][j]代表到第i个点,走了j步的状态,注意判断新增路由器数量是否超过K. #include<cstdio> #include& ...

  6. c++ 中变量成员的初始化时机

    代码如下: 注意一下我打断点的位置. 最后的结果: 在程序进入MyTest()的函数体之前,控制台就打印出来了I have been constructed. 即:在进入构造函数的函数体之前,类中的成 ...

  7. k8s系列---dns部署

    1:首先创建kube-dns和dnsmasq这两个yaml,然后生成相应的pod.svc等. 2:然后在去创建其他的验证pod和svc 3:验证nslookup解析的是其他pod的svc的name,而 ...

  8. Kubernetes最新版核心命令

    #查看所有namespace的pods运行情况 kubectl get pods --all-namespaces #查看具体pods,记得后边跟namespace名字哦 kubectl get po ...

  9. Android 7.0新特性“Nougat”(牛轧糖)。

    1.Unicode 9支持和全新的emoji表情符号 Android Nougat将会支持Unicode 9,并且会新增大约70种emoji表情符号.这些表情符号大多数都是人形的,并且提供不同的肤色, ...

  10. 网页延迟加载动画的实现-WOW.js

    网页内容一开始不显示,随着鼠标下拉延迟显示,还有时间差 一开始觉得好难好复杂好高大上,直到我发现 wow.js …… 首先是演示地址:https://www.delac.io/wow/ 嗯,狗子确实很 ...