【GO】go语言入门实战 —— 命令行在线词典
文章目录
- 程序介绍
- 抓包
- 代码生成
- 生成request body
- 解析respond body
- 完整代码
字节青训营基础班学习记录。
程序介绍
在运行程序的时候以命令行的形式输入要查询的单词,然后程序返回单词的音标、释义等信息。
示例如下:
抓包
我们选择与网站https://fanyi.caiyunapp.com/
建立连接。
首先我们打开网站,Windows
下直接按f12
进入开发者工具,
然后输入一个单词并点击翻译按钮,在网络tag
部分找到请求方法为POST
的dict
:
这里有部分请求表头,是我们后续需要在代码中添加设置的。
然后去负载tag
可以看到这次请求的source
也就是要翻译的单词,
也能看到要翻译的类型,这里是en2zh
,也就是英文到中文:
这个请求是JSON
类型的,我们后续存储会用结构体去存储这个数据,所以传请求信息的时候要先对结构体进行序列化。
预览tag
就是这次请求返回的信息:
返回信息也是JSON
,为了方便打印我们后续也会将返回得到的JSON
反序列化存放到对应的结构体变量中。
代码生成
我们要实现一个在线词典,所以我们运行程序后需要程序发送对应的请求。
上面也看了,部分代码会特别复杂,
所以这里用到一个代码生成工具Convert curl to Go (curlconverter.com)帮我们完成部分代码工作。
鼠标指向刚刚的请求tag
-> 右键点击 -> 选择复制 -> 选择复制为cURL(cmd)
:
然后将我们复制后的信息粘贴到网站的curl command
下面的代码框内得到生成的go
语言代码:
此时可能会出现这样的报错信息:
这是因为我们复制的command
的每一行是以^
结尾的,我们需要手动改成\
,这样就没问题了:
然后把代码复制到我们的代码编辑器中就OK了。
下面添加了部分注释帮助理解代码:
package mainimport ("fmt""io""log""net/http""strings"
)func main() {//创建http客户端client := &http.Client{}//data是请求参数,也就是翻译类型和要翻译的单词//由于创建请求信息的时候请求参数是io.Reader流类型//所以需要用strings.NewReader()将JSON转化为io.Reader类型var data = strings.NewReader(`{"trans_type":"en2zh","source":"good"}`)//创建请求信息,需要传入请求方法,请求地址,请求参数req, err := http.NewRequest("POST", "https://api.interpreter.caiyunai.com/v1/dict", data)if err != nil {log.Fatal(err)}//添加请求头,根据需要添加req.Header.Set("authority", "api.interpreter.caiyunai.com")req.Header.Set("accept", "application/json, text/plain, */*")req.Header.Set("accept-language", "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6")req.Header.Set("app-name", "xy")req.Header.Set("content-type", "application/json;charset=UTF-8")req.Header.Set("device-id", "dde0d0432ca66c7fe6175b7bf940b7f2")req.Header.Set("origin", "https://fanyi.caiyunapp.com")req.Header.Set("os-type", "web")req.Header.Set("os-version", "")req.Header.Set("referer", "https://fanyi.caiyunapp.com/")req.Header.Set("sec-ch-ua", `\"Not/A)Brand\";v=\"99\", \"Microsoft Edge\";v=\"115\", \"Chromium\";v=\"115\"`)req.Header.Set("sec-ch-ua-mobile", "?0")req.Header.Set("sec-ch-ua-platform", `\"Windows\"`)req.Header.Set("sec-fetch-dest", "empty")req.Header.Set("sec-fetch-mode", "cors")req.Header.Set("sec-fetch-site", "cross-site")req.Header.Set("user-agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/115.0.0.0 Safari/537.36 Edg/115.0.1901.183")req.Header.Set("x-authorization", "token:qgemv4jr1y38jyq6vhvi")//发送请求resp, err := client.Do(req)if err != nil {log.Fatal(err)}//成功拿到请求后关闭请求,如果不关闭请求可能会造成资源泄漏//defer关键字会在函数结束时调用resp.Body.Close()方法defer resp.Body.Close()//读取请求返回的数据//这里使用ioutil.ReadAll()方法读取,返回的是[]byte类型的数据bodyText, err := io.ReadAll(resp.Body)if err != nil {log.Fatal(err)}//打印返回的信息fmt.Printf("%s\n", bodyText)
}
生成request body
上面的代码已经能够成功发送请求并接收序列化的返回信息。
但是现在有一个缺点,请求信息是一个写死的JSON
转成的流,
而我们需要从命令行自己输入单词,而单词又是请求信息的一部分,
所以我们需要定义一个能存放请求信息的结构体,结构体核心字段为trans_type
和source
:
我们是要从命令行中获取要翻译的单词,
所以可以从命令行参数中读取单词并用其初始化我们要创建的请求信息结构体变量,
trans_type
字段是我们要翻译的类型,需要初始化为en2zh
。
这样就可以用结构体来表示我们的请求信息。
但是发送请求时请求信息需要转为io.Reader
类型,
所以我们还需要先将结构体序列化,
然后使用func bytes.NewReader(b []byte) *bytes.Reader
将序列化后的JSON转成io.Reader
类型,
之后就可以创建请求。
// DictRequest 请求参数
// 因为需要序列化,所以结构体每个字段的首字母都要大写
type DictRequest struct {TransType string `json:"trans_type"`Source string `json:"source"`UserID string `json:"user_id"`
}//读取命令行参数,也就是我们要翻译的单词
word := os.Args[1]
//用我们读到的单词初始化请求信息结构体
request := DictRequest{TransType: "en2zh", Source: word} //将结构体序列化为json字符串
buf, err := json.Marshal(request)
if err != nil {log.Fatal(err)
}//将json字符串转换为io.Reader类型,因为NewRequest()方法需要传入io.Reader类型的参数
var data = bytes.NewReader(buf)
req, err := http.NewRequest("POST", "https://api.interpreter.caiyunai.com/v1/dict", data)
if err != nil {log.Fatal(err)
}
解析respond body
此时我们的程序已经可以翻译单个从命令行输入的单词,
并接收全部的返回信息:
请求返回的信息是JSON
序列,
而我们需要把这些信息反序列到一个结构体中,
然后打印出结构体中我们想要的部分信息。
所以我们还需要定义一个结构体用以接收返回信息反序列化的结果。
而返回信息中有太多字段了,一一去定义这些字段不仅低效并且容易出错,
所以这里使用到了另一款工具:JSON转Golang Struct - 在线工具 - OKTools。
首先还是打开开发者工具,找到POST
请求,点击预览:
然后单击鼠标右键选择复制值。
随后进入上面那个网站,将复制的内容 粘贴在左边一栏,然后点击转换嵌套:
这样就得到了对应的接收返回信息的结构体类型。
现在就可以把这段代码改个结构体的名添加到我们的代码中:
type DictResponse struct {Rc int `json:"rc"`Wiki struct {} `json:"wiki"`Dictionary struct {Prons struct {EnUs string `json:"en-us"`En string `json:"en"`} `json:"prons"`Explanations []string `json:"explanations"`Synonym []string `json:"synonym"`Antonym []string `json:"antonym"`WqxExample [][]string `json:"wqx_example"`Entry string `json:"entry"`Type string `json:"type"`Related []interface{} `json:"related"`Source string `json:"source"`} `json:"dictionary"`
}
此前代码生成部分接收的返回数据是以[]byte
类型在bodyTest
中存储的,
此时我们就可以将其反序列化为结构体对象,
然后选择性地打印我们所需要的字段:
//将返回的数据反序列化解析到结构体中
var dictResponse DictResponse
err = json.Unmarshal(bodyText, &dictResponse)
if err != nil {log.Fatal(err)
}//只需要打印我们需要的字段
//这里打印的是单词的音标和汉语翻译
fmt.Println(word, "UK:", dictResponse.Dictionary.Prons.En, "US:", dictResponse.Dictionary.Prons.EnUs)
for _, item := range dictResponse.Dictionary.Explanations {fmt.Println(item)
}
完整代码
此时将上面的代码整合一下就得到了完整的代码,
这里我们对代码进行了部分封装,
从main
函数中读取待翻译的单词,
然后将发送请求到接受返回信息并打印这一部分封装成query
函数,
将待翻译的单词作为参数,
如果有多个待翻译的单词可以在main
函数中一个个读取并调用query
函数 :
package mainimport ("bytes""encoding/json""fmt""io/ioutil""log""net/http""os"
)// DictRequest 请求参数
type DictRequest struct {TransType string `json:"trans_type"`Source string `json:"source"`UserID string `json:"user_id"`
}// DictResponse 返回参数
type DictResponse struct {Rc int `json:"rc"`Wiki struct {} `json:"wiki"`Dictionary struct {Prons struct {EnUs string `json:"en-us"`En string `json:"en"`} `json:"prons"`Explanations []string `json:"explanations"`Synonym []string `json:"synonym"`Antonym []string `json:"antonym"`WqxExample [][]string `json:"wqx_example"`Entry string `json:"entry"`Type string `json:"type"`Related []interface{} `json:"related"`Source string `json:"source"`} `json:"dictionary"`
}func query(word string) {//创建http客户端client := &http.Client{}//创建请求//POST请求,需要传入请求方法,请求地址,请求参数//这里的地址是我自己搭建的一个测试接口,大家可以自行搭建或者使用其他接口//data是请求参数,需要使用strings.NewReader()方法转换为io.Reader类型request := DictRequest{TransType: "en2zh", Source: word}//将结构体序列化为json字符串buf, err := json.Marshal(request)if err != nil {log.Fatal(err)}//将json字符串转换为io.Reader类型,因为NewRequest()方法需要传入io.Reader类型的参数var data = bytes.NewReader(buf)req, err := http.NewRequest("POST", "https://api.interpreter.caiyunai.com/v1/dict", data)if err != nil {log.Fatal(err)}//添加请求头,根据需要添加req.Header.Set("Connection", "keep-alive")req.Header.Set("DNT", "1")req.Header.Set("os-version", "")req.Header.Set("sec-ch-ua-mobile", "?0")req.Header.Set("User-Agent", "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99.0.4844.51 Safari/537.36")req.Header.Set("app-name", "xy")req.Header.Set("Content-Type", "application/json;charset=UTF-8")req.Header.Set("Accept", "application/json, text/plain, */*")req.Header.Set("device-id", "")req.Header.Set("os-type", "web")req.Header.Set("X-Authorization", "token:qgemv4jr1y38jyq6vhvi")req.Header.Set("Origin", "https://fanyi.caiyunapp.com")req.Header.Set("Sec-Fetch-Site", "cross-site")req.Header.Set("Sec-Fetch-Mode", "cors")req.Header.Set("Sec-Fetch-Dest", "empty")req.Header.Set("Referer", "https://fanyi.caiyunapp.com/")req.Header.Set("Accept-Language", "zh-CN,zh;q=0.9")req.Header.Set("Cookie", "_ym_uid=16456948721020430059; _ym_d=1645694872")//发送请求resp, err := client.Do(req)if err != nil {log.Fatal(err)}//成功拿到请求后关闭请求//这里需要注意,如果不关闭请求,会造成内存泄漏//defer关键字会在函数结束时调用resp.Body.Close()方法defer resp.Body.Close()//读取请求返回的数据//这里使用ioutil.ReadAll()方法读取,返回的是[]byte类型的数据bodyText, err := ioutil.ReadAll(resp.Body)if err != nil {log.Fatal(err)}//有可能会请求失败,这里判断一下状态码if resp.StatusCode != 200 {log.Fatal("bad StatusCode: ", resp.StatusCode, " body: ", string(bodyText))}//将返回的数据反序列化解析到结构体中var dictResponse DictResponseerr = json.Unmarshal(bodyText, &dictResponse)if err != nil {log.Fatal(err)}//只需要打印我们需要的字段fmt.Println(word, "UK:", dictResponse.Dictionary.Prons.En, "US:", dictResponse.Dictionary.Prons.EnUs)for _, item := range dictResponse.Dictionary.Explanations {fmt.Println(item)}
}func main() {// 当命令行没有信息传入时就没必要继续执行下去if os.Args == nil || len(os.Args) < 2 {fmt.Fprintf(os.Stderr, "usage: simpleDict WORDexample: simpleDict hello")os.Exit(1)}for _, word := range os.Args[1:] {query(word)fmt.Println() //每打印完一个单词的信息用空行隔开}
}
至此就完成了一个 简单的命令行在线词典。
相关文章:

