測試是自動化測試的簡稱,即編寫簡單的程序來確保程序(產(chǎn)品代碼)在該測試中針對特定輸入產(chǎn)生預(yù)期的輸出。這些測試主要分兩種:
創(chuàng)新互聯(lián)公司專注于獲嘉網(wǎng)站建設(shè)服務(wù)及定制,我們擁有豐富的企業(yè)做網(wǎng)站經(jīng)驗(yàn)。 熱誠為您提供獲嘉營銷型網(wǎng)站建設(shè),獲嘉網(wǎng)站制作、獲嘉網(wǎng)頁設(shè)計(jì)、獲嘉網(wǎng)站官網(wǎng)定制、重慶小程序開發(fā)公司服務(wù),打造獲嘉網(wǎng)絡(luò)公司原創(chuàng)品牌,更為您提供獲嘉網(wǎng)站排名全網(wǎng)營銷落地服務(wù)。
go test 子命令是 Go 語言包的測試驅(qū)動程序。在一個(gè)包目錄中,以 _test.go 結(jié)尾的文件不是 go build 命令編譯的目標(biāo),而是 go test 編譯的目標(biāo)。
在 *_test.go 的測試源碼文件中,有三種類型的函數(shù):
功能測試函數(shù),以 Test 開頭,用來檢測一些程序邏輯的正確性。
基準(zhǔn)測試函數(shù),以 Benchmark 開頭,用來測試程序的性能。
示例函數(shù),以 Example 開頭,提供一個(gè)機(jī)器檢查過的示例文檔。
每一個(gè)測試文件必須導(dǎo)入 testing 包。這些函數(shù)的函數(shù)簽名如下:
func TestName(t *testing.T) {
// ...
}
參數(shù) t 提供了匯報(bào)測試失敗和日志記錄的功能。
下面先定義一個(gè)用來測試的示例,這個(gè)示例包含一個(gè)函數(shù) IsPalindrome,用來判斷一個(gè)字符串是否是回文:
// word 包提供了文字游戲相關(guān)的工具函數(shù)
package word
// IsPalindrome 判斷一個(gè)字符串是否是回文
func IsPalindrome(s string) bool {
for i := range s {
if s[i] != s[len(s)-1-i] {
return false
}
}
return true
}
這個(gè)函數(shù)對于一個(gè)字符串是否是回文字符串前后重復(fù)測試了兩次,其實(shí)只要檢查完字符串一半的字符就可以結(jié)束了。這個(gè)在稍后測試性能的時(shí)候會做改進(jìn),這里先關(guān)注功能。
在同一個(gè)目錄中,再寫一個(gè)測試文件。假設(shè)上面的示例的文件名是 word.go,那么這個(gè)測試文件的文件名可以是 word_test.go(命名沒有強(qiáng)制要求,但是這樣的命名使得文件的意義一目了然)。文件中包含了兩個(gè)功能測試函數(shù),這兩個(gè)函數(shù)都是檢查 IsPalindrome 函數(shù)是否針對某個(gè)輸入的參數(shù)能給出正確的結(jié)果,并且用 t.Error 來報(bào)錯(cuò):
package word
import "testing"
func TestPalindrome(t *testing.T) {
if !IsPalindrome("civic") {
t.Error(`IsPalindrome("civic") = false`)
}
if !IsPalindrome("madam") {
t.Error(`IsPalindrome("madam") = false`)
}
}
func TestNonPalindrome(t *testing.T) {
if IsPalindrome("palindrome") {
t.Error(`IsPalindrome("palindrome") = true`)
}
}
這個(gè)最初版本的回文判斷函數(shù)比較簡陋,有些明顯也是回文的情況,但是無法被現(xiàn)在這個(gè)版本的函數(shù)檢測出來:
針對上面兩種回文,又寫了新的測試用例:
func TestChinesePalindrome(t *testing.T) {
input := "上海自來水來自海上"
if !IsPalindrome(input) {
t.Errorf(`IsPalindrome(%q) = false`, input)
}
}
func TestSentencePalindrome(t *testing.T) {
input := "Madam, I'm Adam"
if !IsPalindrome(input) {
t.Errorf(`IsPalindrome(%q) = false`, input)
}
}
這里用了 Errorf 函數(shù),具有格式化的功能。
添加了新的測試后,再運(yùn)行 go test 命令失敗了,錯(cuò)誤信息如下:
PS G:\Steed\Documents\Go\src\gopl\ch21\word1> go test
--- FAIL: TestChinesePalindrome (0.00s)
word_test.go:23: IsPalindrome("上海自來水來自海上") = false
--- FAIL: TestSentencePalindrome (0.00s)
word_test.go:30: IsPalindrome("Madam, I'm Adam") = false
FAIL
exit status 1
FAIL gopl/ch21/word1 0.292s
PS G:\Steed\Documents\Go\src\gopl\ch21\word1>
這里是一個(gè)比較好的實(shí)踐,先寫測試然后發(fā)現(xiàn)它觸發(fā)的的錯(cuò)誤。通過這步,可以定位到真正要解決的問題,并在修復(fù)后確認(rèn)問題已經(jīng)解決。
運(yùn)行 go test 還可以指定一些參數(shù):
PS G:\Steed\Documents\Go\src\gopl\ch21\word1> go test -v -run="Chinese|Sentence"
=== RUN TestChinesePalindrome
--- FAIL: TestChinesePalindrome (0.00s)
word_test.go:23: IsPalindrome("上海自來水來自海上") = false
=== RUN TestSentencePalindrome
--- FAIL: TestSentencePalindrome (0.00s)
word_test.go:30: IsPalindrome("Madam, I'm Adam") = false
FAIL
exit status 1
FAIL gopl/ch21/word1 0.250s
PS G:\Steed\Documents\Go\src\gopl\ch21\word1>
參數(shù) -v 可以輸出包中每個(gè)測試用例的名稱和執(zhí)行時(shí)間。默認(rèn)只會輸出有問題的測試。
參數(shù) -run 是一個(gè)正則表達(dá)式,可以使 go test 只運(yùn)行那些測出函數(shù)名稱匹配的函數(shù)。
上面選擇性地只運(yùn)行新的測試用例。一旦之后的修復(fù)使得測試用例通過后,還必須使用不帶開關(guān)的 go test 來運(yùn)行一次完整的測試。
上一版本的函數(shù)比較簡單,使用字節(jié)序列而不是字符序列,因此無法支持非 ASCII 字符的檢查。另外也沒有忽略空格、標(biāo)點(diǎn)符號和字母大小寫。下面重寫了這個(gè)函數(shù):
// word 包提供了文字游戲相關(guān)的工具函數(shù)
package word
import "unicode"
// IsPalindrome 判斷一個(gè)字符串是否是回文
func IsPalindrome(s string) bool {
var letters []rune
for _, r := range s {
if unicode.IsLetter(r) {
letters = append(letters, unicode.ToLower(r))
}
}
for i := range letters {
if letters[i] != letters[len(letters)-1-i] {
return false
}
}
return true
}
測試用例也重新寫。這里是一個(gè)更加全面的測試用例,把之前的用例和新的用例結(jié)合到一個(gè)表里:
package word
import "testing"
func TestIsPalindrome(t *testing.T) {
var tests = []struct {
input string
want bool
}{
{"", true},
{"a", true},
{"aa", true},
{"ab", false},
{"kayak", true},
{"palindrome", false},
{"desserts", false},
{"上海自來水來自海上", true},
{"Madam, I'm Adam", true},
}
for _, test := range tests {
if got := IsPalindrome(test.input); got != test.want {
t.Errorf(`IsPalindrome(%q) = %v`, test.input, got)
}
}
}
這種基于表的測試方式在 Go 里面很常見。根據(jù)需要添加新的表項(xiàng)很直觀,并且由于斷言邏輯沒有重復(fù),因此可以花點(diǎn)精力讓輸出的錯(cuò)誤消息更好看一點(diǎn)。
調(diào)用 t.Errorf 輸出的失敗的測試用例信息沒有包含整個(gè)跟蹤棧信息,也不會導(dǎo)致程序終止執(zhí)行。這樣可以在一次測試過程中發(fā)現(xiàn)多個(gè)失敗的情況。
如果需要在測試函數(shù)中終止,比如由于初始化代碼失敗,可以使用 t.Fatal 或 t.Fatalf 函數(shù)來終止當(dāng)前測試函數(shù),它們必須在測試函數(shù)的同一個(gè) goroutine 內(nèi)調(diào)用。
測試錯(cuò)誤消息的建議
測試錯(cuò)誤消息一般格式是 f(x)=y, want z
,這里 f(x) 表示需要執(zhí)行的操作和它的輸入,y 是實(shí)際的輸出結(jié)果,z 是期望得到的結(jié)果。在測試一個(gè)布爾函數(shù)的時(shí)候,省略 “want z” 部分,因?yàn)樗鼪]有給出有用的信息。上面的測試用例輸出的錯(cuò)誤消息基本也是這么做的,
基于表的測試方便針對精心選擇的輸入檢測函數(shù)是否工作正常,以測試邏輯上引人關(guān)注的用例。另外一種方式是隨機(jī)測試,通過構(gòu)建隨機(jī)輸入來擴(kuò)展測試的覆蓋范圍。
對于隨機(jī)的輸入,要如何確認(rèn)輸出是否正確,這里有兩種策略:
下面的例子使用了第二種模式,randomPalindrome 函數(shù)可以隨機(jī)的創(chuàng)建回文字符串,使用這些回文字符串來驗(yàn)證進(jìn)行測試:
import (
"math/rand"
"testing"
"time"
)
// randomPalindrome 返回一個(gè)回文字符串,它的長度和內(nèi)容都是隨機(jī)生成的
func randomPalindrome(rng *rand.Rand) string {
n := rng.Intn(25) // 隨機(jī)字符串最大長度24
runes := make([]rune, n)
for i := 0; i < (n+1)/2; i++ {
r := rune(rng.Intn(0x1000)) // 隨機(jī)字符最大是 `\u0999
runes[i] = r
runes[n-1-i] = r
}
return string(runes)
}
func TestRandomPalindromes(t *testing.T) {
seed := time.Now().UTC().UnixNano()
t.Logf("Random seed: %d", seed)
rng := rand.New(rand.NewSource(seed))
for i := 0; i < 1000; i++ {
p := randomPalindrome(rng)
if !IsPalindrome(p) {
t.Errorf("IsPalindrome(%q) = false", p)
}
}
}
由于隨機(jī)測試的不確定性,在遇到測試用例失敗的情況下,一定要記錄足夠多的信息以便于重現(xiàn)這個(gè)問題。這里記錄偽隨機(jī)數(shù)生成的種子會比轉(zhuǎn)存儲整個(gè)輸入數(shù)據(jù)結(jié)構(gòu)要簡單得多。有了隨機(jī)數(shù)的種子,就可以簡單地修改測試代碼來準(zhǔn)確地重現(xiàn)錯(cuò)誤。
通過使用當(dāng)前時(shí)間作為偽隨機(jī)數(shù)的種子源,在測試的整個(gè)生命周期中,每次運(yùn)行的時(shí)候都會得到新的輸入。如果你的項(xiàng)目使用自動化系統(tǒng)來周期地運(yùn)行測試,這一點(diǎn)很重要。
就是測試命令源碼文件,其實(shí)和測試包源碼文件差不多。畢竟都是一樣的代碼,不過需要額外做一些特殊的處理。
對于包的測試,go test 很有用,但是稍加修改,也能夠?qū)⑺脕頊y試可執(zhí)行程序。一個(gè) main 包可以生成可執(zhí)行程序,不過也可以當(dāng)做庫來導(dǎo)入。
下面的 echo 程序,可以輸出命令行參數(shù):
// 輸出命令行參數(shù)
package main
import (
"flag"
"fmt"
"strings"
)
var n = flag.Bool("n", false, "omit trailing newline")
var sep = flag.String("s", " ", "separator")
func main() {
flag.Parse()
fmt.Print(strings.Join(flag.Args(), *sep))
if !*n {
fmt.Println()
}
}
為了便于測試,需要對程序進(jìn)行修改。把程序分成兩個(gè)函數(shù),echo 執(zhí)行邏輯,main 用來讀取和解析命令行參數(shù)以及報(bào)告 echo 函數(shù)可能返回的錯(cuò)誤:
// 輸出命令行參數(shù)
package main
import (
"flag"
"fmt"
"io"
"os"
"strings"
)
var (
n = flag.Bool("n", false, "omit trailing newline")
sep = flag.String("s", " ", "separator")
)
var out io.Writer = os.Stdout // 測試過程中將會被更改
func main() {
flag.Parse()
if err := echo(!*n, *sep, flag.Args()); err != nil {
fmt.Fprintf(os.Stderr, "echo: %v\n", err)
os.Exit(1)
}
}
func echo(newline bool, sep string, args []string) error {
fmt.Fprintf(out, strings.Join(args, sep))
if newline {
fmt.Fprintln(out)
}
return nil
}
分離出執(zhí)行邏輯
把程序的主要功能從 main 函數(shù)里分離出來了,運(yùn)行程序的時(shí)候通過 main 函數(shù)來調(diào)用 echo。而測試的時(shí)候,就可以直接對 echo 函數(shù)進(jìn)行測試。
避免依賴全局變量
在接下來的測試中,將通過不同的參數(shù)和開關(guān)來調(diào)用 echo,以檢查它在不同的模式下都能正常工作。這里的 echo 函數(shù)調(diào)用的時(shí)候,通過傳參獲取這些信息,這是為了避免函數(shù)依賴全局變量,這樣測試的時(shí)候也可以直接傳參來調(diào)用 echo 不同的模式。
控制輸出的變量
這里還另外引入了一個(gè)全局變量 out,該變量是 io.Writer 類型,所有的結(jié)果都將輸出到這里。echo 函數(shù)的輸出是輸出到 out 變量而不是直接輸出到 os.Stdout。這樣正常使用的時(shí)候,就是輸出到用戶界面,而測試的時(shí)候,可以覆蓋掉這個(gè)變量輸出到其他地方。這樣是實(shí)現(xiàn)了記錄寫入的內(nèi)容以便于檢查。
下面是測試代碼,在文件 echo_test.go 中:
package main
import (
"bytes"
"fmt"
"testing"
)
func TestEcho(t *testing.T) {
var tests = []struct {
newline bool
sep string
args []string
want string
}{
{true, "", []string{}, "\n"},
{false, "", []string{}, ""},
{true, "\t", []string{"one", "two", "three"}, "one\ttwo\tthree\n"},
{true, ",", []string{"a", "b", "c"}, "a,b,c\n"},
{false, ":", []string{"1", "2", "3"}, "1:2:3"},
}
for _, test := range tests {
descr := fmt.Sprintf("echo(%v, %q, %q)", test.newline, test.sep, test.args)
out = new(bytes.Buffer) // 捕獲的輸出
if err := echo(test.newline, test.sep, test.args); err != nil {
t.Errorf("%s failed: %v", descr, err)
continue
}
got := out.(*bytes.Buffer).String()
if got != test.want {
t.Errorf("%s = %q, want %q", descr, got, test.want)
}
}
}
這里依然是通過表來組織測試用例,這樣可以很容易地添加新的測試用例。下面是添加了一行到測試用例中:
{false, ":", []string{"1", "2", "3"}, "1:2:3\n"},
上面添加的這條是有錯(cuò)誤的,正好可以看看測試失敗的時(shí)候的輸出:
PS H:\Go\src\gopl\ch21\echo> go test
--- FAIL: TestEcho (0.00s)
echo_test.go:32: echo(false, ":", ["1" "2" "3"]) = "1:2:3", want "1:2:3\n"
FAIL
exit status 1
FAIL gopl/ch21/echo 0.163s
PS H:\Go\src\gopl\ch21\echo>
錯(cuò)誤信息首先描述了想要進(jìn)行的操作,使用了類似 Go 的語法,就像一個(gè)函數(shù)調(diào)用。然后依次是實(shí)際獲得個(gè)值和預(yù)期的結(jié)果。這樣的錯(cuò)誤信息就很有幫助。
測試中的錯(cuò)誤處理
還要注意,測試代碼里并沒有調(diào)用 log.Fatal 或 os.Exit,因?yàn)檫@兩個(gè)調(diào)用會阻止跟蹤的過程,這兩個(gè)函數(shù)的調(diào)用可以認(rèn)為是 main 函數(shù)的特權(quán)。如果有時(shí)候發(fā)生了未預(yù)期的錯(cuò)誤或者崩潰,即使測試用例本身失敗了,測試驅(qū)動程序也還可以繼續(xù)工作。預(yù)期的的錯(cuò)誤應(yīng)該通過返回一個(gè)非空的 error 值來報(bào)告,就像上面的測試代碼里做的那樣。
測試的一種分類方式是基于對所要進(jìn)行測試的包的內(nèi)部的了解程度:
白盒這個(gè)名字是傳統(tǒng)的說法,凈盒(clear box)的說法更準(zhǔn)確。
以上兩種方法是互補(bǔ)的。黑盒測試通常更加健壯,程序更新后基本不需要修改。并且可以幫助測試者了解用戶的情況以及發(fā)現(xiàn)API設(shè)計(jì)的缺陷。反之,白盒測試可以對實(shí)現(xiàn)的特定之處提供更詳細(xì)的覆蓋測試。
之前的內(nèi)容已經(jīng)分別給出了這兩種測試方法的例子:
偽實(shí)現(xiàn)
在寫 TestEcho 的時(shí)候,通過修改 echo 函數(shù),從而在輸出結(jié)果時(shí)使用了一個(gè)包級別的變量,使得測試可以使用一個(gè)額外的實(shí)現(xiàn)代替標(biāo)準(zhǔn)輸出來記錄要檢查的數(shù)據(jù)。通過這樣的技術(shù),可以使用易于測試的偽實(shí)現(xiàn)來替換部分產(chǎn)品代碼。這種偽實(shí)現(xiàn)的優(yōu)點(diǎn)是更易于配置、預(yù)測和觀察,并且更可靠。
下面的代碼演示了向用戶提供存儲服務(wù)的 Web 服務(wù)中的限額邏輯。當(dāng)用戶使用的額度超過 90% 的時(shí)候,系統(tǒng)自動發(fā)送一封告警郵件:
package storage
import (
"fmt"
"log"
"net/smtp"
)
var usage = make(map[string]int64)
func bytesInUse(username string) int64 { return usage[username] }
// 郵件發(fā)送者配置
// 注意:永遠(yuǎn)不要把密碼放到源代碼中
const sender = "notifications@example.com"
const password = "password"
const hostname = "smtp.example.com"
const template = `Warning: you are using %d bytes of storage,
%d%% of your quota.`
func CheckQuota(username string) {
used := bytesInUse(username)
const quota = 1000000000 // 1GB
percent := 100 * used / quota
if percent < 90 {
return // OK
}
msg := fmt.Sprintf(template, used, percent)
auth := smtp.PlainAuth("", sender, password, hostname)
err := smtp.SendMail(hostname+":587", auth, sender,
[]string{username}, []byte(msg))
if err != nil {
log.Printf("smtp.SendMail(%s) failed: %s", username, err)
}
}
現(xiàn)在想要測試上面的功能,但是并不想真的發(fā)送郵件。所以要把發(fā)送郵件的邏輯移動到獨(dú)立的函數(shù)中,并且把它存儲到一個(gè)不可導(dǎo)出的變量 notifyUser 中:
var notifyUser = func(username, msg string) {
auth := smtp.PlainAuth("", sender, password, hostname)
err := smtp.SendMail(hostname+":587", auth, sender,
[]string{username}, []byte(msg))
if err != nil {
log.Printf("smtp.SendMail(%s) failed: %s", username, err)
}
}
func CheckQuota(username string) {
used := bytesInUse(username)
const quota = 1000000000 // 1GB
percent := 100 * used / quota
if percent < 90 {
return // OK
}
msg := fmt.Sprintf(template, used, percent)
notifyUser(username, msg)
}
現(xiàn)在可以寫測試了。
下面是一個(gè)簡單的測試,這個(gè)測試用偽造的通知機(jī)制而不是真的發(fā)送郵件。這個(gè)測試會記錄下需要通知的用戶和通知的內(nèi)容,并驗(yàn)證是否符合期望:
package storage
import (
"strings"
"testing"
)
func TestCheckQuotaNotifiesUser(t *testing.T) {
var notifiedUser, notifiedMsg string
notifyUser = func(user, msg string) {
notifiedUser, notifiedMsg = user, msg
}
const user = "steed@example.org"
usage[user] = 980000000 // 模擬已經(jīng)使用了 980M 的情況
CheckQuota(user)
if notifiedUser == "" && notifiedMsg == "" {
t.Fatalf("notifyUser not called") // 比如沒有超過限額,就會進(jìn)入這個(gè)分支
}
if notifiedUser != user {
t.Errorf("wrong user (%s) notified, want %s", notifiedUser, user)
}
const wantSubstring = "98% of your quota"
if !strings.Contains(notifiedMsg, wantSubstring) {
t.Errorf("unexpected notification message <<%s>>, want substring %q", notifiedMsg, wantSubstring)
}
}
目前來看,這個(gè)測試本身完成的很好,但是還有一個(gè)遺留問題。因?yàn)閷?CheckQuota 測試中使用了偽實(shí)現(xiàn)替換了原本的 notifyUser 的內(nèi)容,這樣在之后的其他測試中,notifyUser 依然是這里被替換上的偽實(shí)現(xiàn),這可能使得其他的測試無法正常工作(對于全局變量的更新一直都是存在風(fēng)險(xiǎn)的)。這里還必須再修改一下這個(gè)測試讓他最后可以恢復(fù) notifyUser 原來的值,這樣之后的測試就不會收到影響。這里必須在所有的測試執(zhí)行路徑上這樣做,包括測試失敗和崩潰的情況。通常這種情況下建議使用 defer :
func TestCheckQuotaNotifiesUser(t *testing.T) {
// 保存留待恢復(fù)的notifyUser
saved := notifyUser
defer func() { notifyUser = saved }()
// 設(shè)置測試的偽通知notifyUser
var notifiedUser, notifiedMsg string
notifyUser = func(user, msg string) {
notifiedUser, notifiedMsg = user, msg
}
// ...測試其余的部分...
}
以這種方式來使用全局變量是安全的,因?yàn)?go test 一般不會并發(fā)執(zhí)行多個(gè)測試。
這種方式有很多用處:
先來看一下 net/url 包,這個(gè)包提供了 URL 解析的功能。還有 net/http 包,這個(gè)包提供了 Web 服務(wù)器和 HTTP 客戶端的庫。高級的 net/http 包依賴于低級的 net/url 包。然而,在 net/url 包中有一個(gè)測試是用來演示 URL 和 HTTP 庫之間進(jìn)行交互的例子。也就是說,低級別包的測試導(dǎo)入了高級別包。這種情況下,在 net/url 包中聲明的這個(gè)測試函數(shù)會導(dǎo)致包的循環(huán)引用,但是 Go 規(guī)范禁止循環(huán)引用。
為了解決測試時(shí)可能會出現(xiàn)的循環(huán)引用的問題,可以將這個(gè)測試函數(shù)定義在外部測試包中。
具體做法就是,測試文件的包名不和被測試的包同名,而是使用一個(gè)新的包名。在這個(gè)例子里,就是原本包名是 url,現(xiàn)在因?yàn)橐獙?dǎo)入高級別的包會出現(xiàn)循環(huán)引用,所以將包名改成一個(gè)別的名稱,比如 url_test。這個(gè)額外的后綴 _test 告訴 go test 工具,它應(yīng)該單獨(dú)地編譯這個(gè)包,然后進(jìn)行它的測試。為了便于理解,可以認(rèn)為這個(gè)外部測試包的導(dǎo)入路徑是 net/url_test,但事實(shí)上它無法通過任何路徑導(dǎo)入。
由于外部測試在一個(gè)單獨(dú)的包里,因此它們可以引用一些依賴于被測試包的幫助包,這個(gè)是包內(nèi)測試無法做到的。從設(shè)計(jì)層次來看,外部測試包邏輯上在它所依賴的兩個(gè)包之上。
為了避免包循環(huán)導(dǎo)入,外部測試包允許測試用例,尤其是集成測試用例(用來測試多個(gè)組件的交互),自由地導(dǎo)入其他的包,就像一個(gè)引用程序那樣。
可以使用 go list 工具來匯總一個(gè)包目錄中哪些是產(chǎn)品代碼,哪些是包內(nèi)測試、哪些是外部測試。這里用 fmt 包作為例子。
GoFiles
這類文件是包含產(chǎn)品代碼的文件列表,這些文件是 go build 命令將編譯進(jìn)程序的代碼:
PS H:\Go\src\gopl\ch21> go list -f="{{.GoFiles}}" fmt
[doc.go format.go print.go scan.go]
TestGoFiles
這類文件也屬于 fmt 包,但是這些以 _test.go 結(jié)尾的文件是測試源碼文件,僅在編譯測試的時(shí)候才會使用:
PS H:\Go\src\gopl\ch21> go list -f="{{.TestGoFiles}}" fmt
[export_test.go]
這里的 export_test.go 這個(gè)文件還有特殊的意義,后面會單獨(dú)講。
XTestGoFiles
這類是包外部測試文件列表,這些同樣的測試源碼文件,僅用在測試過程中:
PS H:\Go\src\gopl\ch21> go list -f="{{.XTestGoFiles}}" fmt
[example_test.go fmt_test.go scan_test.go stringer_test.go]
這是一個(gè)在外部測試中使用白盒測試的技巧,包內(nèi)的白盒測試沒有這個(gè)問題。
有時(shí)候,外部測試包需要對被測試包擁有特殊的訪問權(quán)限。比如這種的情況:為了避免循環(huán)引用,需要聲明外部測試包,但是又要做白盒測試,需要調(diào)用非導(dǎo)出的變量和函數(shù)。
應(yīng)對這種情況,需要使用一種小技巧:在包內(nèi)測試文件中添加一些聲明,將包內(nèi)部的功能暴露給外部測試。由于是聲明在測試文件中的,所以暴露的后門只有在測試時(shí)可用。如果一個(gè)源文件存在的唯一目的就在于此,并且也不包含任何測試,這個(gè)文件一般就命名為 export_test.go。
下面是 fmt 包的 export_test.go 文件里所有的代碼部分:
package fmt
var IsSpace = isSpace
var Parsenum = parsenum
fmt 包的實(shí)現(xiàn)需要功能 unicode.isSpace 作為 fmt.Scanf 的一部分。為了避免創(chuàng)建不合理的依賴,fmt 沒有導(dǎo)入 unicode 包及其巨大的數(shù)據(jù)表,而是包含了一個(gè)更加簡單的實(shí)現(xiàn) isSpace。
為了確保 fmt.isSpace 和 unicode.isSpace 的功能一致,fmt 添加了一個(gè)測試。這是一個(gè)集成測試,所以用了外部測試包。但是測試中需要訪問 isSpace,這是一個(gè)非導(dǎo)出的函數(shù)。所以就有了上面的代碼,定義了一個(gè)可導(dǎo)出的變量來引用 isSpace 函數(shù)。并且這段代碼是定義在測試文件中的,所以無法在產(chǎn)品代碼中訪問到這個(gè)函數(shù)。
這個(gè)技巧在任何外部測試需要使用白盒測試技術(shù)的時(shí)候都可以使用。
Go 語言的測試期望測試的編寫者自己來做大部分工作,通過定義函數(shù)來避免重復(fù)。測試的過程不是死記硬背地填表格,測試也是有用戶界面的,雖然它的用戶也是它的維護(hù)者。
一個(gè)好的測試,不會在發(fā)生錯(cuò)誤時(shí)崩潰,而是要輸出一個(gè)簡潔、清晰的現(xiàn)象描述來報(bào)告錯(cuò)誤,以及與之上下文相關(guān)的信息。理想情況下,不需要再通過閱讀源代碼來探究失敗的原因。
一個(gè)好的測試,不應(yīng)該在發(fā)現(xiàn)一次測試失敗后就終止,而是要在一次運(yùn)行中嘗試報(bào)告多個(gè)錯(cuò)誤,因?yàn)殄e(cuò)誤發(fā)生的方式本身會揭露錯(cuò)誤的原因。
下面的斷言函數(shù)比較兩個(gè)值,構(gòu)建一條一般的錯(cuò)誤消息,并且停止程序。這是一個(gè)錯(cuò)誤的例子,輸出的錯(cuò)誤消息毫無用處。它的最大的問題就是沒有提供一個(gè)好的用戶界面:
import (
"fmt"
"strings"
"testing"
)
// 一個(gè)糟糕的斷言函數(shù)
func assertEqual(x, y int) {
if x != y {
panic(fmt.Sprintf("%d != %d", x, y))
}
}
func TestSplit(t *testing.T) {
words := strings.Split("a:b:c", ":")
assertEqual(len(words), 3)
// ...
}
合適的做法
這里斷言函數(shù)犯了過早抽象的錯(cuò)誤:僅僅測試兩個(gè)整數(shù)是否相同,而沒能根據(jù)上下文提供更有意義的錯(cuò)誤信息。這里可以根據(jù)具體的錯(cuò)誤信息提供一個(gè)更好的錯(cuò)誤輸出。比如下面的做法。只有在測試中出現(xiàn)了重復(fù)的模式時(shí)才需要引入抽象:
func TestSplit(t *testing.T) {
s, sep := "a:b:c", ":"
words := strings.Split(s, sep)
if got, want := len(words), 3; got != want {
t.Errorf("Split(%q, %q) returned %d words, want %d",
s, sep, got, want)
}
// ...
}
現(xiàn)在測試函數(shù)友好的用戶界面表現(xiàn)在一下幾個(gè)方面
當(dāng)有了這樣的一個(gè)測試函數(shù)之后,下一步不是定義一個(gè)函數(shù)來替代整個(gè) if 語句,而是在一個(gè)循環(huán)中執(zhí)行這個(gè)測試,就像之前基于表的測試方式那樣。
當(dāng)然定義一個(gè)函數(shù)來替代整個(gè) if 語句也是可以的做法,只是這個(gè)例子太簡單了,并不需要任何工具函數(shù)。但是為了使得測試代碼更簡潔,也可以考慮引入工具函數(shù),如果上面的 assertEqual 函數(shù)的實(shí)現(xiàn)的用戶界面更加友好的話。并且如果這種模式在其他測試代碼里也會重復(fù)用到,那就更有必要進(jìn)行抽象了。
一個(gè)好的測試的關(guān)鍵是首先實(shí)現(xiàn)你所期望的具體行為,之后再使用工具函數(shù)來使代碼簡潔并且避免重復(fù)。好的結(jié)果很少是從抽象的、通用的測試函數(shù)開始的。
這里再預(yù)告一點(diǎn),比較兩個(gè)變量的值在測試中很常見,并且會需要對各種類型的值進(jìn)行比較,這就需要基于反射來實(shí)現(xiàn)。另外還會需要比較復(fù)合類型,這通過基于地址來判斷引用的變量是否是同一個(gè)變量來實(shí)現(xiàn),這是 unsafe 包的內(nèi)容。在掌握了反射的內(nèi)容之后,在 unsafe 包的內(nèi)容里,會實(shí)現(xiàn)一個(gè)深度相等的工具函數(shù)。
如果一個(gè)應(yīng)用在遇到新的合法輸入的情況下經(jīng)常崩潰,那么這個(gè)程序是有缺陷的。
如果在程序發(fā)生可靠的改動的時(shí)候測試用例奇怪地失敗了,那么這個(gè)測試用例也是脆弱的。
避免寫出脆弱測試的最簡單的方法就是僅檢查你關(guān)心的屬性。例如,不要對輸出的字符串進(jìn)行完全匹配,而是尋找到在程序進(jìn)化過程中不會發(fā)生改變的子串。通常情況下,這值得寫一個(gè)穩(wěn)定的函數(shù)來從復(fù)雜的輸出中提取核心內(nèi)容,只有這樣之后的斷言才會可靠。這雖然需要一些額外的工作,但這是值得的,否則這些時(shí)間會被花在修復(fù)那些奇怪地失敗的測試上面。
語句覆蓋率是一種最簡單的且廣泛使用的方法之一。一個(gè)測試套件的語句覆蓋率是指部分語句在一次執(zhí)行中執(zhí)行執(zhí)行一次。可以使用 go cover 工具,這個(gè)工具被集成到了 go test 中,用來衡量語句覆蓋率并幫助識別測試之間的明顯差別。
如果使用VSCode,直接通過測試源碼文件里的按鈕運(yùn)行測試,再切換到源碼文件中就能看到測試覆蓋率的效果。下面講的是不依賴編輯器和插件的做法。
通過下面的命令可以輸出覆蓋工具的使用方法:
PS G:\Steed\Documents\Go\src\gopl\ch21\storage2> go tool cover
Usage of 'go tool cover':
Given a coverage profile produced by 'go test':
go test -coverprofile=c.out
Open a web browser displaying annotated source code:
go tool cover -html=c.out
...
命令 go tool 運(yùn)行 Go 工具鏈里的一個(gè)可執(zhí)行文件。這些程序位于 $GOROOT/pkg/tool/${GOOS}_{GOARCH}
,就是 Go 安裝目錄里的文件夾下,都是一些 exe 文件。這里多虧了 go build 工具,我們不需要直接運(yùn)行它。
-coverprofile 標(biāo)記
要生成覆蓋率報(bào)告,需要帶上 -coverprofile 標(biāo)記來運(yùn)行測試:
PS G:\Steed\Documents\Go\src> go test -run=CheckQuotaNotifiesUser -coverprofile="c.out" gopl/ch21/storage2
ok gopl/ch21/storage2 0.349s coverage: 58.3% of statements
PS G:\Steed\Documents\Go\src>
這個(gè)標(biāo)記通過檢測產(chǎn)品代碼,啟用了覆蓋數(shù)據(jù)收集。也就是說,它修改了源代碼的副本,這樣在這個(gè)語句塊執(zhí)行之前,設(shè)置一個(gè)布爾變量,每個(gè)語句塊都對應(yīng)一個(gè)變量。在修改程序退出之前,它將每個(gè)變量的值都寫入到指定的日志文件,這里是 c.out,并記錄被執(zhí)行語句的匯總信息。
-cover 標(biāo)記
如果不需要記錄這個(gè)日志文件而只要查看命令行輸出的內(nèi)容,可以使用 -cover 標(biāo)記:
PS G:\Steed\Documents\Go\src> go test -run=CheckQuotaNotifiesUser -cover gopl/ch21/storage2
ok gopl/ch21/storage2 0.366s coverage: 58.3% of statements
PS G:\Steed\Documents\Go\src>
效果是一樣的,只是不生成記錄文件。
-convermode=count 標(biāo)記
默認(rèn)的 mode 是 set。這個(gè)標(biāo)記使每個(gè)語句塊的檢測使用一個(gè)遞增計(jì)數(shù)器來替代原本的布爾值。這樣日志中就能統(tǒng)計(jì)到每個(gè)塊的執(zhí)行次數(shù),由此可以識別出執(zhí)行頻率較高的“熱塊”和相反的“冷塊”。
VSCode似乎不能指定這個(gè)模式,所以只能生成查看布爾值的報(bào)告,檢查代碼是否被覆蓋,看不到熱塊和冷塊的效果。
在生成數(shù)據(jù)后,運(yùn)行 cover 工具來處理生成的日志,可以生成一個(gè) HTML 報(bào)告。可以在瀏覽器里直觀的查看:
PS G:\Steed\Documents\Go\src> go tool cover -html="c.out"
基準(zhǔn)測試就是在一定的工作負(fù)載之下檢測程序性能的一種方法。
基準(zhǔn)測試函數(shù)看上去和功能測試函數(shù)差不多,前綴是 Benchmark 并且擁有一個(gè) *testing.B 參數(shù)。*testing.B 和 *testing.T 差不多,還額外增加了一些和性能檢測相關(guān)的方法。另外它還有一個(gè)整型成員 *testing.B.N,用來指定被檢測操作的執(zhí)行次數(shù)。
回到之前的檢查回文的函數(shù),下面是 IsPalindrome 函數(shù)的基準(zhǔn)測試,它在一個(gè)循環(huán)中調(diào)用了 IsPalindrome 共 N 次:
func BenchmarkIsPalindrome(b *testing.B) {
for i := 0; i < b.N; i++ {
IsPalindrome("山西懸空寺空懸西山")
}
}
上面的基準(zhǔn)測試函數(shù)直接加到之前的測試源碼文件中。
在基準(zhǔn)測試函數(shù)中手動寫代碼來實(shí)現(xiàn)循環(huán),而不是在測試驅(qū)動程序中自動實(shí)現(xiàn)是有原因的。在基準(zhǔn)測試函數(shù)中,for循環(huán)之外,可以執(zhí)行一些必要的初始化代碼并且這段時(shí)間不會加到每次迭代的時(shí)間中去。如果有代碼會干擾結(jié)果,參數(shù) testing.B 還提供了方法來停止、恢復(fù)和重置計(jì)時(shí)器(需要用到的場景并不多)。
依然是使用 go test 命令來進(jìn)行測試,但是默認(rèn)情況下不會運(yùn)行任何基準(zhǔn)測試。需要加上 -bench 參數(shù)并指定有運(yùn)行的基準(zhǔn)測試。它是一個(gè)匹配 Benchmark 函數(shù)名稱的正則表達(dá)式,默認(rèn)值不匹配任何函數(shù)。可以使用點(diǎn)來匹配所有的基準(zhǔn)測試函數(shù):
PS G:\Steed\Documents\Go\src\gopl\ch21\word2> go test -bench="."
goos: windows
goarch: amd64
pkg: gopl/ch21/word2
BenchmarkIsPalindrome-4 1000000 1052 ns/op
PASS
ok gopl/ch21/word2 2.253s
PS G:\Steed\Documents\Go\src\gopl\ch21\word2>
基準(zhǔn)測試函數(shù)名稱后面的數(shù)字后綴表示 GOMAXPROCS 的值。這對于一些并發(fā)相關(guān)的基準(zhǔn)測試是一個(gè)重要的信息。
報(bào)告顯示每次調(diào)用 IsPalindrome 的平均耗時(shí)是 1.052ms,這個(gè)是 1000000 次調(diào)用的平均值。基準(zhǔn)測試運(yùn)行器在開始的時(shí)候并不清楚測試操作的耗時(shí),所以開始會用比較小的N值來做檢測,然后為了檢測穩(wěn)定的運(yùn)行時(shí)間,會推斷出一個(gè)較大的次數(shù)來保證得到穩(wěn)定的測試結(jié)果。
現(xiàn)在有了基準(zhǔn)測試,那么就先想辦法來讓程序更快一點(diǎn),然后再運(yùn)行基準(zhǔn)測試來檢查具體快了多少。
有一處是明顯可以改進(jìn)的,只需要遍歷字符串前面一半的字符就可以完成字符串的檢查。避免了第二次的重復(fù)比較:
n := len(letters)
for i := 0; i < n; i++ {
if letters[i] != letters[len(letters)-1-i] {
return false
}
}
return true
但是通常情況下,優(yōu)化并不能總是帶來期望的好處。這個(gè)優(yōu)化后的運(yùn)行時(shí)間也就 1.004ms,只有4.5%的提升。
另外還有一處可以優(yōu)化,為 letters 預(yù)分配一個(gè)容量足夠大的數(shù)組,避免在 append 調(diào)用的時(shí)候多次進(jìn)行擴(kuò)容:
// var letters []rune
letters := make([]rune, 0, len(s))
for _, r := range s {
if unicode.IsLetter(r) {
letters = append(letters, unicode.ToLower(r))
}
}
這次改進(jìn)后平均運(yùn)行時(shí)間縮短到了 0.839ms,提升了20%。
如上面的例子所示,最快的程序通常是那些進(jìn)行內(nèi)存分配數(shù)量最少的程序。命令行標(biāo)記 -benchmem 在報(bào)告中會包含內(nèi)存分配統(tǒng)計(jì)數(shù)據(jù)。下面是優(yōu)化前后兩個(gè)函數(shù)的基準(zhǔn)測試報(bào)告:
Running tool: D:\Go\bin\go.exe test -benchmem -run=^$ gopl\ch21\word2 -bench . -coverprofile=C:\Users\Steed\AppData\Local\Temp\vscode-gotvbvaq\go-code-cover
goos: windows
goarch: amd64
pkg: gopl/ch21/word2
BenchmarkIsPalindrome-4 1000000 1095 ns/op 120 B/op 4 allocs/op
BenchmarkIsPalindrome2-4 2000000 871 ns/op 112 B/op 1 allocs/op
PASS
coverage: 88.2% of statements
ok gopl/ch21/word2 4.185s
Success: Benchmarks passed.
優(yōu)化前有4次內(nèi)存分配,分配了120B的內(nèi)存。優(yōu)化有只進(jìn)行了1次內(nèi)存分配,分配了112B的內(nèi)存。(這里關(guān)于內(nèi)存的分配主要是切片擴(kuò)容的機(jī)制。)
之前的性能測試是告訴我們給定操作的絕對耗時(shí),但是在很多情況下,需要關(guān)注的問題是兩個(gè)不同操作之間的相對耗時(shí)。比如如下的場景:
性能比較函數(shù)只是普通的代碼,表現(xiàn)形式通常是帶有一個(gè)參數(shù)的函數(shù),再被多個(gè)不同的 Benchmark 函數(shù)傳入不同的值來調(diào)用,比如下面這樣:
func benchmark(b *testing.B, size int) { /* ... */ }
func Benchmark10(b *testing.B) { benchmark(b, 10) }
func Benchmark100(b *testing.B) { benchmark(b, 100) }
func Benchmark1000(b *testing.B) { benchmark(b, 1000) }
參數(shù) size 指定了輸入的大小,每個(gè) Benchmark 函數(shù)傳入的值都不同但是在每個(gè)函數(shù)內(nèi)部是一個(gè)常量。不要使用 b.N 來控制輸入的大小。除非是把它當(dāng)做固定大小輸入的循環(huán)次數(shù),否則基準(zhǔn)測試的結(jié)果將毫無意義。
基準(zhǔn)測試比較揭示的模式在程序設(shè)計(jì)階段很有用處,但是即使程序正常工作了,也不要丟掉基準(zhǔn)測試。隨著程序的演變,或者它的輸入增長了,或者它被部署在其他的操作系統(tǒng)上并擁有一些新特性,這時(shí)仍然可以重用基準(zhǔn)測試來回顧當(dāng)初的設(shè)計(jì)決策。
當(dāng)希望仔細(xì)地查看程序的速度是,發(fā)現(xiàn)關(guān)鍵代碼的最佳技術(shù)就是性能剖析。性能剖析是通過自動化手段在程序執(zhí)行過程中基于一些性能事件的采樣來進(jìn)行性能評測,然后再從這些采樣中推斷分析,得到的統(tǒng)計(jì)報(bào)告就稱作為性能剖析(profile)。
Go 支持很多種性能剖析方式。其中,工具 go test 內(nèi)置支持一些類別的性能剖析:
CPU 性能剖析
CPU 性能剖析識別出執(zhí)行過程中需要 CPU 最多的函數(shù)。在每個(gè) CPU 上面執(zhí)行的線程都每隔幾毫秒會定期地被操作系統(tǒng)中斷,在每次中斷過程中記錄一個(gè)性能剖析事件,然后恢復(fù)正常執(zhí)行。
堆性能剖析
堆性能剖析識別出負(fù)責(zé)分配最多內(nèi)存的語句。性能剖析庫對協(xié)程內(nèi)部內(nèi)存分配調(diào)用進(jìn)行采樣,平均每 512KB 的內(nèi)存申請會觸發(fā)一個(gè)性能剖析事件。
阻塞性能剖析
阻塞性能剖析識別出那些阻塞協(xié)程最久的操作,例如系統(tǒng)調(diào)用,通道發(fā)送和接收數(shù)據(jù),以及鎖等待等。性能分析庫在一個(gè) goroutine 每次被上述操作之一阻塞的時(shí)候記錄一個(gè)事件。
獲取性能剖析報(bào)告很容易,只需要像下面這樣指定一個(gè)標(biāo)志參數(shù)即可。一次只獲取一種性能剖析報(bào)告,如果使用了多個(gè)標(biāo)志,一種類別的報(bào)告會把其他類別的報(bào)告覆蓋掉:
$ go test -cpuprofile=cpu.out
$ go test -blockprofile=block.out
$ go test -memprofile=mem.out
還可以對非測試程序進(jìn)行性能剖析,性能剖析對于長時(shí)間運(yùn)行的程序尤其有用。所以 Go 運(yùn)行時(shí)的性能剖析特性可以通過 runtime API 來啟用。
在獲取了性能剖析報(bào)告后,需要使用 pprof 工具來分析它。這是 Go 自帶的一個(gè)工具,但是因?yàn)椴唤?jīng)常使用,所以通過 go tool pprof 間接來使用它。它有很多特性和選項(xiàng),但是基本的用法只有兩個(gè)參數(shù):
為了使得性能剖析過程高效并且節(jié)約空間,性能剖析日志里沒有包含函數(shù)名稱而是使用它們的地址。這就需要可執(zhí)行文件才能理解理解數(shù)據(jù)內(nèi)容。通常情況下 go test 工具在測試完成之后就丟棄了用于測試而臨時(shí)產(chǎn)生的可執(zhí)行文件,但在性能剖析啟用的時(shí)候,它保存并把可執(zhí)行文件命名為 foo.test,其中 foo 是被測試包的名字。
下面的命令演示如何獲取和顯示簡單的 CPU 性能剖析。這里選擇了 net\/http 包中的一個(gè)基準(zhǔn)測試。通常情況下最后對我們關(guān)心的具有代表性的具體負(fù)載而構(gòu)建的基準(zhǔn)測試進(jìn)行性能剖析。對測試用例進(jìn)行基準(zhǔn)測試永遠(yuǎn)沒有代表性,這里使用了過濾器 -run=NONE 來禁止那些測試:
F:\>go test -run=NONE -bench=ClientServerParallelTLS64 -cpuprofile=cpu.log net/http
goos: windows
goarch: amd64
pkg: net/http
BenchmarkClientServerParallelTLS64-4 2019/04/24 15:40:39 http: TLS handshake error from 127.0.0.1:55188: read tcp 127.0.0.1:55163->127.0.0.1:55188: use of closed network connection
2019/04/24 15:40:39 http: TLS handshake error from 127.0.0.1:55366: read tcp 127.0.0.1:55264->127.0.0.1:55366: use of closed network connection
2019/04/24 15:40:41 http: TLS handshake error from 127.0.0.1:57477: read tcp 127.0.0.1:57266->127.0.0.1:57477: use of closed network connection
10000 198886 ns/op 9578 B/op 107 allocs/op
PASS
ok net/http 3.697s
F:\>
運(yùn)行完上面的測試后,會生成兩個(gè)文件,一個(gè)是測試報(bào)告,一個(gè)是用于測試而臨時(shí)產(chǎn)生的可執(zhí)行文件。再用下面的命令打印測試報(bào)告:
F:\>go tool pprof -text -nodecount=10 ./http.test cpu.log
./http.test: open ./http.test: The system cannot find the file specified.
Fetched 1 source profiles out of 2
Type: cpu
Time: Apr 24, 2019 at 3:40pm (CST)
Duration: 2.71s, Total samples = 9820ms (362.69%)
Showing nodes accounting for 5720ms, 58.25% of 9820ms total
Dropped 370 nodes (cum <= 49.10ms)
Showing top 10 nodes out of 217
flat flat% sum% cum cum%
4220ms 42.97% 42.97% 4270ms 43.48% runtime.cgocall
210ms 2.14% 45.11% 260ms 2.65% runtime.step
200ms 2.04% 47.15% 490ms 4.99% runtime.pcvalue
190ms 1.93% 49.08% 190ms 1.93% math/big.addMulVVW
180ms 1.83% 50.92% 180ms 1.83% runtime.osyield
160ms 1.63% 52.55% 320ms 3.26% runtime.scanobject
160ms 1.63% 54.18% 160ms 1.63% vendor/golang_org/x/crypto/curve25519.ladderstep
150ms 1.53% 55.70% 150ms 1.53% runtime.findObject
140ms 1.43% 57.13% 140ms 1.43% runtime.memmove
110ms 1.12% 58.25% 1020ms 10.39% runtime.gentraceback
F:\>
標(biāo)記 -text 指定輸出的格式,這里用的是一個(gè)文本表格,表格中每行是一個(gè)函數(shù),這些函數(shù)是根據(jù)消耗CPU最多的規(guī)則排序的“熱函數(shù)”。
標(biāo)記 -nodecount=10 限制輸出最高的10條記錄。
這里是一份書上的性能剖析結(jié)果:
$ go tool pprof -text -nodecount=10 ./http.test cpu.log
2570ms of 3590ms total (71.59%)
Dropped 129 nodes (cum <= 17.95ms)
Showing top 10 nodes out of 166 (cum >= 60ms)
flat flat% sum% cum cum%
1730ms 48.19% 48.19% 1750ms 48.75% crypto/elliptic.p256ReduceDegree
230ms 6.41% 54.60% 250ms 6.96% crypto/elliptic.p256Diff
120ms 3.34% 57.94% 120ms 3.34% math/big.addMulVVW
110ms 3.06% 61.00% 110ms 3.06% syscall.Syscall
90ms 2.51% 63.51% 1130ms 31.48% crypto/elliptic.p256Square
70ms 1.95% 65.46% 120ms 3.34% runtime.scanobject
60ms 1.67% 67.13% 830ms 23.12% crypto/elliptic.p256Mul
60ms 1.67% 68.80% 190ms 5.29% math/big.nat.montgomery
50ms 1.39% 70.19% 50ms 1.39% crypto/elliptic.p256ReduceCarry
50ms 1.39% 71.59% 60ms 1.67% crypto/elliptic.p256Sum
這個(gè)性能剖析結(jié)果告訴我們,HTTPS基準(zhǔn)測試中 crypto\/elliptic.p256ReduceDegree 函數(shù)占用了將近一半的CPU資源,對性能占很大比重。
相比之下,上面的性能剖析結(jié)果中,主要是runtime包的內(nèi)存分配的函數(shù),那么減少內(nèi)存消耗是一個(gè)有價(jià)值的優(yōu)化。
對于更微妙的問題,最好使用 pprof 的圖形顯示功能。這需要 GraphViz 工具,可以從 http://www.graphviz.org
下載。然后使用標(biāo)記 -web 生成函數(shù)的有向圖,并能標(biāo)記出函數(shù)的CPU消耗數(shù)值,以及有顏色突出“熱函數(shù)”。點(diǎn)到為止,未展開。
這是第三種也是最后一種測試函數(shù),示例函數(shù)。名字以 Example 開頭,既沒有參數(shù),也沒有返回值。
下面是IsPalindrome函數(shù)對應(yīng)的示例函數(shù):
func ExampleIsPalindrome() {
fmt.Println(IsPalindrome("A man, a plan, a canal: Panama"))
fmt.Println(IsPalindrome("palindrome"))
// Output:
// true
// false
}
示例函數(shù)有三個(gè)目的:
比起乏味的描述,舉一個(gè)好的例子是描述庫函數(shù)功能最簡潔直觀的方式。
基于 Example 函數(shù)的后綴,基于 Web 的文檔服務(wù)器 godoc 可以將示例函數(shù)(比如:ExampleIsPalindrome)和它所演示的函數(shù)或包(比如:IsPalindrome函數(shù)),關(guān)聯(lián)起來。
如果是一個(gè)名字叫 Example 的函數(shù),那么就會和包的文檔關(guān)聯(lián)。
示例函數(shù)是可以通過 go test 運(yùn)行的可執(zhí)行測試。示例函數(shù)的最后如果有一段類型 // Output:
的注釋,就像上面的例子里一樣。測試驅(qū)動程序?qū)?zhí)行這個(gè)函數(shù)并且檢查輸出到終端的內(nèi)容與注釋是否匹配。
http://golang.org 就是由 godoc 提供的文檔服務(wù),它使用 Go Playground 來讓用戶在 Web 瀏覽器上編輯和運(yùn)行每個(gè)示例函數(shù)。這可以作為了解特定函數(shù)功能或者了解語言特性最快捷的方法。
新聞名稱:gopl測試
本文來源:http://vcdvsql.cn/article26/peipcg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供商城網(wǎng)站、網(wǎng)站制作、、網(wǎng)站建設(shè)、虛擬主機(jī)、品牌網(wǎng)站設(shè)計(jì)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)