Fork me on GitHub
go-tour源码阅读

go-tour源码阅读

2012-12-13 15:38 by 轩脉刃, 87 阅读, 0 评论, 收藏编辑

go-tour是一个web项目,具有很强的可读性。

下面先看下里面的目录

appengine:这个包是当你想把go-tour布置到GAE上的时候使用的安装,我们这里只是使用本地安装,所以这里的代码可以不看

solutions:这个文件夹可以忽略,只是对文档中execise的答案,与go-tour网站无关

talks:这个是作者做的关于go-tour的演讲放在这里面,也可以直接忽略

static:这个文件夹是存放静态资源,js,css,html都是放在这里面

tree:这个文件夹存放程序自动生成一个树的代码包。这个是具体的功能才使用到的,与网站无关

wc:测试套件。这个是具体的练习题中才使用的到,与网站无关

pic:对图片的处理包。这个是具体的练习题中才使用到的,与网站无关

gotour:这个才是真正的go-tour的入口,main包在这里

 

从gotour/local.go看起

直接在全局变量中写了两个参数http和html。

http说明是在哪个端口监听。

html是说明是输出html页面还是打开,这个html是没有用的,估计作者想做这个功能来着,还没做完。

init函数是将编译生成的可执行文件存放的临时文件夹设置好。

进入main函数,有个uniq的chan int,这个是为了给临时文件做唯一标示的,会不断增长的。

main中的uniq是一个id生成器,这个方法很常用

 

var (
     // a source of numbers, for naming temporary files
     uniq = make(chan int)
)
 
func main() {
     flag.Parse()
     // source of unique numbers
     go func() {
          for i := 0; ; i++ {
               uniq <- i
          }
     }()
}

main中首先获取go-tour的目录,它是使用build包来获取,可以学之当做固定模板

// find and serve the go tour files
p, err := build.Default.Import(basePkg, "", build.FindOnly)
if err != nil {
    log.Fatalf("Couldn't find tour files: %v", err)
}
root := p.Dir

 

 

下面就是使用http包做不同的路由了。

有几个不同的路由器:

/ ,  /favicon.ico, /static/ , talk/,  kill

 

kill是杀死一个running的命令。这里有个running是为了跑正在运行的程序

run函数很好的演示了如何运行系统命令,并且输出结果

好了,这里的main看完了,当你实际运行的时候你会发现,程序中对代码的编译实际上是使用/compile和/fmt两种。这两种访问路径是在goplay.go和fmt.go中定义的

 

先看goplay.go

直接在init()中写上了HandleFunc,这样也能更醒目地告诉所有人,这里的包是为了编译使用

Compile函数中的具体实现步骤是用compile函数(local.go中)来实现的,后面就是json编码后输出。

下面看main中的compile,main中有维持一个running结构,这里值得注意的是它使用的初始化直接在struct定义的时候加个var,这个可以记一下

var running struct {

     sync.Mutex

     cmd *exec.Cmd

}

 

继续看compile,这个时候,参数request里面是包含要运行的go代码

先将go代码放到临时文件中,然后bin是目标生成文件,src是源码文件

第一步,将代码写入到src中

第二步,运行go build,并使用-o 参数,将目标文件生成到bin中

第三步,运行bin,输出结果(当然这个结果是有做一些处理,之类的commentRe的作用)

第四步,删除src和bin这两个文件

 

下面就看fmt.go

对代码进行格式化应该也可以使用go fmt工具,但是这里不是这样使用的。

同goplay.go一样,在init的时候设置了一个Handle

然后使用gofmt函数对代码格式化,进到gofmt函数

gofmt函数使用了go/token,go/parser对代码进行格式化,实际的工作就是:

1 顺序化import包

2 美化go代码

 

哈哈,这里就是说go的程序能对自身(go程序)进行美化和运行(自举),这个感觉太爽了,使用的就是go/XXX的包

这里的几个关键函数:

token.NewFileSet() : 创建fset结构

parser.ParseFile() : 解析go编码(这步最后的mode能制定是解析到什么地步)

ast.SortImports() : 将import包进行排序

printer.Fprint() : 美化工具

这下代码就结束了,至于页面上出现的分页的效果等,都是js和css作用的结果,与服务器无关了。

整个页面只需要3个代码文件(静态页面除外)!300行代码不到的量!(当然它更多的操作是在js中)少即是极多!

Creative Commons License

本文基于署名-非商业性使用 3.0许可协议发布,欢迎转载,演绎,但是必须保留本文的署名叶剑峰(包含链接http://www.cnblogs.com/yjf512/),且不得用于商业目的。如您有任何疑问或者授权方面的协商,请与我联系

分类: Go
posted on 2012-12-13 17:54  HackerVirus  阅读(192)  评论(0编辑  收藏  举报