【GO】go语言入门实战 —— 命令行在线词典
文章目录 程序介绍抓包代码生成生成request body解析respond body完整代码 字节青训营基础班学习记录。 程序介绍 在运行程序的时候以命令行的形式输入要查询的单词,然后程序返回单词的音标、释义等信息。 示例如下: 抓包 我们选择与网站https://fany…...

模电模电基础知识学习笔记汇总
来源:一周搞(不)定数电模电全集,电子基础知识 11小时 一:模电学习笔记 模电主要讲述:对模拟信号进行产生、放大和处理的模拟集成电路重点知识:常用电子元器件:电阻、电容、电感、保…...
招商银行秋招攻略和考试内容详解
招商银行秋招简介 招商银行是一家股份制商业银行,银行的服务理念已经深入人心,在社会竞争愈来愈烈的今天,招商银行的招牌无疑是个香饽饽,很多人也慕名而至,纷纷向招商银行投出了简历。那么秋招银行的秋招开始时间是多…...
【Linux】四、开发工具
一、vim 编辑器(只能写代码) 1、只关注如何写代码,不会关注代码的正确性; 2、一般写代码在Windows环境下写,而vim是Linux下相对来说功能最强的编辑器; 二、vim的操作 vim ---打开vim shift键 加 ࿱…...

