欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Go每日一庫之quicktemplate的使用

 更新時間:2021年07月05日 15:51:22   作者:darjun  
quicktemplate快速、功能強大、易于使用的Go模板引擎。比html/模板快20倍,本文我們就詳細的介紹一下quicktemplate的具體使用,感興趣的可以了解一下

簡介

最近在整理我們項目代碼的時候,發(fā)現(xiàn)有很多活動的代碼在結(jié)構(gòu)和提供的功能上都非常相似。為了方便今后的開發(fā),我花了一點時間編寫了一個生成代碼框架的工具,最大程度地降低重復(fù)勞動。代碼本身并不復(fù)雜,且與項目代碼關(guān)聯(lián)性較大,這里就不展開介紹了。在這個過程中,我發(fā)現(xiàn) Go 標(biāo)準(zhǔn)的模板庫text/template和html/template使用起來比較束手束腳,很不方便。我從 GitHub 了解到quicktemplate這個第三方模板庫,功能強大,語法簡單,使用方便。今天我們就來介紹一下quicktemplate。

快速使用

本文代碼使用 Go Modules。

先創(chuàng)建代碼目錄并初始化:

$ mkdir quicktemplate && cd quicktemplate
$ go mod init github.com/darjun/go-daily-lib/quicktemplate

quicktemplate會將我們編寫的模板代碼轉(zhuǎn)換為 Go 語言代碼。因此我們需要安裝quicktemplate包和一個名為qtc的編譯器:

$ go get -u github.com/valyala/quicktemplate
$ go get -u github.com/valyala/quicktemplate/qtc

首先,我們需要編寫quicktemplate格式的模板文件,模板文件默認以.qtpl作為擴展名。下面我編寫了一個簡單的模板文件greeting.qtpl:

All text outside function is treated as comments.

{% func Greeting(name string, count int) %}
  {% for i := 0; i < count; i++ %}
    Hello, {%s name %}
  {% endfor %}
{% endfunc %}

模板語法非常簡單,我們只需要簡單了解以下 2 點:

  • 模板以函數(shù)為單位,函數(shù)可以接受任意類型和數(shù)量的參數(shù),這些參數(shù)可以在函數(shù)中使用。所有函數(shù)外的文本都是注釋,qtc編譯時會忽視注釋;
  • 函數(shù)內(nèi)的內(nèi)容,除了語法結(jié)構(gòu),其他都會原樣輸出到渲染后的文本中,包括空格和換行。

將greeting.qtpl保存到templates目錄,然后執(zhí)行qtc命令。該命令會生成對應(yīng)的 Go 文件greeting.qtpl.go,包名為templates。現(xiàn)在,我們就可以使用這個模板了:

package main

import (
  "fmt"

  "github.com/darjun/go-daily-lib/quicktemplate/get-started/templates"
)

func main() {
  fmt.Println(templates.Greeting("dj", 5))
}

調(diào)用模板函數(shù),傳入?yún)?shù),返回渲染后的文本:

$ go run .
    Hello, dj

    Hello, dj

    Hello, dj

    Hello, dj

    Hello, dj

{%s name %}執(zhí)行文本替換,{% for %}循環(huán)生成重復(fù)文本。輸出中出現(xiàn)多個空格和換行,這是因為函數(shù)內(nèi)除了語法結(jié)構(gòu),其他內(nèi)容都會原樣保留,包括空格和換行。

需要注意的是,由于quicktemplate是將模板轉(zhuǎn)換為 Go 代碼使用的,所以如果模板有修改,必須先執(zhí)行qtc命令重新生成 Go 代碼,否則修改不生效。

語法結(jié)構(gòu)

quicktemplate支持 Go 常見的語法結(jié)構(gòu),if/for/func/import/return。而且寫法與直接寫 Go 代碼沒太大的區(qū)別,幾乎沒有學(xué)習(xí)成本。只是在模板中使用這些語法時,需要使用{%和%}包裹起來,而且if和for等需要添加endif/endfor明確表示結(jié)束。

變量

