校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁(yè) > 知識(shí)庫(kù) > 一步步教你編寫可測(cè)試的Go語(yǔ)言代碼

一步步教你編寫可測(cè)試的Go語(yǔ)言代碼

熱門標(biāo)簽:最短的地圖標(biāo)注 浙江人工智能外呼管理系統(tǒng) 電銷機(jī)器人可以補(bǔ)救房產(chǎn)中介嗎 百度地圖標(biāo)注搜索關(guān)鍵詞 成都呼叫中心外呼系統(tǒng)平臺(tái) 谷歌便利店地圖標(biāo)注 電梯外呼訪客系統(tǒng) 騰訊外呼系統(tǒng)價(jià)格 ?兓?

第一個(gè)測(cè)試 “Hello Test!”

首先,在我們$GOPATH/src目錄下創(chuàng)建hello目錄,作為本文涉及到的所有示例代碼的根目錄。

然后,新建名為hello.go的文件,定義一個(gè)函數(shù)hello() ,功能是返回一個(gè)由若干單詞拼接成句子:

package hello

func hello() string {
 words := []string{"hello", "func", "in", "package", "hello"}
 wl := len(words)

 sentence := ""
 for key, word := range words {
  sentence += word
  if key  wl-1 {
   sentence += " "
  } else {
   sentence += "."
  }
 }
 return sentence
}

接著,新建名為hello_test.go的文件,填入如下內(nèi)容:

package hello

import (
 "fmt"
 "testing"
)

func TestHello(t *testing.T) {
 got := hello()
 expect := "hello func in package hello."

 if got != expect {
  t.Errorf("got [%s] expected [%s]", got, expect)
 }
}

func BenchmarkHello(b *testing.B) {
 for i := 0; i  b.N; i++ {
  hello()
 }
}

func ExampleHello() {
 hl := hello()
 fmt.Println(hl)
 // Output: hello func in package hello.
}

最后,打開終端,進(jìn)入hello目錄,輸入go test命令并回車,可以看到如下輸出:

PASS
ok  hello 0.007s

編寫測(cè)試代碼

Golang的測(cè)試代碼位于某個(gè)包的源代碼中名稱以_test.go結(jié)尾的源文件里,測(cè)試代碼包含測(cè)試函數(shù)、測(cè)試輔助代碼和示例函數(shù);測(cè)試函數(shù)有以Test開頭的功能測(cè)試函數(shù)和以Benchmark開頭的性能測(cè)試函數(shù)兩種,測(cè)試輔助代碼是為測(cè)試函數(shù)服務(wù)的公共函數(shù)、初始化函數(shù)、測(cè)試數(shù)據(jù)等,示例函數(shù)則是以Example開頭的說(shuō)明被測(cè)試函數(shù)用法的函數(shù)。

大部分情況下,測(cè)試代碼是作為某個(gè)包的一部分,意味著它可以訪問(wèn)包中不可導(dǎo)出的元素。但在有需要的時(shí)候(如避免循環(huán)依賴)也可以修改測(cè)試文件的包名,如package hello的測(cè)試文件,包名可以設(shè)為package hello_test。

功能測(cè)試函數(shù)

功能測(cè)試函數(shù)需要接收*testing.T類型的單一參數(shù)t,testing.T 類型用來(lái)管理測(cè)試狀態(tài)和支持格式化的測(cè)試日志。測(cè)試日志在測(cè)試執(zhí)行過(guò)程中積累起來(lái),完成后輸出到標(biāo)準(zhǔn)錯(cuò)誤輸出。

下面是從Go標(biāo)準(zhǔn)庫(kù)摘抄的 testing.T類型的常用方法的用法:

測(cè)試函數(shù)中的某條測(cè)試用例執(zhí)行結(jié)果與預(yù)期不符時(shí),調(diào)用t.Error()t.Errorf()方法記錄日志并標(biāo)記測(cè)試失敗

# /usr/local/go/src/bytes/compare_test.go
func TestCompareIdenticalSlice(t *testing.T) {
 var b = []byte("Hello Gophers!")
 if Compare(b, b) != 0 {
  t.Error("b != b")
 }
 if Compare(b, b[:1]) != 1 {
  t.Error("b > b[:1] failed")
 }
}

使用t.Fatal()t.Fatalf()方法,在某條測(cè)試用例失敗后就跳出該測(cè)試函數(shù)

# /usr/local/go/src/bytes/reader_test.go
func TestReadAfterBigSeek(t *testing.T) {
 r := NewReader([]byte("0123456789"))
 if _, err := r.Seek(131+5, os.SEEK_SET); err != nil {
  t.Fatal(err)
 }
 if n, err := r.Read(make([]byte, 10)); n != 0 || err != io.EOF {
  t.Errorf("Read = %d, %v; want 0, EOF", n, err)
 }
}

使用t.Skip()t.Skipf()方法,跳過(guò)某條測(cè)試用例的執(zhí)行

# /usr/local/go/src/archive/zip/zip_test.go
func TestZip64(t *testing.T) {
 if testing.Short() {
  t.Skip("slow test; skipping")
 }
 const size = 1  32 // before the "END\n" part
 buf := testZip64(t, size)
 testZip64DirectoryRecordLength(buf, t)
}

執(zhí)行測(cè)試用例的過(guò)程中通過(guò)t.Log()t.Logf()記錄日志

# /usr/local/go/src/regexp/exec_test.go
func TestFowler(t *testing.T) {
 files, err := filepath.Glob("testdata/*.dat")
 if err != nil {
  t.Fatal(err)
 }
 for _, file := range files {
  t.Log(file)
  testFowler(t, file)
 }
}

使用t.Parallel()標(biāo)記需要并發(fā)執(zhí)行的測(cè)試函數(shù)

# /usr/local/go/src/runtime/stack_test.go
func TestStackGrowth(t *testing.T) {
 t.Parallel()
 var wg sync.WaitGroup

 // in a normal goroutine
 wg.Add(1)
 go func() {
  defer wg.Done()
  growStack()
 }()
 wg.Wait()

 // ...
}

性能測(cè)試函數(shù)

性能測(cè)試函數(shù)需要接收*testing.B類型的單一參數(shù)b,性能測(cè)試函數(shù)中需要循環(huán)b.N次調(diào)用被測(cè)函數(shù)。testing.B 類型用來(lái)管理測(cè)試時(shí)間和迭代運(yùn)行次數(shù),也支持和testing.T相同的方式管理測(cè)試狀態(tài)和格式化的測(cè)試日志,不一樣的是testing.B的日志總是會(huì)輸出。

下面是從Go標(biāo)準(zhǔn)庫(kù)摘抄的 testing.B類型的常用方法的用法:

在函數(shù)中調(diào)用t.ReportAllocs() ,啟用內(nèi)存使用分析

# /usr/local/go/src/bufio/bufio_test.go
func BenchmarkWriterFlush(b *testing.B) {
 b.ReportAllocs()
 bw := NewWriter(ioutil.Discard)
 str := strings.Repeat("x", 50)
 for i := 0; i  b.N; i++ {
  bw.WriteString(str)
  bw.Flush()
 }
}

通過(guò) b.StopTimer() 、b.ResetTimer() b.StartTimer()來(lái)停止、重置、啟動(dòng) 時(shí)間經(jīng)過(guò)和內(nèi)存分配計(jì)數(shù)

# /usr/local/go/src/fmt/scan_test.go
func BenchmarkScanInts(b *testing.B) {
 b.ResetTimer()
 ints := makeInts(intCount)
 var r RecursiveInt
 for i := b.N - 1; i >= 0; i-- {
  buf := bytes.NewBuffer(ints)
  b.StartTimer()
  scanInts(r, buf)
  b.StopTimer()
 }
}

調(diào)用b.SetBytes()記錄在一個(gè)操作中處理的字節(jié)數(shù)

# /usr/local/go/src/testing/benchmark.go
func BenchmarkFields(b *testing.B) {
 b.SetBytes(int64(len(fieldsInput)))
 for i := 0; i  b.N; i++ {
  Fields(fieldsInput)
 }
}

通過(guò)b.RunParallel()方法和 *testing.PB類型的Next()方法來(lái)并發(fā)執(zhí)行被測(cè)對(duì)象

# /usr/local/go/src/sync/atomic/value_test.go
func BenchmarkValueRead(b *testing.B) {
 var v Value
 v.Store(new(int))
 b.RunParallel(func(pb *testing.PB) {
  for pb.Next() {
   x := v.Load().(*int)
   if *x != 0 {
    b.Fatalf("wrong value: got %v, want 0", *x)
   }
  }
 })
}

測(cè)試輔助代碼

測(cè)試輔助代碼是編寫測(cè)試代碼過(guò)程中因代碼重用和代碼質(zhì)量考慮而產(chǎn)生的。主要包括如下方面:

引入依賴的外部包,如每個(gè)測(cè)試文件都需要的 testing 包等:

# /usr/local/go/src/log/log_test.go:
import (
 "bytes"
 "fmt"
 "os"
 "regexp"
 "strings"
 "testing"
 "time"
)

定義多次用到的常量和變量,測(cè)試用例數(shù)據(jù)等:

# /usr/local/go/src/log/log_test.go:
const (
 Rdate   = `[0-9][0-9][0-9][0-9]/[0-9][0-9]/[0-9][0-9]`
 Rtime   = `[0-9][0-9]:[0-9][0-9]:[0-9][0-9]`
 Rmicroseconds = `\.[0-9][0-9][0-9][0-9][0-9][0-9]`
 Rline   = `(57|59):` // must update if the calls to l.Printf / l.Print below move
 Rlongfile  = `.*/[A-Za-z0-9_\-]+\.go:` + Rline
 Rshortfile = `[A-Za-z0-9_\-]+\.go:` + Rline
)

// ...

var tests = []tester{
 // individual pieces:
 {0, "", ""},
 {0, "XXX", "XXX"},
 {Ldate, "", Rdate + " "},
 {Ltime, "", Rtime + " "},
 {Ltime | Lmicroseconds, "", Rtime + Rmicroseconds + " "},
 {Lmicroseconds, "", Rtime + Rmicroseconds + " "}, // microsec implies time
 {Llongfile, "", Rlongfile + " "},
 {Lshortfile, "", Rshortfile + " "},
 {Llongfile | Lshortfile, "", Rshortfile + " "}, // shortfile overrides longfile
 // everything at once:
 {Ldate | Ltime | Lmicroseconds | Llongfile, "XXX", "XXX" + Rdate + " " + Rtime + Rmicroseconds + " " + Rlongfile + " "},
 {Ldate | Ltime | Lmicroseconds | Lshortfile, "XXX", "XXX" + Rdate + " " + Rtime + Rmicroseconds + " " + Rshortfile + " "},
}

和普通的Golang源代碼一樣,測(cè)試代碼中也能定義init函數(shù),init函數(shù)會(huì)在引入外部包、定義常量、聲明變量之后被自動(dòng)調(diào)用,可以在init函數(shù)里編寫測(cè)試相關(guān)的初始化代碼。

# /usr/local/go/src/bytes/buffer_test.go
func init() {
 testBytes = make([]byte, N)
 for i := 0; i  N; i++ {
  testBytes[i] = 'a' + byte(i%26)
 }
 data = string(testBytes)
}

封裝測(cè)試專用的公共函數(shù),抽象測(cè)試專用的結(jié)構(gòu)體等:

# /usr/local/go/src/log/log_test.go:
type tester struct {
 flag int
 prefix string
 pattern string // regexp that log output must match; we add ^ and expected_text$ always
}

// ...

func testPrint(t *testing.T, flag int, prefix string, pattern string, useFormat bool) {
 // ...
}

示例函數(shù)

示例函數(shù)無(wú)需接收參數(shù),但需要使用注釋的 Output: 標(biāo)記說(shuō)明示例函數(shù)的輸出值,未指定Output:標(biāo)記或輸出值為空的示例函數(shù)不會(huì)被執(zhí)行。