前后端分离实现博客系统
文章目录 博客系统前言1. 前端1.1 登陆页面1.2 博客列表页面1.3 博客详情页面1.4 博客编辑页面 2. 后端2.1 项目部署2.1.1 创建maven项目2.1.2 引入依赖2.1.3 创建目录结构2.1.4 部署程序 2.2 逻辑设计2.2.1 数据库设计2.2.2 实体类设计2.2.3 Dao层设计2.2.3.1 BlogDao 2.2.4 D…...
面试题-TS(六):TypeScript 中的泛型是什么?
面试题-TS(6):TypeScript 中的泛型是什么? 在TypeScript中,泛型(Generics)是一种强大的特性,它允许我们在编写可重用的代码时增加灵活性。泛型使得我们可以编写不特定数据类型的代码,从而提高代…...

QT DAY4
1.思维导图 2.手动完成服务器的实现,并具体程序要注释清楚 头文件 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTcpServer> #include <QTcpSocket> #include <QMessageBox> #include <QList> #include <QD…...

最新Ai创作源码ChatGPT商用运营源码/支持GPT4.0+支持ai绘画+支持Mind思维导图生成
本系统使用Nestjs和Vue3框架技术,持续集成AI能力到本系统! 支持GPT3模型、GPT4模型Midjourney专业绘画(全自定义调参)、Midjourney以图生图、Dall-E2绘画Mind思维导图生成应用工作台(Prompt)AI绘画广场自定…...
一个go的支持多语言的error自动生成插件
大家好,我是peachesTao,今天给大家推荐一个go的支持多语言的error自动生成的插件,插件主页可以访问下方链接。 在一个多语言国际化的项目中,后端接口返回给前端的错误描述也需要国际化,我们来看一下后端给前端返回多语…...

