当前位置: 首页 > 工具软件 > rust-jsonnet > 使用案例 >

golang web学习随便记7-XML、JSON、Web服务

单于楚
2023-12-01

xml的解析比json复杂,最起码xml在元素的属性中也能蕴藏信息。下面是一个xml文件post1.xml:

<?xml version="1.0" encoding="utf-8" ?>
<post id="1">
    <content>你好,Golang</content>
    <author id="2">张三</author>
</post>

表面上看,上述xml就是外层元素post里面有两个子元素content和author,但post和author都是带有id属性的,其实这意味着author有2项信息,不能用一个字段表达,所以,下面的代码为author定义了结构体。

package main

import (
	"encoding/xml"
	"fmt"
	"io/ioutil"
	"os"
)

type Post struct {
	XMLName xml.Name `xml:"post"`      // 元素名
	Id      string   `xml:"id,attr"`   // 元素的属性,模式标志 attr
	Content string   `xml:"content"`   // “常规”子元素
	Author  Author   `xml:"author"`    // 复合型子元素
	Xml     string   `xml:",innerxml"` // 元素内部的原始xml,模式标志 innerxml
}

type Author struct {
	Id   string `xml:"id,attr"`
	Name string `xml:",chardata"` // 元素的字符数据,模式标志 chardata
}

func main() {
	xmlFile, err := os.Open("post1.xml")
	if err != nil {
		fmt.Println("打开xml文件错误:", err)
		return
	}
	defer xmlFile.Close()
	xmlData, err := ioutil.ReadAll(xmlFile)
	if err != nil {
		fmt.Println("读取xml数据错误:", err)
		return
	}
	var post Post
	xml.Unmarshal(xmlData, &post)
	fmt.Println(post)
}

运行后输出结构为:

sjg@sjg-PC:~/go/src/xml$ go run .
{{ post} 1 你好,Golang {2 张三} 
    <content>你好,Golang</content>
    <author id="2">张三</author>
}

我们修改一下post1.xml为以下post2.xml,设想返回该帖子时同时返回了对应的2条评论:

<?xml version="1.0" encoding="utf-8" ?>
<post id="1">
    <content>你好,Golang</content>
    <author id="2">张三</author>
    <comments>
        <comment id="1">
            <content>C++才是好语言</content>
            <author id="3">李四</author>
        </comment>
        <comment id="2">
            <content>Rust比C++好</content>
            <author id="4">王五</author>
        </comment>        
    </comments>
</post>

在修改解析代码前,我们先需要考虑:是否需要为comments元素定义一个结构体,然后再为comment元素定义一个结构体,前者是后者的列表?更广泛的,我们是否必须定义上级元素才能获取到下一级元素?答案是否定的。我们可以使用`xml:"a>b>c"`这样的结构标签实现不指定树状结构的情况下直接获取指定的XML元素(即跳过中间元素a和b,直接获取c)。代码添加或修改如下:

// ....................................................
type Post struct {
    // ...................................
	Xml      string    `xml:",innerxml"` // 元素内部的原始xml,模式标志 innerxml
	Comments []Comment `xml:"comments>comment"`
}

type Author struct {
    // ...................................
}

type Comment struct {
	Id      string `xml:"id,attr"`
	Content string `xml:"content"`
	Author  Author `xml:"author"`
}

func main() {
	xmlFile, err := os.Open("post2.xml")
	// ......................................
}

注意上面对列表的处理。结果输出为:

sjg@sjg-PC:~/go/src/xml$ go run .
{{ post} 1 你好,Golang {2 张三} 
    <content>你好,Golang</content>
    <author id="2">张三</author>
    <comments>
        <comment id="1">
            <content>C++才是好语言</content>
            <author id="3">李四</author>
        </comment>
        <comment id="2">
            <content>Rust比C++好</content>
            <author id="4">王五</author>
        </comment>        
    </comments>
 [{1 C++才是好语言 {3 李四}} {2 Rust比C++好 {4 王五}}]}

unmarshal这种方式可以一次性获取所有信息,但这种方式适合体积不太大的xml。对于大个头xml,我们需要使用 Decoder 来手动解码:

//  ..........................
	defer xmlFile.Close()

	decoder := xml.NewDecoder(xmlFile)
	for {
		t, err := decoder.Token()
		if err == io.EOF {
			break
		}
		if err != nil {
			fmt.Println("解码 xml 到令牌出错:", err)
			return
		}
		switch se := t.(type) {        // 判断令牌类型
		case xml.StartElement:        // 是xml某个元素起始标签
			if se.Name.Local == "comment" {  // 元素名为 comment
				var comment Comment
				decoder.DecodeElement(&comment, &se)
				fmt.Println(comment)
			}
		}
	}
}
sjg@sjg-PC:~/go/src/xml$ go run .
{1 C++才是好语言 {3 李四}}
{2 Rust比C++好 {4 王五}}

要生成xml比解析要容易一些,并且也有2种方式:marshal方式或者encoder方式

// .............结构体与前面的代码相同....................

func main() {
	post := Post{
		Id:      "1",
		Content: "你好,Golang",
		Author: Author{
			Id:   "2",
			Name: "张三",
		},
		Comments: []Comment{
			{
				Id:      "1",
				Content: "C++才是好语言",
				Author: Author{
					Id:   "3",
					Name: "李四",
				},
			},
			{
				Id:      "2",
				Content: "Rust比C++好",
				Author: Author{
					Id:   "4",
					Name: "王五",
				},
			},
		},
	}
	output, err := xml.MarshalIndent(&post, "", "    ") // 用 Marshal 则不缩进
	if err != nil {
		fmt.Println("列集到xml出错:", err)
		return
	}
	err = ioutil.WriteFile("post.xml", []byte(xml.Header+string(output)), 0644)
	if err != nil {
		fmt.Println("写入xml文件出错:", err)
		return
	}
}

生成的文件带了缩进,这里不再列出。下面改写的代码,是用编码器方式生成xml文件:

	post := Post{
//  ..........................................
    }
	xmlFile, err := os.Create("post2.xml")
	if err != nil {
		fmt.Println("创建xml文件出错:", err)
		return
	}
	xmlFile.WriteString(xml.Header)
	encoder := xml.NewEncoder(xmlFile)
	encoder.Indent("", "\t")
	err = encoder.Encode(&post)
	if err != nil {
		fmt.Println("编码xml到文件出错:", err)
		return
	}

对于JSON,我们先来看如何创建。同样,我们可以用marshal方式或者encoder方式。下面是marshal方式生成JSON,不过我们先需要适当改造前述代码中的结构体,使它们符合JSON标签的格式(JSON标签比xml简单些,也不需要头部声明,本身结构体赋值还和JSON数据很像):

package main

import (
	"encoding/json"
	"fmt"
	"io/ioutil"
)

type Post struct {
	Id       int       `json:"id"`
	Content  string    `json:"content"`
	Author   Author    `json:"author"`
	Comments []Comment `json:"comments"`
}

type Author struct {
	Id   int    `json:"id"`
	Name string `json:"name"`
}

type Comment struct {
	Id      int    `json:"id"`
	Content string `json:"content"`
	Author  Author `json:"author"`
}

func main() {
	post := Post{
		Id:      1,
		Content: "你好,Golang",
		Author: Author{
			Id:   2,
			Name: "张三",
		},
		Comments: []Comment{
			{
				Id:      1,
				Content: "C++才是好语言",
				Author: Author{
					Id:   3,
					Name: "李四",
				},
			},
			{
				Id:      2,
				Content: "Rust比C++好",
				Author: Author{
					Id:   4,
					Name: "王五",
				},
			},
		},
	}
	output, err := json.MarshalIndent(&post, "", "\t")
	if err != nil {
		fmt.Println("列集到JSON时出错:", err)
		return
	}
	err = ioutil.WriteFile("post.json", output, 0644)
	if err != nil {
		fmt.Println("写入JSON到文件出错:", err)
		return
	}
}

使用编码方式生成JSON的代码:

	jsonFile, err := os.Create("post.json")
	if err != nil {
		fmt.Println("创建JSON文件出错:", err)
		return
	}
	encoder := json.NewEncoder(jsonFile)
	encoder.SetIndent("", "\t")
	err = encoder.Encode(&post)
	if err != nil {
		fmt.Println("编码JSON到文件出错:", err)
		return
	}

解析JSON也是两种方式:unmarshal或decoder。一般地Body,对于流式输入(如http.Request的),用解码方式较好;对于字符串或者内存的JSON数据,用unmarshal更好。

