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. SpringBoot学习(一):SpringBoot入门

    1.Spring Boot 简介 1) 简化Spring应用开发的一个框架: 2) 整个Spring技术栈的一个大整合: 3) J2EE开发的一站式解决方案: 2.微服务 2014,martin fo ...

  2. SVN: 在Ecplise管理SVN资源库

    Window->Show View->SVN

  3. CentOS 7 上安装 Django 2.2.4,解决报错:No module named ‘_sqlite3′

    1.首先下载最新版的sqlite :https://www.sqlite.org/download.html 下载源码包: 配置和编译方法如下: ./configure --prefix=/usr/l ...

  4. 树莓派4b点亮led灯基本步骤

    方法/步骤1: 首先要了解树莓派上的针脚,下面以树莓派4b为例子 把LED的正极插在GPIO脚上,把负极插在GND上 这里的例子是:正极插在GPIO21 方法/步骤2: 创建脚本 在配置好的树莓派系统 ...

  5. 实验13:VLAN/TRUNK/VTP/

    实验10-1: 划分VLAN Ø    实验目的通过本实验,读者可以掌握如下技能:(1) 熟悉VLAN 的创建(2) 把交换机接口划分到特定VLAN Ø    实验拓扑 实验步骤要配置VLAN,首先要 ...

  6. CCF_201312-4_有趣的数

    dp题,dp[i][j]代表i位数,j状态的数量.其中,j 的状态表示值有6种. 0 1 2     √ j = 0 3 01 02   √ j = 1 03 12 13 23   √ j = 2 0 ...

  7. what can we do if just only want to truncate transaction log without backup ?

    n some circumstances, we just want to truncate transaction log without backup and refuce change data ...

  8. Python常用库 - logging日志库

    logging的简单介绍 用作记录日志,默认分为六种日志级别(括号为级别对应的数值) NOTSET(0) DEBUG(10) INFO(20) WARNING(30) ERROR(40) CRITIC ...

  9. 【C++】应用程序无法正常启动0xc000007b

    在Windows平台编程时,或运行应用程序时,偶尔会遇到“应用程序无法正常启动0xc000007b”或“缺少***.dll”的问题, 首先需要考虑的就是程序相关联的dll有没有放到系统环境中,dll通 ...

  10. NR / 5G - The Proportional Fair algorithm