示例函數(shù)需要?dú)w屬于某個(gè) 包/函數(shù)/類型/類型 的方法,具體命名規(guī)則如下:

func Example() { ... }  # 包的示例函數(shù)
func ExampleF() { ... }  # 函數(shù)F的示例函數(shù)
func ExampleT() { ... }  # 類型T的示例函數(shù)
func ExampleT_M() { ... } # 類型T的M方法的示例函數(shù)

# 多示例函數(shù) 需要跟下劃線加小寫字母開頭的后綴
func Example_suffix() { ... }
func ExampleF_suffix() { ... }
func ExampleT_suffix() { ... }
func ExampleT_M_suffix() { ... }

go doc 工具會(huì)解析示例函數(shù)的函數(shù)體作為對(duì)應(yīng) 包/函數(shù)/類型/類型的方法 的用法。

測(cè)試函數(shù)的相關(guān)說(shuō)明,可以通過(guò)go help testfunc來(lái)查看幫助文檔。

使用 go test 工具

Golang中通過(guò)命令行工具go test來(lái)執(zhí)行測(cè)試代碼,打開shell終端,進(jìn)入需要測(cè)試的包所在的目錄執(zhí)行 go test,或者直接執(zhí)行go test $pkg_name_in_gopath即可對(duì)指定的包執(zhí)行測(cè)試。

通過(guò)形如go test github.com/tabalt/...的命令可以執(zhí)行$GOPATH/github.com/tabalt/目錄下所有的項(xiàng)目的測(cè)試。go test std命令則可以執(zhí)行Golang標(biāo)準(zhǔn)庫(kù)的所有測(cè)試。

如果想查看執(zhí)行了哪些測(cè)試函數(shù)及函數(shù)的執(zhí)行結(jié)果,可以使用-v參數(shù):

[tabalt@localhost hello] go test -v
=== RUN TestHello
--- PASS: TestHello (0.00s)
=== RUN ExampleHello
--- PASS: ExampleHello (0.00s)
PASS
ok  hello 0.006s

假設(shè)我們有很多功能測(cè)試函數(shù),但某次測(cè)試只想執(zhí)行其中的某一些,可以通過(guò)-run參數(shù),使用正則表達(dá)式來(lái)匹配要執(zhí)行的功能測(cè)試函數(shù)名。如下面指定參數(shù)后,功能測(cè)試函數(shù)TestHello不會(huì)執(zhí)行到。

[tabalt@localhost hello] go test -v -run=xxx
PASS
ok  hello 0.006s

性能測(cè)試函數(shù)默認(rèn)并不會(huì)執(zhí)行,需要添加-bench參數(shù),并指定匹配性能測(cè)試函數(shù)名的正則表達(dá)式;例如,想要執(zhí)行某個(gè)包中所有的性能測(cè)試函數(shù)可以添加參數(shù)-bench . 或 -bench=.。

[tabalt@localhost hello] go test -bench=.
PASS
BenchmarkHello-8  2000000   657 ns/op
ok  hello 1.993s

想要查看性能測(cè)試時(shí)的內(nèi)存情況,可以再添加參數(shù)-benchmem:

[tabalt@localhost hello] go test -bench=. -benchmem
PASS
BenchmarkHello-8  2000000   666 ns/op   208 B/op   9 allocs/op
ok  hello 2.014s

參數(shù)-cover可以用來(lái)查看我們編寫的測(cè)試對(duì)代碼的覆蓋率:


詳細(xì)的覆蓋率信息,可以通過(guò)-coverprofile輸出到文件,并使用go tool cover來(lái)查看,用法請(qǐng)參考go tool cover -help 。

更多go test命令的參數(shù)及用法,可以通過(guò)go help testflag來(lái)查看幫助文檔。

高級(jí)測(cè)試技術(shù)

IO相關(guān)測(cè)試

testing/iotest包中實(shí)現(xiàn)了常用的出錯(cuò)的Reader和Writer,可供我們?cè)趇o相關(guān)的測(cè)試中使用。主要有:

觸發(fā)數(shù)據(jù)錯(cuò)誤dataErrReader,通過(guò)DataErrReader()函數(shù)創(chuàng)建

讀取一半內(nèi)容的halfReader,通過(guò)HalfReader()函數(shù)創(chuàng)建

讀取一個(gè)byte的oneByteReader,通過(guò)OneByteReader()函數(shù)創(chuàng)建

觸發(fā)超時(shí)錯(cuò)誤的timeoutReader,通過(guò)TimeoutReader()函數(shù)創(chuàng)建

寫入指定位數(shù)內(nèi)容后停止的truncateWriter,通過(guò)TruncateWriter()函數(shù)創(chuàng)建

讀取時(shí)記錄日志的readLogger,通過(guò)NewReadLogger()函數(shù)創(chuàng)建

寫入時(shí)記錄日志的writeLogger,通過(guò)NewWriteLogger()函數(shù)創(chuàng)建

黑盒測(cè)試

testing/quick包實(shí)現(xiàn)了幫助黑盒測(cè)試的實(shí)用函數(shù) Check和CheckEqual。

Check函數(shù)的第1個(gè)參數(shù)是要測(cè)試的只返回bool值的黑盒函數(shù)f,Check會(huì)為f的每個(gè)參數(shù)設(shè)置任意值并多次調(diào)用,如果f返回false,Check函數(shù)會(huì)返回錯(cuò)誤值 *CheckError。Check函數(shù)的第2個(gè)參數(shù) 可以指定一個(gè)quick.Config類型的config,傳nil則會(huì)默認(rèn)使用quick.defaultConfig。quick.Config結(jié)構(gòu)體包含了測(cè)試運(yùn)行的選項(xiàng)。

# /usr/local/go/src/math/big/int_test.go
func checkMul(a, b []byte) bool {
 var x, y, z1 Int
 x.SetBytes(a)
 y.SetBytes(b)
 z1.Mul(x, y)

 var z2 Int
 z2.SetBytes(mulBytes(a, b))

 return z1.Cmp(z2) == 0
}

func TestMul(t *testing.T) {
 if err := quick.Check(checkMul, nil); err != nil {
  t.Error(err)
 }
}

CheckEqual函數(shù)是比較給定的兩個(gè)黑盒函數(shù)是否相等,函數(shù)原型如下:

func CheckEqual(f, g interface{}, config *Config) (err error)

HTTP測(cè)試

net/http/httptest包提供了HTTP相關(guān)代碼的工具,我們的測(cè)試代碼中可以創(chuàng)建一個(gè)臨時(shí)的httptest.Server來(lái)測(cè)試發(fā)送HTTP請(qǐng)求的代碼:

ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 fmt.Fprintln(w, "Hello, client")
}))
defer ts.Close()

res, err := http.Get(ts.URL)
if err != nil {
 log.Fatal(err)
}

greeting, err := ioutil.ReadAll(res.Body)
res.Body.Close()
if err != nil {
 log.Fatal(err)
}

fmt.Printf("%s", greeting)

還可以創(chuàng)建一個(gè)應(yīng)答的記錄器httptest.ResponseRecorder來(lái)檢測(cè)應(yīng)答的內(nèi)容:

handler := func(w http.ResponseWriter, r *http.Request) {
 http.Error(w, "something failed", http.StatusInternalServerError)
}

req, err := http.NewRequest("GET", "http://example.com/foo", nil)
if err != nil {
 log.Fatal(err)
}

w := httptest.NewRecorder()
handler(w, req)

fmt.Printf("%d - %s", w.Code, w.Body.String())

測(cè)試進(jìn)程操作行為

當(dāng)我們被測(cè)函數(shù)有操作進(jìn)程的行為,可以將被測(cè)程序作為一個(gè)子進(jìn)程執(zhí)行測(cè)試。下面是一個(gè)例子:

//被測(cè)試的進(jìn)程退出函數(shù)
func Crasher() {
 fmt.Println("Going down in flames!")
 os.Exit(1)
}

//測(cè)試進(jìn)程退出函數(shù)的測(cè)試函數(shù)
func TestCrasher(t *testing.T) {
 if os.Getenv("BE_CRASHER") == "1" {
  Crasher()
  return
 }
 cmd := exec.Command(os.Args[0], "-test.run=TestCrasher")
 cmd.Env = append(os.Environ(), "BE_CRASHER=1")
 err := cmd.Run()
 if e, ok := err.(*exec.ExitError); ok  !e.Success() {
  return
 }
 t.Fatalf("process ran with err %v, want exit status 1", err)
}

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家學(xué)習(xí)或者使用Go語(yǔ)言能有所幫助,如果有疑問(wèn)大家可以留言交流。

您可能感興趣的文章:
  • 深入理解Golang的單元測(cè)試和性能測(cè)試

標(biāo)簽:紹興 宜昌 盤錦 邢臺(tái) 上海 雅安 眉山 七臺(tái)河

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《一步步教你編寫可測(cè)試的Go語(yǔ)言代碼》,本文關(guān)鍵詞  一,步步,教你,編寫,可,測(cè)試,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《一步步教你編寫可測(cè)試的Go語(yǔ)言代碼》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于一步步教你編寫可測(cè)試的Go語(yǔ)言代碼的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    成人福利在线看| 成人av电影在线观看| 免费成人在线观看| 日韩欧美一区二区免费| 美女在线视频一区| 精品剧情在线观看| 99国产精品久| 日韩av电影免费观看高清完整版| 欧美一区二区三区免费视频| 久久9热精品视频| 久久精品人人做人人爽人人| av电影在线观看一区| 日一区二区三区| 国产欧美日韩另类视频免费观看 | 亚洲色图在线播放| 欧美日韩国产美| 成人动漫中文字幕| 亚洲成a人v欧美综合天堂 | 中文字幕av一区二区三区高| 色噜噜狠狠成人网p站| 日本怡春院一区二区| 亚洲色图一区二区| 国产亚洲一区二区三区四区| 色综合久久66| 国产成人一级电影| 日韩精品亚洲一区| 中文字幕字幕中文在线中不卡视频| 在线不卡免费欧美| 91久久精品一区二区二区| 国产一区二区在线看| 石原莉奈在线亚洲二区| 亚洲天堂av老司机| 国产日韩三级在线| 91精品国产福利在线观看| 色播五月激情综合网| 成人av网在线| 成人美女视频在线观看| 麻豆传媒一区二区三区| 樱花草国产18久久久久| 国产精品白丝在线| 综合在线观看色| 国产精品国产馆在线真实露脸| 日韩欧美国产一二三区| 7777精品伊人久久久大香线蕉经典版下载 | 欧美乱熟臀69xxxxxx| 97国产一区二区| av电影在线观看完整版一区二区| 国产精品一卡二卡在线观看| 精品制服美女丁香| 免费观看一级欧美片| 蜜臀av性久久久久av蜜臀妖精| 调教+趴+乳夹+国产+精品| 丝袜美腿亚洲一区二区图片| 亚洲妇女屁股眼交7| 丝袜亚洲另类欧美综合| 毛片av一区二区| 国产米奇在线777精品观看| 国产在线精品免费av| 国产在线精品一区二区夜色 | 精品国产乱码91久久久久久网站| 欧美一区二区三区日韩视频| 91精品国产欧美一区二区成人| 在线不卡免费av| 91精品国产入口在线| 久久婷婷综合激情| 亚洲国产精品ⅴa在线观看| 国产精品第一页第二页第三页| 亚洲精品成人天堂一二三| 日韩av中文字幕一区二区三区 | 欧美一区二区久久久| 精品国产电影一区二区| 国产精品私人影院| 亚洲精品成人在线| 毛片一区二区三区| 成人福利电影精品一区二区在线观看| 91网页版在线| 欧美一级理论片| 国产日韩欧美制服另类| 亚洲自拍与偷拍| 国产一区二区三区在线观看精品 | 亚洲一区二区视频| 精品在线你懂的| 91亚洲国产成人精品一区二区三| 欧美色精品在线视频| 精品黑人一区二区三区久久| 中文字幕中文在线不卡住| 亚洲电影一区二区| 国内精品免费**视频| 欧美在线一区二区三区| 精品久久久久久久久久久院品网| 国产精品久久久久久久久免费樱桃| 亚洲综合一区二区三区| 国产毛片精品视频| 欧美日韩一区二区三区免费看| 欧美电影免费观看高清完整版 | 亚洲一区二区四区蜜桃| 激情丁香综合五月| 欧美肥胖老妇做爰| 亚洲丝袜另类动漫二区| 美腿丝袜一区二区三区| 91一区二区在线| 日本一区二区三区dvd视频在线| 亚洲成人777| 99精品一区二区三区| 久久久久久9999| 亚洲成人动漫av| 99久久99精品久久久久久| 国产亚洲一区二区三区四区| 青青草97国产精品免费观看无弹窗版| 成人久久久精品乱码一区二区三区| 欧美精品在线视频| 亚洲精品成人天堂一二三| 成人综合在线观看| 久久欧美中文字幕| 精品影视av免费| 91精品国产免费久久综合| 一区二区三区在线免费视频| 成人av网站在线观看免费| 国产日韩欧美激情| 国产精品一区不卡| 欧美一区二区黄色| 久久久不卡影院| 国产一区不卡精品| 国产一区二区三区观看| 欧美人与性动xxxx| 亚洲成人自拍一区| 日本精品一级二级| 一区二区三区日韩| 在线免费不卡电影| 性欧美疯狂xxxxbbbb| 在线免费观看日韩欧美| 亚洲一区二区3| 欧美日韩精品欧美日韩精品| 亚洲成人自拍一区| 91精品国产麻豆国产自产在线 | 日韩国产一二三区| 欧美一级搡bbbb搡bbbb| 蜜桃传媒麻豆第一区在线观看| 欧美日韩1234| 老司机午夜精品99久久| 久久久天堂av| 成人一区二区三区| 亚洲色图一区二区三区| 欧美色欧美亚洲另类二区| 亚洲欧洲无码一区二区三区| 99久精品国产| 日韩电影在线观看电影| 欧美大片一区二区三区| 成人性生交大片免费看中文网站| 国产精品国产自产拍在线| 色8久久人人97超碰香蕉987| 天天影视色香欲综合网老头| 亚洲精品在线免费播放| 99re这里都是精品| 日本不卡在线视频| 久久免费偷拍视频| 色猫猫国产区一区二在线视频| 亚洲伊人色欲综合网| 日韩欧美在线不卡| 99久久精品国产毛片| 日本美女一区二区三区| 欧美激情一区不卡| 欧美电影影音先锋| 成人免费视频视频在线观看免费| 亚洲最大成人综合| 久久久精品黄色| 在线不卡欧美精品一区二区三区| 国产成a人亚洲| 视频一区在线播放| 综合网在线视频| 精品国产凹凸成av人网站| 在线看日韩精品电影| 国产麻豆91精品| 日韩黄色在线观看| 一区二区三区在线视频观看58 | 九九在线精品视频| 亚洲精品水蜜桃| 久久天堂av综合合色蜜桃网| 欧美丝袜自拍制服另类| 国产69精品久久99不卡| 美腿丝袜亚洲色图| 首页国产欧美日韩丝袜| 亚洲精品成人悠悠色影视| 欧美国产欧美综合| 国产午夜精品一区二区三区视频 | 欧美日本国产视频| 欧美亚洲自拍偷拍| 色婷婷亚洲精品| 成年人网站91| 国产黄人亚洲片| 精品一区二区成人精品| 另类调教123区| 日日摸夜夜添夜夜添国产精品| 亚洲精品网站在线观看| 中文字幕一区二区三区四区| 日韩欧美一区二区久久婷婷| 91精品国产一区二区人妖| 欧美群妇大交群的观看方式 | 91久久免费观看| 色拍拍在线精品视频8848|