// .................................................
func main() {
	jsonFile, err := os.Open("post.json")
	if err != nil {
		fmt.Println("打开JSON文件出错:", err)
		return
	}
	defer jsonFile.Close()
	jsonData, err := ioutil.ReadAll(jsonFile)
	if err != nil {
		fmt.Println("读取JSON数据出错:", err)
		return
	}

	var post Post
	json.Unmarshal(jsonData, &post)
	fmt.Println(post)
}
// ..............................................
	defer jsonFile.Close()
	decoder := json.NewDecoder(jsonFile)
	for {
		var post Post
		err := decoder.Decode(&post)
		if err == io.EOF {
			break
		}
		if err != nil {
			fmt.Println("解码JSON出错:", err)
			return
		}
		fmt.Println(post)
	}
}

下面我们仅就posts表来编写web服务:需要准备好mariadb 10.3数据库,另外,基本的表操作代码基本上是拷贝 golang web学习随便记4-内存、文件、数据库_sjg20010414的博客-CSDN博客 一文中的,只是 Post 结构体添加JSON标签。下面是main包中的data.go文件:

package main

import (
	"database/sql"
	"log"

	_ "github.com/go-sql-driver/mysql"
)

type Post struct {
	Id      int    `json:"id"`
	Content string `json:"content"`
	Author  string `json:"author"`
}

var Db *sql.DB

func init() { // 此 init 不显式调用,自动隐式调用,实现初始化全局变量 Db
	var err error
	Db, err = sql.Open("mysql", "gwp:dbpassword@tcp(172.17.0.1:3306)/gwp?charset=utf8mb4,utf8")
	if err != nil {
		panic(err)
	}
}

func Posts(limit int) (posts []Post, err error) {
	rows, err := Db.Query("SELECT id, content, author FROM posts LIMIT ?", limit) // Query(..) 预期返回多行结果集
	if err != nil {
		return
	}
	for rows.Next() { // 用循环遍历多行结果集
		post := Post{}
		err = rows.Scan(&post.Id, &post.Content, &post.Author) // Scan(..) 将结果集当前列值绑定到变量
		if err != nil {
			return
		}
		posts = append(posts, post) // 结果依次放入切片 posts
	}
	rows.Close() // 关闭结果集,清理内存
	return
}

func GetPost(id int) (post Post, err error) {
	post = Post{}
	err = Db.QueryRow("SELECT id, content, author FROM posts WHERE id = ?", id). // QueryRow(..) 预期返回单行结果集
											Scan(&post.Id, &post.Content, &post.Author) // pgsql 可以 RETURNING id
	return
}

func (post *Post) Create() (err error) {
	sql := "INSERT INTO posts (content, author) VALUES (?, ?)"
	stmt, err := Db.Prepare(sql) // 对于插入使用准备语句
	if err != nil {
		log.Fatal(err)
		return
	}
	defer stmt.Close()

	// err = stmt.QueryRow(post.Content, post.Author).Scan(&post.Id)  // pgsql 可以一步设置 post.id
	if result, err := stmt.Exec(post.Content, post.Author); err != nil { // Exec(..)返回执行结果
		log.Fatal(err)
	} else {
		if last_insert_id, err := result.LastInsertId(); err != nil {
			log.Fatal(err)
		} else {
			post.Id = int(last_insert_id) // Mariadb/MySQL应该支持执行结果的 LastInsertId()
		}
	}
	return
}

func (post *Post) Update() (err error) {
	_, err = Db.Exec("UPDATE posts SET content = ?, author = ? WHERE id = ?",
		post.Content, post.Author, post.Id)
	return
}

func (post *Post) Delete() (err error) {
	_, err = Db.Exec("DELETE FROM posts WHERE id = ?", post.Id)
	return
}

下面是main包中的main.go文件:

package main

import (
	"encoding/json"
	"net/http"
	"path"
	"strconv"
)

func main() {
	server := http.Server{
		Addr: "127.0.0.1:8088",
	}
	http.HandleFunc("/post/", handleRequest)
	http.HandleFunc("/posts/", handleList)
	server.ListenAndServe()
}

func handleRequest(w http.ResponseWriter, r *http.Request) {
	var err error
	switch r.Method {
	case "GET":
		err = handleGet(w, r)
	case "POST":
		err = handlePost(w, r)
	case "PUT":
		err = handlePut(w, r)
	case "DELETE":
		err = handleDelete(w, r)
	}
	if err != nil {
		http.Error(w, err.Error(), http.StatusInternalServerError)
		return
	}
}