wireshark抓包新手使用教程(超详细)
一、简介 Wireshark是一款非常流行的网络封包分析软件,可以截取各种网络数据包,并显示数据包详细信息。 为了安全考虑,wireshark只能查看封包,而不能修改封包的内容,或者发送封包。 wireshark能获取HTTP,也…...

平均列顺序对列排斥能的影响
( A, B )---3*30*2---( 1, 0 )( 0, 1 ) 让网络的输入只有3个节点,AB训练集各由5张二值化的图片组成,让A有6个1,B有4个1,并且让这10个1的位置没有重合。比较迭代次数的顺序。 其中有9组数据 差值结构 A-B 迭代次数 构造平均列 …...
微信小程序-处理ios无法播放语音的问题
背景 框架:tarovue3 问题:今天搞小程序语音播放功能,开放工具播放正常,但是到ios手机上调试时无法播放,在网上找到个好办法 解决方案 核心代码 Taro.setInnerAudioOption({obeyMuteSwitch: false // 解决有一些IOS无…...

区块链 2.0笔记
区块链 2.0 以太坊概述 相对于比特币的几点改进 缩短出块时间至10多秒ghost共识机制mining puzzle BTC:计算密集型ETH:memory-hard(限制ASIC) proof of work->proof of stake对智能合约的支持 BTC:decentralized currencyETH:decentral…...

深入理解Vue响应式系统:数据绑定探索
🌷🍁 博主猫头虎 带您 Go to New World.✨🍁 🦄 博客首页——猫头虎的博客🎐 🐳《面试题大全专栏》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺 &a…...

web流程自动化详解
今天给大家带来Selenium的相关解释操作 一、Selenium Selenium是一个用于自动化Web浏览器操作的开源工具和框架。它提供了一组API(应用程序接口),可以让开发人员使用多种编程语言(如Java、Python、C#等)编写测试脚本&…...

什么是框架?为什么要学框架?
一、什么是框架 框架是整个或部分应用的可重用设计,是可定制化的应用骨架。它可以帮开发人员简化开发过程,提高开发效率。 项目里有一部分代码:和业务无关,而又不得不写的代码>框架 项目里剩下的部分代码:实现业务…...
什么是 Sass?
Sass 介绍 什么是 Sass? 官方标语 世界上最成熟、最稳定、最强大的专业级 CSS 扩展语言。怎么理解这句话呢?我们平时写的 CSS 代码可以理解为静态样式语言,而 Scss 就是动态样式语言,何为动态?就是让你写 CSS 跟写 …...

Kotlin~Memento备忘录模式
概念 备忘录模式是一种行为型设计模式,用于捕获和存储对象的内部状态,并在需要时将对象恢复到之前的状态。 备忘录模式允许在不暴露对象内部实现细节的情况下,对对象进行状态的保存和恢复。 角色介绍 Originator:原发器&#x…...

单链表的多语言表达:C++、Java、Python、Go、Rust
单链表 是一种链式数据结构,由一个头节点和一些指向下一个节点的指针组成。每个节点包含一个数据元素和指向下一个节点的指针。头节点没有数据,只用于表示链表的开始位置。 单链表的主要操作包括: 添加元素:在链表的头部添加新…...
微信小程序 background-image直接设置本地图片路径,编辑器正常显示,真机运行不显示解决方法
项目场景 微信小程序,设置background-image直接设置本地图片路径。 问题描述 编辑器正常显示,真机运行不显示 原因分析 background-image只能用网络url或者base64图片编码。 解决方案 1、将本地图片转为网络url后设置到background-image上 例如&…...

基于ASP.NET+ SQL Server实现(Web)医院信息管理系统
医院信息管理系统 1. 课程设计内容 在 visual studio 2017 平台上,开发一个“医院信息管理系统”Web 程序。 2. 课程设计目的 综合运用 c#.net 知识,在 vs 2017 平台上,进行 ASP.NET 应用程序和简易网站的开发;初步熟悉开发一…...

k8s业务程序联调工具-KtConnect
概述 原理 工具作用是建立了一个从本地到集群的单向VPN,根据VPN原理,打通两个内网必然需要借助一个公共中继节点,ktconnect工具巧妙的利用k8s原生的portforward能力,简化了建立连接的过程,apiserver间接起到了中继节…...
稳定币的深度剖析与展望
一、引言 在当今数字化浪潮席卷全球的时代,加密货币作为一种新兴的金融现象,正以前所未有的速度改变着我们对传统货币和金融体系的认知。然而,加密货币市场的高度波动性却成为了其广泛应用和普及的一大障碍。在这样的背景下,稳定…...
Typeerror: cannot read properties of undefined (reading ‘XXX‘)
最近需要在离线机器上运行软件,所以得把软件用docker打包起来,大部分功能都没问题,出了一个奇怪的事情。同样的代码,在本机上用vscode可以运行起来,但是打包之后在docker里出现了问题。使用的是dialog组件,…...
在web-view 加载的本地及远程HTML中调用uniapp的API及网页和vue页面是如何通讯的?
uni-app 中 Web-view 与 Vue 页面的通讯机制详解 一、Web-view 简介 Web-view 是 uni-app 提供的一个重要组件,用于在原生应用中加载 HTML 页面: 支持加载本地 HTML 文件支持加载远程 HTML 页面实现 Web 与原生的双向通讯可用于嵌入第三方网页或 H5 应…...

day36-多路IO复用
一、基本概念 (服务器多客户端模型) 定义:单线程或单进程同时监测若干个文件描述符是否可以执行IO操作的能力 作用:应用程序通常需要处理来自多条事件流中的事件,比如我现在用的电脑,需要同时处理键盘鼠标…...
探索Selenium:自动化测试的神奇钥匙
目录 一、Selenium 是什么1.1 定义与概念1.2 发展历程1.3 功能概述 二、Selenium 工作原理剖析2.1 架构组成2.2 工作流程2.3 通信机制 三、Selenium 的优势3.1 跨浏览器与平台支持3.2 丰富的语言支持3.3 强大的社区支持 四、Selenium 的应用场景4.1 Web 应用自动化测试4.2 数据…...
DiscuzX3.5发帖json api
参考文章:PHP实现独立Discuz站外发帖(直连操作数据库)_discuz 发帖api-CSDN博客 简单改造了一下,适配我自己的需求 有一个站点存在多个采集站,我想通过主站拿标题,采集站拿内容 使用到的sql如下 CREATE TABLE pre_forum_post_…...
boost::filesystem::path文件路径使用详解和示例
boost::filesystem::path 是 Boost 库中用于跨平台操作文件路径的类,封装了路径的拼接、分割、提取、判断等常用功能。下面是对它的使用详解,包括常用接口与完整示例。 1. 引入头文件与命名空间 #include <boost/filesystem.hpp> namespace fs b…...

【Vue】scoped+组件通信+props校验
【scoped作用及原理】 【作用】 默认写在组件中style的样式会全局生效, 因此很容易造成多个组件之间的样式冲突问题 故而可以给组件加上scoped 属性, 令样式只作用于当前组件的标签 作用:防止不同vue组件样式污染 【原理】 给组件加上scoped 属性后…...