上面我們已經(jīng)看到如何渲染傳入的參數(shù)name,使用{%s name %}。由于name是 string 類型,所以在{%后使用s指定類型。quicktemplate還支持其他類型的值:

  • 整型:{%d int %},{%dl int64 %},{%dul uint64 %};
  • 浮點數(shù):{%f float %}。還可以設(shè)置輸出的精度,使用{%f.precision float %}。例如{%f.2 1.2345 %}輸出1.23;
  • 字節(jié)切片([]byte):{%z bytes %};
  • 字符串:{%q str %}或字節(jié)切片:{%qz bytes %},引號轉(zhuǎn)義為&quot;;
  • 字符串:{%j str %}或字節(jié)切片:{%jz bytes %},沒有引號;
  • URL 編碼:{%u str %},{%uz bytes %};
  • {%v anything %}:輸出等同于fmt.Sprintf("%v", anything)。

先編寫模板:

{% func Types(a int, b float64, c []byte, d string) %}
  int: {%d a %}, float64: {%f.2 b %}, bytes: {%z c %}, string with quotes: {%q d %}, string without quotes: {%j d %}.
{% endfunc %}

然后使用:

func main() {
  fmt.Println(templates.Types(1, 5.75, []byte{'a', 'b', 'c'}, "hello"))
}

運行:

$ go run .

  int: 1, float64: 5.75, bytes: abc, string with quotes: &quot;hello&quot;, string without quotes: hello.

調(diào)用函數(shù)

quicktemplate支持在模板中調(diào)用模板函數(shù)、標(biāo)準(zhǔn)庫的函數(shù)。由于qtc會直接生成 Go 代碼,我們甚至還可以在同目錄下編寫自己的函數(shù)給模板調(diào)用,模板 A 中也可以調(diào)用模板 B 中定義的函數(shù)。

我們先在templates目錄下編寫一個文件rank.go,定義一個Rank函數(shù),傳入分數(shù),返回評級:

package templates

func Rank(score int) string {
  if score >= 90 {
    return "A"
  } else if score >= 80 {
    return "B"
  } else if score >= 70 {
    return "C"
  } else if score >= 60 {
    return "D"
  } else {
    return "E"
  }
}

然后我們可以在模板中調(diào)用這個函數(shù):

{% import "fmt" %}
{% func ScoreList(name2score map[string]int) %}
  {% for name, score := range name2score %}
    {%s fmt.Sprintf("%s: score-%d rank-%s", name, score, Rank(score)) %}
  {% endfor %}
{% endfunc %}

編譯模板:

$ qtc

編寫程序:

func main() {
  name2score := make(map[string]int)
  name2score["dj"] = 85
  name2score["lizi"] = 96
  name2score["hjw"] = 52

  fmt.Println(templates.ScoreList(name2score))
}

運行程序輸出:

$ go run .

    dj: score-85 rank-B

    lizi: score-96 rank-A

    hjw: score-52 rank-E

由于我們在模板中用到fmt包,需要先使用{% import %}將該包導(dǎo)入。

在模板中調(diào)用另一個模板的函數(shù)也是類似的,因為模板最終都會轉(zhuǎn)為 Go 代碼。Go 代碼中有同樣簽名的函數(shù)。

Web

quicktemplate常用來編寫 HTML 頁面的模板:

{% func Index(name string) %}
<html>
  <head>
    <title>Awesome Web</title>
  </head>
  <body>
    <h1>Hi, {%s name %}
    <p>Welcome to the awesome web!!!</p>
  </body>
</html>
{% endfunc %}

下面編寫一個簡單的 Web 服務(wù)器:

func index(w http.ResponseWriter, r *http.Request) {
  templates.WriteIndex(w, r.FormValue("name"))
}

func main() {
  mux := http.NewServeMux()
  mux.HandleFunc("/", index)

  server := &http.Server{
    Handler: mux,
    Addr:    ":8080",
  }

  log.Fatal(server.ListenAndServe())
}

qtc會生成一個Write*的方法,它接受一個io.Writer的參數(shù)。將模板渲染的結(jié)果寫入這個io.Writer中,我們可以直接將http.ResponseWriter作為參數(shù)傳入,非常便捷。

運行:

$ qtc
$ go run .

瀏覽器輸入localhost:8080?name=dj查看結(jié)果。

總結(jié)

quicktemplate至少有下面 3 個優(yōu)勢:

  • 語法與 Go 語言非常類似,幾乎沒有學(xué)習(xí)成本;
  • 會先轉(zhuǎn)換為 Go,渲染速度非???,比標(biāo)準(zhǔn)庫html/template快 20 倍以上;
  • 為了安全考慮,會執(zhí)行一些編碼,避免受到攻擊。

從我個人的實際使用情況來看,確實很方便,很實用。感興趣的還可以去看看qtc生成的 Go 代碼。
大家如果發(fā)現(xiàn)好玩、好用的 Go 語言庫,歡迎到 Go 每日一庫 GitHub 上提交 issue😄

參考

quicktemplate GitHub:https://github.com/valyala/quicktemplate
Go 每日一庫 GitHub:https://github.com/darjun/go-daily-lib

到此這篇關(guān)于Go每日一庫之quicktemplate的使用的文章就介紹到這了,更多相關(guān)Go quicktemplate內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Golang實現(xiàn)根據(jù)某個特定字段對結(jié)構(gòu)體的順序進行排序

    Golang實現(xiàn)根據(jù)某個特定字段對結(jié)構(gòu)體的順序進行排序

    這篇文章主要為大家詳細介紹了Golang如何實現(xiàn)根據(jù)某個特定字段對結(jié)構(gòu)體的順序進行排序,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2024-03-03
  • Minio基本介紹及如何搭建Minio集群

    Minio基本介紹及如何搭建Minio集群

    MinIO主要采用Golang語言實現(xiàn),客戶端與存儲服務(wù)器之間采用http/https通信協(xié)議,本文重點給大家介紹什么是Minio?如何搭建Minio集群?感興趣的朋友一起看看吧
    2022-06-06
  • Go prometheus metrics條目自動回收與清理方法

    Go prometheus metrics條目自動回收與清理方法

    這篇文章主要為大家介紹了Go prometheus metrics條目自動回收與清理方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-11-11
  • 詳解Go 并發(fā)

    詳解Go 并發(fā)

    這篇文章主要介紹了Go 并發(fā)的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)go語言,感興趣的朋友可以了解下
    2020-09-09
  • Golang實現(xiàn)簡易的命令行功能

    Golang實現(xiàn)簡易的命令行功能

    這篇文章主要為大家詳細介紹了如何通過Golang實現(xiàn)一個簡易的命令行功能,文中的示例代碼講解詳細,具有一定的學(xué)習(xí)價值,感興趣的可以了解一下
    2023-02-02
  • 淺談golang二進制bit位的常用操作

    淺談golang二進制bit位的常用操作

    這篇文章主要介紹了淺談golang二進制bit位的常用操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • golang使用OpenTelemetry實現(xiàn)跨服務(wù)全鏈路追蹤詳解

    golang使用OpenTelemetry實現(xiàn)跨服務(wù)全鏈路追蹤詳解

    這篇文章主要為大家介紹了golang使用OpenTelemetry實現(xiàn)跨服務(wù)全鏈路追蹤詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-09-09
  • Go語言里的結(jié)構(gòu)體文法實例分析

    Go語言里的結(jié)構(gòu)體文法實例分析

    這篇文章主要介紹了Go語言里的結(jié)構(gòu)體文法,實例分析了結(jié)構(gòu)體文法的概念及使用技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-02-02
  • Go語言中使用 buffered channel 實現(xiàn)線程安全的 pool

    Go語言中使用 buffered channel 實現(xiàn)線程安全的 pool

    這篇文章主要介紹了Go語言中使用 buffered channel 實現(xiàn)線程安全的 pool,因為Go語言自帶的sync.Pool并不是很好用,所以自己實現(xiàn)了一線程安全的 pool,需要的朋友可以參考下
    2014-10-10
  • Go?json自定義Unmarshal避免判斷nil示例詳解

    Go?json自定義Unmarshal避免判斷nil示例詳解

    這篇文章主要為大家介紹了Go?json自定義Unmarshal避免判斷nil示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-06-06

最新評論