func handleGet(w http.ResponseWriter, r *http.Request) (err error) {
	id, err := strconv.Atoi(path.Base(r.URL.Path)) // 路径中最后一个元素转换成整数
	if err != nil {
		return
	}
	post, err := GetPost(id)
	if err != nil {
		return
	}
	output, err := json.MarshalIndent(&post, "", "\t")
	if err != nil {
		return
	}
	w.Header().Set("Content-Type", "application/json")
	w.Write(output)
	return
}

func handlePost(w http.ResponseWriter, r *http.Request) (err error) {
	len := r.ContentLength
	body := make([]byte, len)
	r.Body.Read(body)
	var post Post
	json.Unmarshal(body, &post)
	err = post.Create()
	if err != nil {
		return
	}
	w.WriteHeader(200)
	return
}

func handlePut(w http.ResponseWriter, r *http.Request) (err error) {
	id, err := strconv.Atoi(path.Base(r.URL.Path)) // 路径中最后一个元素转换成整数
	if err != nil {
		return
	}
	post, err := GetPost(id)
	if err != nil {
		return
	}
	len := r.ContentLength
	body := make([]byte, len)
	r.Body.Read(body)
	json.Unmarshal(body, &post)
	err = post.Update()
	if err != nil {
		return
	}
	w.WriteHeader(200)
	return
}

func handleDelete(w http.ResponseWriter, r *http.Request) (err error) {
	id, err := strconv.Atoi(path.Base(r.URL.Path)) // 路径中最后一个元素转换成整数
	if err != nil {
		return
	}
	post, err := GetPost(id)
	if err != nil {
		return
	}
	err = post.Delete()
	if err != nil {
		return
	}
	w.WriteHeader(200)
	return
}

func handleList(w http.ResponseWriter, r *http.Request) {
	posts, err := Posts(10)
	if err != nil {
		return
	}
	output, err := json.MarshalIndent(&posts, "", "\t")
	if err != nil {
		return
	}
	w.Header().Set("Content-Type", "application/json")
	w.Write(output)
	return
}

从 docker 启动 mariadb数据库,go get github.com/go-sql-driver/mysql 添加依赖包,然后 go run .运行服务。我们用 curl 来测试各项服务:

sjg@sjg-PC:~$ curl -i -X POST -H "Content-Type: application/json" -d '{"content":"我的第一条帖子","author":"赵六"}' http://127.0.0.1:8088/post/
HTTP/1.1 200 OK
Date: Thu, 18 May 2023 08:50:42 GMT
Content-Length: 0
sjg@sjg-PC:~$ curl -i -X GET http://127.0.0.1:8088/posts/
HTTP/1.1 200 OK
Content-Type: application/json
Date: Thu, 18 May 2023 08:52:35 GMT
Content-Length: 316

[
        {
                "id": 1,
                "content": "你好, C++! 08:33:04",
                "author": "李四"
        },
        {
                "id": 2,
                "content": "你好, C++! 08:46:48",
                "author": "李四"
        },
        {
                "id": 3,
                "content": "你好, C++! 08:51:08",
                "author": "李四"
        },
        {
                "id": 4,
                "content": "我的第一条帖子",
                "author": "赵六"
        }
]
sjg@sjg-PC:~$ curl -i -X GET http://127.0.0.1:8088/post/4
HTTP/1.1 200 OK
Content-Type: application/json
Date: Thu, 18 May 2023 08:53:28 GMT
Content-Length: 70

{
        "id": 4,
        "content": "我的第一条帖子",
        "author": "赵六"
}
sjg@sjg-PC:~$  curl -i -X PUT -H "Content-Type: application/json" -d '{"content":"我的第一条帖子(修改版)","author":"赵六"}' http://127.0.0.1:8088/post/4
HTTP/1.1 200 OK
Date: Thu, 18 May 2023 08:55:08 GMT
Content-Length: 0
sjg@sjg-PC:~$ curl -i -X DELETE http://127.0.0.1:8088/post/3
HTTP/1.1 200 OK
Date: Thu, 18 May 2023 08:56:18 GMT
Content-Length: 0

最后的结果是:

sjg@sjg-PC:~$ curl -i -X GET http://127.0.0.1:8088/posts/HTTP/1.1 200 OK
Content-Type: application/json
Date: Thu, 18 May 2023 08:56:59 GMT
Content-Length: 252

[
        {
                "id": 1,
                "content": "你好, C++! 08:33:04",
                "author": "李四"
        },
        {
                "id": 2,
                "content": "你好, C++! 08:46:48",
                "author": "李四"
        },
        {
                "id": 4,
                "content": "我的第一条帖子(修改版)",
                "author": "赵六"
        }
]

 类似资料: