这是我参与「第五届青训营 」伴学笔记创作活动的第 1 天
课上小项目
猜谜游戏
学到的知识点:
1.使用 math/rand 包里面的rand时,为了避免每次运行时产生固定随机数,需要提前指定随机数种子,而这个指定的随机数种子可以用系统运行的时间戳(他在不断的变化,几乎不会相同)。
2.bufio利用了一个缓冲区,使得每次读取文件内容时会首先从缓冲区读取,提高了读取速度,也避免了频繁文件io,这样可以在大多数时候降低读取方法的执行时间。
3.对于go的输入可参考该文章:blog.csdn.net/qq_52698632…
课上作业:
win10平台在按回车之后,会在参数的末尾加上\r\n作为换行符,而旧的代码只处理了\n的这种情况,没有考虑\r\n这种情况。
package main
import (
"fmt"
"math/rand"
"time"
)
func main() {
maxNum := 100
rand.Seed(time.Now().UnixNano())
secretNumber := rand.Intn(maxNum)
fmt.Println("Please input your guess")
for {
var guess int
_, err := fmt.Scanf("%d\n", &guess)
if err != nil {
fmt.Println("Invalid input. Please enter an integer value")
continue
}
fmt.Println("You guess is", guess)
if guess > secretNumber {
fmt.Println("Your guess is bigger than the secret number. Please try again")
} else if guess < secretNumber {
fmt.Println("Your guess is smaller than the secret number. Please try again")
} else {
fmt.Println("Correct, you Legend!")
break
}
}
}
在线查单词
知识点:
1.在对应的翻译网站f12,找到请求接受翻译信息的文件右键,以cURL(bash)的格式复制,然后在curlconverter网站进行代码生成go代码粘贴,如此便可返回翻译信息的json。
2.将返回的json在go中需要构造结构体来格式化数据,复制json在oktools里进行go的结构体化,然后从结构体中取出需要的数据输出。
课上作业:
作业一:用有道翻译替换了之前的彩云翻译引擎。
遇到的问题:彩云的请求是json类型可以直接用map存,有道的是formdata,用Endode()进行转化即可放到流里面。
作业二:用两个翻译引擎并行,提高响应,这里使用信道(channel)来进行阻塞主线程,然后在两个goroutine的结尾ch <- 0,这样当有一个信道有数据进来的时候主线程就会结束,这样就达到了逻辑上的并行。
package main
import (
"bytes"
"encoding/json"
"fmt"
"io/ioutil"
"log"
"net/http"
"net/url"
"os"
"strings"
)
var ch chan int = make(chan int)
// 有道
type Dict struct {
//有道json抓取地址:https://dict.youdao.com/
}
type DictRequest struct {
TransType string `json:"trans_type"`
Source string `json:"source"`
UserID string `json:"user_id"`
}
// 彩云
type DictResponse struct {
//彩云json抓取地址:https://fanyi.caiyunapp.com/#/
}
func query_caiyun(word string) {
client := &http.Client{}
request := DictRequest{TransType: "en2zh", Source: word}
buf, err := json.Marshal(request)
if err != nil {
log.Fatal(err)
}
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()
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 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)
}
ch <- 0 // 结束阻塞
}
func query_youdao(word string) {
client := &http.Client{}
web1 := "https://dict.youdao.com/result?"
web2 := "=network&lang=en"
datas := url.Values{"q": {word}, "le": {"en"}, "t": {"4"}, "client": {"web"}, "sign": {"7abc6797e761d7855d1e08feb525e0be"}, "keyfrom": {"webdict"}}
var data = strings.NewReader(datas.Encode()) // 用Encode函数可以转成formdata写入流里面
req, err := http.NewRequest("POST", "https://dict.youdao.com/jsonapi_s?doctype=json&jsonversion=4", data) // 在把流装到req里面
if err != nil {
log.Fatal(err)
}
req.Header.Set("Accept", "application/json, text/plain, */*")
req.Header.Set("Accept-Language", "zh-CN,zh;q=0.9")
req.Header.Set("Connection", "keep-alive")
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
req.Header.Set("Cookie", "OUTFOX_SEARCH_USER_ID=-1948840662@10.108.162.138; OUTFOX_SEARCH_USER_ID_NCOO=1018562459.6805735; __yadk_uid=3t8BES7fv7c7edJARw3BEViGgNhFuz4s; rollNum=true; advertiseCookie=advertiseValue; ___rl__test__cookies=1674032172243")
req.Header.Set("Origin", "https://dict.youdao.com")
req.Header.Set("Referer", web1+word+web2)
req.Header.Set("Sec-Fetch-Dest", "empty")
req.Header.Set("Sec-Fetch-Mode", "cors")
req.Header.Set("Sec-Fetch-Site", "same-origin")
req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36")
req.Header.Set("sec-ch-ua", `"Not_A Brand";v="99", "Google Chrome";v="109", "Chromium";v="109"`)
req.Header.Set("sec-ch-ua-mobile", "?0")
req.Header.Set("sec-ch-ua-platform", `"Windows"`)
resp, err := client.Do(req)
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
bodyText, err := ioutil.ReadAll(resp.Body)
if err != nil {
log.Fatal(err)
}
fmt.Printf("%s\n", bodyText)
var dict Dict
err = json.Unmarshal(bodyText, &dict) // 反序列化
fmt.Println("英 [", dict.Ec.Word.Ukphone, "] 美[", dict.Ec.Word.Usphone, "]")
for _, item := range dict.Ec.Word.Trs {
fmt.Println(item.Pos, " ", item.Tran)
}
ch <- 0 // 结束阻塞
}
func main() {
var word string
fmt.Println("请输入单词:")
_, err := fmt.Scanln(&word)
if err != nil {
fmt.Println("没有输入单词")
os.Exit(1)
}
go query_caiyun(word) // 启动第一个协程
go query_youdao(word) // 启动第二个协程
<-ch // 阻塞主函数
}