介紹涵蓋內容:
- 為載入和儲存方法建立一個資料結構體
- 引用http包來建立一個web應用
- 引用template包來處理HTML模板
- 引用regexp包來驗證使用者的輸入
- 引用 閉包操作
可能涉及到的知識:
- 設計經驗
- 明白基礎的web技術(HTTP,HTML)
- 一些UNIX命令列知識
從這裡開始你要有一個可以運行Go語言的電腦或虛擬機器,怎麼樣安裝Go,請參考安裝Go教程。首先建立一個目錄,在目錄下建立一個wiki.go檔案,用你喜歡的編輯器開啟並輸入以下內容:
package main
import (
"fmt"
"io/ioutil"
"os"
)
這fmt,ioutil和os都是go語言的標準庫,一會我將增加其他方法和更多的包。
資料結構
讓我們聲明一個資料結構,這個結構主要包含兩個欄位,一個是標題,一個是內容。
type Page struct {
Title string
Body []byte
}
接下來,我們給Page 這個結構體寫個儲存方法,代碼如下:
func (p *Page) save() os.Error {
filename := p.Title + ".txt"
return ioutil.WriteFile(filename, p.Body, 0600)
}
這個方法的簽名是:接收一個Page結構體指標,返回一個os.Error錯誤。
在一下的代碼中還是用了http包和模板包,具體內容參考具體代碼,再這裡就不詳細貼出來了。下面是模板內容,把他們放到wiki.go同一目錄下。
編輯頁面 模板eidt.html
<h1>Editing {{.Title |html}}</h1>
<form action="/save/{{.Title |html}}" method="POST">
<div><textarea name="body" rows="20" cols="80">{{printf "%s" .Body |html}}</textarea></div>
<div><input type="submit" value="Save"></div>
</form>
查看頁面模板view.html
<h1>{{.Title |html}}</h1>
<p>[<a href="/edit/{{.Title |html}}">edit</a>]</p>
<div>{{printf "%s" .Body |html}}</div>
完整代碼:wiki.go
package main
import (
"http"
"io/ioutil"
"os"
"regexp"
"template"
)
type Page struct {
Title string
Body []byte
}
func (p *Page) save() os.Error {
filename := p.Title + ".txt"
return ioutil.WriteFile(filename, p.Body, 0600)
}
func loadPage(title string) (*Page, os.Error) {
filename := title + ".txt"
body, err := ioutil.ReadFile(filename)
if err != nil {
return nil, err
}
return &Page{Title: title, Body: body}, nil
}
func viewHandler(w http.ResponseWriter, r *http.Request, title string) {
p, err := loadPage(title)
if err != nil {
http.Redirect(w, r, "/edit/"+title, http.StatusFound)
return
}
renderTemplate(w, "view", p)
}
func editHandler(w http.ResponseWriter, r *http.Request, title string) {
p, err := loadPage(title)
if err != nil {
p = &Page{Title: title}
}
renderTemplate(w, "edit", p)
}
func saveHandler(w http.ResponseWriter, r *http.Request, title string) {
body := r.FormValue("body")
p := &Page{Title: title, Body: []byte(body)}
err := p.save()
if err != nil {
http.Error(w, err.String(), http.StatusInternalServerError)
return
}
http.Redirect(w, r, "/view/"+title, http.StatusFound)
}
var templates = make(map[string]*template.Template)
func init() {
for _, tmpl := range []string{"edit", "view"} {
t := template.Must(template.ParseFile(tmpl + ".html"))
templates[tmpl] = t
}
}
func renderTemplate(w http.ResponseWriter, tmpl string, p *Page) {
err := templates[tmpl].Execute(w, p)
if err != nil {
http.Error(w, err.String(), http.StatusInternalServerError)
}
}
const lenPath = len("/view/")
var titleValidator = regexp.MustCompile("^[a-zA-Z0-9]+$")
func makeHandler(fn func(http.ResponseWriter, *http.Request, string)) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
title := r.URL.Path[lenPath:]
if !titleValidator.MatchString(title) {
http.NotFound(w, r)
return
}
fn(w, r, title)
}
}
func main() {
http.HandleFunc("/view/", makeHandler(viewHandler))
http.HandleFunc("/edit/", makeHandler(editHandler))
http.HandleFunc("/save/", makeHandler(saveHandler))
http.ListenAndServe(":8080", nil)
}
運行測試:
$ 8g wiki.go$ 8l wiki.8$ ./8.out
在地址欄輸入地址:http://localhost:8080/view/aNewPage
: