GO语言圣经 第五章习题

练习5.1

修改findlinks代码中遍历n.FirstChild链表的部分,将循环调用visit,改成递归调用。

go 复制代码
func visit(links []string, n *html.Node) []string {
	if n == nil {
		return links
	}
	if n.Type == html.ElementNode && n.Data == "a" {
		for _, a := range n.Attr {
			if a.Key == "href" {
				links = append(links, a.Val)
			}
		}
	}
	links = visit(links, n.NextSibling)
	links = visit(links, n.FirstChild)
	return links
}

练习5.2

编写函数,记录在HTML树中出现的同名元素的次数。

go 复制代码
package main

import (
	"fmt"
	"os"

	"golang.org/x/net/html"
)

type NodeCount map[string]int

func main() {
	doc, err := html.Parse(os.Stdin)
	if err != nil {
		fmt.Fprintf(os.Stderr, "findlinks1: %v\n", err)
		os.Exit(1)
	}
	nodeCount := NodeCount{}
	fill(&nodeCount, doc)
	fmt.Printf("%v", nodeCount)
}

func fill(nc *NodeCount, cn *html.Node) {
	if cn.Type == html.ElementNode {
		(*nc)[cn.Data]++
	}
	for next := cn.FirstChild; next != nil; next = next.NextSibling {
		fill(nc, next)
	}
}

练习5.3

编写函数输出所有text结点的内容。注意不要访问<script>和<style>元素,因为这些元素对浏览者是不可见的。

go 复制代码
func getText(texts []string, n *html.Node) []string {
	if n.Type == html.TextNode {
		texts = append(texts, n.Data)
	}
	for c := n.FirstChild; c != nil; c = c.NextSibling {
		if c.Data == "script" || c.Data == "style" {
			continue
		}
		texts = getText(texts, c)
	}
	return texts
}

练习5.4

扩展visit函数,使其能够处理其他类型的结点,如images、scripts和style sheets。

go 复制代码
func visit(links []string, n *html.Node) []string {
	if n.Type == html.ElementNode && (n.Data == "a" || n.Data == "img" || n.Data == "link" || n.Data == "scripts") {
		for _, a := range n.Attr {
			if a.Key == "href" {
				// fmt.Println(n.Data)
				links = append(links, a.Val)
			}
		}
	}
	for c := n.FirstChild; c != nil; c = c.NextSibling {
		links = visit(links, c)
	}
	return links
}

练习5.5

实现countWordsAndImages。(参考练习4.9如何分词)

go 复制代码
func countWordsAndImages(n *html.Node) (words, images int) {

	texts, images := visit(nil, 0, n)

	for _, v := range texts {
		words += len(strings.Split(v, " "))
		v = strings.Trim(strings.TrimSpace(v), "\r\n")
		if v == "" {
			continue
		}
		fmt.Println(strings.Split(v, " "))
		words += len(strings.Split(v, " "))
		fmt.Println(len(strings.Split(v, " ")))
	}
	return
}

func visit(texts []string, imgs int, n *html.Node) ([]string, int) {
	if n.Type == html.TextNode {
		texts = append(texts, n.Data)
	}
	if n.Type == html.ElementNode && (n.Data == "img") {
		imgs++
	}
	for c := n.FirstChild; c != nil; c = c.NextSibling {
		if c.Data == "script" || c.Data == "style" {
			continue
		}
		texts, imgs = visit(texts, imgs, c)
	}
	return texts, imgs
}

练习5.6

修改gopl.io/ch3/surface(§3.2)中的corner函数,将返回值命名,并使用bare return。

go 复制代码
func corner(i, j int) (sx float64, sy float64) {
    // Find point (x,y) at corner of cell (i,j).
    x := xyrange * (float64(i)/cells - 0.5)
    y := xyrange * (float64(j)/cells - 0.5)

    // Compute surface height z.
    z := f(x, y)

    // Project (x,y,z) isometrically onto 2-D SVG canvas (sx,sy).
    sx := width/2 + (x-y)*cos30*xyscale
    sy := height/2 + (x+y)*sin30*xyscale - z*zscale
    return
}
相关推荐
Tony Bai2 小时前
释放 Go 的极限潜能:CPU 缓存友好的数据结构设计指南
开发语言·后端·缓存·golang
h7997103 小时前
go资深之路笔记(九)kafka浅析
笔记·golang·kafka
Yeats_Liao3 小时前
Go Web 编程快速入门 02 - 认识 net/http 与 Handler 接口
前端·http·golang
千码君20163 小时前
Go语言:关于导包的两个重要说明
开发语言·后端·golang·package·导包
Tony Bai7 小时前
【Go 网络编程全解】06 UDP 数据报编程:速度、不可靠与应用层弥补
开发语言·网络·后端·golang·udp
Dobby_057 小时前
【Go】C++ 转 Go 第(一)天:环境搭建 Windows + VSCode 远程连接 Linux
linux·运维·c++·vscode·golang
JCGKS7 小时前
Go|sync.Pool|临时对象池,实现临时对象的复用,降低GC压力
golang·对象池·对象复用·降低gc压力
星星点点洲15 小时前
PostgreSQL 15二进制文件
开发语言·设计模式·golang
youliroam21 小时前
成语接龙学习
学习·golang·uniapp·成语接龙
驰羽1 天前
[GO]什么是热重载,如何使用Air工具
开发语言·后端·golang