3atv精品不卡视频,97人人超碰国产精品最新,中文字幕av一区二区三区人妻少妇,久久久精品波多野结衣,日韩一区二区三区精品

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

《Go语言实战》William Kennedy中文版学习笔记

發(fā)布時間:2023/12/10 编程问答 29 豆豆
生活随笔 收集整理的這篇文章主要介紹了 《Go语言实战》William Kennedy中文版学习笔记 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

第1章 Go語言的介紹

本章主要內(nèi)容:用Go解決現(xiàn)代計算難題,使用 Go 語言工具。C 和 C++這類語言提供了很快的執(zhí)行速度,而 Ruby 和 Python 這類語言則擅長快速開發(fā)。Go 語言在這兩者間架起了橋梁,不僅提供了高性能的語言,同時也讓開發(fā)更快速。

1.1.1 開發(fā)速度

Go 語言使用了更加智能的編譯器,并簡化了解決依賴的算法,最終提供了更快的編譯速度。編譯 Go 程序時,編譯器只會關(guān)注那些直接被引用的庫,而不是像 Java、C 和 C++那樣,要遍歷依賴鏈中所有依賴的庫。

1.1.2 并發(fā)

Go 語言對并發(fā)的支持是這門語言最重要的特性之一。goroutine 很像線程,但是它占用的內(nèi)存遠(yuǎn)少于線程,使用它需要的代碼更少。通道(channel)是一種內(nèi)置的數(shù)據(jù)結(jié)構(gòu),可以讓用戶在不同的 goroutine 之間同步發(fā)送具有類型的消息。這讓編程模型更傾向于在 goroutine之間發(fā)送消息,而不是讓多個 goroutine 爭奪同一個數(shù)據(jù)的使用權(quán)。

1.goroutine

圖 1-2 在單一系統(tǒng)線程上執(zhí)行多個 goroutine

goroutine 是可以與其他 goroutine 并行執(zhí)行的函數(shù),同時也會與主程序(程序的入口)并執(zhí)行。在其他編程語言中,你需要用線程來完成同樣的事情,而在 Go 語言中會使用同一個線程來執(zhí)行多個 goroutine。如果想在執(zhí)行一段代碼的同時,并行去做另外一些事情,goroutine 是很好的選擇。下面是一個簡單的例子:

func log(msg string) {
? ? //這里是一些記錄日志的代碼
}
//代碼里有些地方檢測到了錯誤
go log("發(fā)生了可怕的事情")

2.通道

圖 1-3 使用通道在 goroutine 之間安全地發(fā)送數(shù)據(jù)

通道是一種數(shù)據(jù)結(jié)構(gòu),可以讓goroutine之間進(jìn)行安全的數(shù)據(jù)通信。通道可以幫助用戶避免其他語言里常見的共享內(nèi)存訪問的問題。通道這一模式保證同一時刻只會有一個goroutine修改數(shù)據(jù)。通道用于幾個運行的goroutine之間發(fā)送數(shù)據(jù)。

1.1.3 Go 語言的類型系統(tǒng)

Go 語言提供了靈活的、無繼承的類型系統(tǒng),Go 開發(fā)者使用組合(composition)設(shè)計模式,只需簡單地將一個類型嵌入到另一個類型,就能復(fù)用所有的功能。

1.類型簡單

Go 開發(fā)者構(gòu)建更小的類型——Customer 和 Admin,然后把這些小類型組合成更大的類型。圖 1-4
展示了繼承和組合之間的不同。

2.Go 接口對一組行為建模

圖 1-4 繼承和組合的對比

Go 語言的接口一般只會描述一個單一的動作。在 Go 語言中,最常使用的接口之一是 io.Reader 。這個接口提供了一個簡單的方法,來聲明一個類型有數(shù)據(jù)可以讀取。標(biāo)準(zhǔn)庫內(nèi)的其他函數(shù)都能理解這個接口。這個接口的定義如下:

type Reader interface {
? ? Read(p []byte) (n int, err error)
}

為了實現(xiàn) io.Reader 這個接口,你只需要實現(xiàn)一個 Read 方法,這個方法接受一個 byte切片,返回一個整數(shù)和可能出現(xiàn)的錯誤。

1.1.4 內(nèi)存管理

Go 語言把無趣的內(nèi)存管理交給專業(yè)的編譯器去做,而讓程序員專注于更有趣的事情。

1.2 你好,Go

用Go語言編寫經(jīng)典的Hello World!應(yīng)用程序:

package main

import "fmt"

func main() {

? ? ? fmt.Printf("%s\n","Hello World!");

}

1.3 小結(jié)

Go語言是現(xiàn)代的、快速的,帶有一個強大的標(biāo)準(zhǔn)庫;Go語言內(nèi)置對并發(fā)的支持;Go語言使用接口作為代碼復(fù)用的基礎(chǔ)模塊。

第2章 快速開始一個Go程序

本章主要內(nèi)容:學(xué)習(xí)如何寫一個復(fù)雜的 Go 程序;聲明類型、變量、函數(shù)和方法;啟動并同步操作 goroutine;使用接口寫通用的代碼;處理程序邏輯和錯誤。

通過一個完整的 Go 語言程序,來看看 Go 語言是如何實現(xiàn)一些功能的。這個程序從不同的數(shù)據(jù)源拉取數(shù)據(jù),將數(shù)據(jù)內(nèi)容與一組搜索項做對比,然后將匹配的內(nèi)容顯示在終端窗口。這個程序會讀取文本文件,進(jìn)行網(wǎng)絡(luò)調(diào)用,解碼 XML 和 JSON 成為結(jié)構(gòu)化類型數(shù)據(jù),并且利用 Go 語言的并發(fā)機制保證這些操作的速度。代碼存放在這個代碼庫:
https://github.com/goinaction/code/tree/master/chapter2/sample

2.1 程序架

圖 2-1 程序架構(gòu)流程圖

這個應(yīng)用的代碼使用了 4 個文件夾,按字母順序列出。文件夾 data 中有一個 JSON 文檔,其內(nèi)容是程序要拉取和處理的數(shù)據(jù)源。文件夾 matchers 中包含程序里用于支持搜索不同數(shù)據(jù)源的代碼。目前程序只完成了支持處理 RSS 類型的數(shù)據(jù)源的匹配器。文件夾 search 中包含使用不同匹配器進(jìn)行搜索的業(yè)務(wù)邏輯,default.go用于搜索數(shù)據(jù)用的默認(rèn)匹配器,feed.go用于讀取 json 數(shù)據(jù)文件,
match.go用于支持不同匹配器的接口,search.go執(zhí)行搜索的主控制邏輯。最后,父級文件夾 sample 中有個 main.go 文件,這是整個程序的入口。

2.2 main 包

main.go文件,每個可執(zhí)行的 Go 程序都有兩個明顯的特征。一個特征是第 18 行聲明的名為 main 的函數(shù)。
構(gòu)建程序在構(gòu)建可執(zhí)行文件時,需要找到這個已經(jīng)聲明的 main 函數(shù),把它作為程序的入口。第二個特征是程序的第 01 行的包名 main 。

package main

import (
? ? "log"
? ? "os"
? ??
? ? _ "github.com/goinaction/code/chapter2/sample/matchers"
? ? ?"github.com/goinaction/code/chapter2/sample/search"
)

//init在main之前調(diào)用
func init() {
? ? //將日志輸出到標(biāo)準(zhǔn)輸出
? ? log.SetOutput(os.Stdout)
}

//main是整個程序的入口
func main() {
? ? //使用特定的項做搜索
? ? search.Run("president")
}

Go 語言的每個代碼文件都屬于一個包,main.go 也不例外。包這個特性對于 Go 語言來說很重要,一個包定義一組編譯過的代碼,包的名字類似命名空間,可以用來間接訪問包內(nèi)聲明的標(biāo)識符。這個特性可以把不同包中定義的同名標(biāo)識符區(qū)別開。

2.3 search 包

這個程序使用的框架和業(yè)務(wù)邏輯都在 search 包里。由于整個程序都圍繞匹配器來運作,這個程序里的匹配器,是指包含特定信息、用于處理某類數(shù)據(jù)源的實例。在這個示例程序中有兩個匹配器。框架本身實現(xiàn)了一個無法獲取任何信息的默認(rèn)匹配器,而在 matchers 包里實現(xiàn)了 RSS 匹配器。RSS匹配器知道如何獲取、讀入并查找 RSS 數(shù)據(jù)源。

2.3.1 search.go

package search

import (
? ? ? ? "log"
? ? ? ? "sync"
)

//注冊用于搜索的匹配器的映射
var matchers = make(map[string]Matcher)

在 Go 語言里,標(biāo)識符要么從包里公開,要么不從包里公開。當(dāng)代碼導(dǎo)入了一個包時,程序可以直接訪問這個包中任意一個公開的標(biāo)識符。這些標(biāo)識符以大寫字母開頭。以小寫字母開頭的標(biāo)識符是不公開的,不能被其他包中的代碼直接訪問。但是,其他包可以間接訪問不公開的標(biāo)識符。例如,一個函數(shù)可以返回一個未公開類型的值,那么這個函數(shù)的任何調(diào)用者,哪怕調(diào)用者不是在這個包里聲明的,都可以訪問這個值。

//Run執(zhí)行搜索邏輯
func Run(searchTherm string) {
? ? //獲取需要搜索的數(shù)據(jù)源列表

? ? //?第一個返回值是一組 Feed 類型的切片。切片是一種實現(xiàn)了一個動態(tài)數(shù)組的引用類型。第二個返回值是一個錯誤值。
? ? feed, err := RetrieveFeeds()
? ? if err != nil {
? ? ? ? ? ? log.Fatal(err)
? ? }
? ??
? ? //創(chuàng)建一個無緩沖的通道,接收匹配后的結(jié)果
? ? results := make(chan *Result)
? ??
? ? //構(gòu)造一個waitGroup,以便處理所有的數(shù)據(jù)源
? ? var waitGroup sync.waitGroup
? ??
? ? //設(shè)置需要等待處理
? ? //每個數(shù)據(jù)源的goroutine數(shù)量
? ? waitGroup.add(len(feeds))
? ??
? ? //為每一個數(shù)據(jù)源啟動一個gorpuntine來查找結(jié)果
? ? for _, feed := range feeds {
? ? ? ? ? ? //獲取一個匹配器用于查找
? ? ? ? ? ? matcher, exits := matchers[feed.Type]
? ? ? ? ? ? if !exits {
? ? ? ? ? ? ? ? ? ? matcher = matchers["default"]
? ? ? ? ? ? }
? ? ? ? ? ??
? ? ? ? ? ? //啟動一個gorountine來執(zhí)行搜索
? ? ? ? ? ? go func(matcher Matcher, feed *Feed) {
? ? ? ? ? ? ? ? ? ? Match(matcher, feed, searchTerm, results)
? ? ? ? ? ? ? ? ? ? waitGroup.done()
? ? ? ? ? ? }(matcher, feed)
? ? }
? ??
? ? //啟動一個gorountine來監(jiān)控是否所有的工作都做完了
? ? go func() {
? ? ? ? ? ? //等待所有任務(wù)完成
? ? ? ? ? ? waitGroup.wait()
? ? ? ? ? ??
? ? ? ? ? ? //用關(guān)閉通道的方式,通知Display函數(shù)
? ? ? ? ? ? //可以退出程序了
? ? ? ? ? ? close(results)
? ? }()
? ??
? ? //啟動函數(shù),顯示返回的結(jié)果,并且
? ? //在最后一個結(jié)果顯示完成后返回
? ? Display(results)
}

//Register調(diào)用時,會注冊一個匹配器,提供給后面的程序使用
func Register(feedType string, matcher Matcher) {
? ? if _, exists := matchers[feedType]; exists {
? ? ? ? log.Fatalln(feedType, "Matcher already registered")
? ? }
? ??
? ? log.Println("Register", feedType, "matcher")
? ? matcher[feedType] = matcher
}

在 Go 語言中,通道(channel)和映射(map)與切片(slice)一樣,也是引用類型,不過通道本身實現(xiàn)的是一組帶類型的值,這組值用于在 goroutine 之間傳遞數(shù)據(jù)。通道內(nèi)置同步機制,從而保證通信安全。

這個程序使用 sync 包的 WaitGroup 跟蹤所有啟動的 goroutine。WaitGroup 是一個計數(shù)信號量,我們可以利用它來統(tǒng)計所有的
goroutine 是不是都完成了工作。

我們使用關(guān)鍵字 for range 對 feeds 切片做迭代。關(guān)鍵字 range 可以用于迭代數(shù)組、字符串、切片、映射和通道。使用 for range 迭代切片時,每次迭代會返回兩個值。第一個值是迭代的元素在切片里的索引位置,第二個值是元素值的一個副本。

Go 語言支持閉包,在匿名函數(shù)內(nèi)訪問 searchTerm 和 results變量,也是通過閉包的形式訪問的。因為有了閉包,函數(shù)可以直接訪問到那些沒有作為參數(shù)傳入的變量。匿名函數(shù)并沒有拿到這些變量的副本,而是直接訪問外層函數(shù)作用域中聲明的這些變量本身。

我們以 goroutine的方式啟動了另一個匿名函數(shù)。這個匿名函數(shù)沒有輸入?yún)?shù),使用閉包訪問了 WaitGroup 和results 變量。這個 goroutine 里面調(diào)用了 WaitGroup 的 Wait 方法。這個方法會導(dǎo)致 goroutine阻塞,直到 WaitGroup 內(nèi)部的計數(shù)到達(dá) 0。之后,goroutine 調(diào)用了內(nèi)置的 close 函數(shù),關(guān)閉了通道,最終導(dǎo)致程序終止。

2.3.2 feed.go

package search

import (
? ? "encoding/json"
? ? "os"
)

const dataFile = "data/data.json"

//Feed包含我們需要處理的數(shù)據(jù)源的信息
type Feed struct {
? ? Name string 'json:"site"'
? ? URI string 'json:"link"'
? ? type string 'json:"type"'
}

//RetrieveFeeds讀取并反序列化源數(shù)據(jù)文件
func RetrieveFeeds() ([]*Feed, error) {
? ? //打開文件
? ? file, err := os.Open(dataFile)
? ? if err != nil {
? ? ? ? return nil, err
? ? }
? ??
? ? //當(dāng)函數(shù)返回時
? ? //關(guān)閉文件
? ? defer file.Close()
? ??
? ? //將文件解碼到一個切片里
? ? //這個切片的每一項是一個指向一個Feed類型值的指針
? ? var feeds []*feed
? ? err = json.NewDecoder(file).Decode(&feedss)
? ??
? ? //這個函數(shù)不需要檢查錯誤,調(diào)用者會做這件事
? ? return feeds, err
}

2.3.3 match.go/default.go

search/match.go

package search

import (
? ? "log"
)

//Result 保存搜索的結(jié)果
type Result strut {
? ? Field string
? ? Content string
}

//Matcher定義要實現(xiàn)的
//新搜索類型的行為
type Matcher interface {
? ? Search(feed *Feed, searchTerm string ) ([]*Result, error)
}

//Match函數(shù),為每個數(shù)據(jù)源單獨啟動goroutine來執(zhí)行這個函數(shù)
// 并發(fā)地執(zhí)行搜索
func Match(matcher Matcher, feed *Feed, searchTerm string, result chan<- *Result) {
? ? //對特定的疲累器執(zhí)行搜索
? ? searchTerm, err := matcher.Search(feed, searchTerm)
? ? if err != nil {
? ? ? ? log.Println(err)
? ? ? ? return
? ? }
? ??
? ? //將結(jié)果寫入通道
? ? for _, result := range searchTerm {
? ? ? ? result <- result
? ? }
}

//Display從每個單獨的gorountine接收到結(jié)果后
//在終端窗口輸出
func Dissplay(results chan *Resssult) {
? ? //通道會一直阻塞,直到有結(jié)果寫入
? ? //一旦通道被關(guān)閉,for循環(huán)就會終止
? ? for result := range results {
? ? ? ? fmt.Printf("%s:\n%s\n\n", result.Field, result.Content)
? ? }
}

interface 關(guān)鍵字聲明了一個接口,這個接口聲明了結(jié)構(gòu)類型或者具名類型需要實現(xiàn)的行為。一個接口的行為最終由在這個接口類型中聲明的方法決定。如果要讓一個用戶定義的類型實現(xiàn)一個接口,這個用戶定義的類型要實現(xiàn)接口類型里聲明的所有方法。

search/default.go

package search

//defaultMatcer實現(xiàn)了默認(rèn)匹配器
type defaultMatcher struct{}

// init函數(shù)將默認(rèn)匹配器注冊到程序里
func init() {
? ? var matcher defaultMatcher
? ? Register("default", matcher)
}

//Search 實現(xiàn)了默認(rèn)匹配器的行為
func (m defaultMatcher) Search(feed *Feed, searchTerm string) ([]*Result, err) {
? ? return nil, nil
}

如果聲明函數(shù)的時候帶有接收者,則意味著聲明了一個方法。這個方法會和指定的接收者的類型綁在一起。在我們的例子里, Search 方法與 defaultMatcher 類型的值綁在一起。這意味著我們可以使用 defaultMatcher 類型的值或者指向這個類型值的指針來調(diào)用 Search 方法。無論我們是使用接收者類型的值來調(diào)用這個方,還是使用接收者類型值的指針來調(diào)用這個方法,編譯器都會正確地引用或者解引用對應(yīng)的值,作為接收者傳遞給 Search 方法。

調(diào)用方法的例子

//方法聲明為使用defaultMatcher類型的值作為接收者
func (m defaultMatcher) Search(feed *Feed, searchTerm string)

//聲明一個指向defaultMatcher類型值的指針
dm := new(defaultMatch)

//編譯器會解開dm指針的引用,使用對應(yīng)的值調(diào)用方法
dm.Search(feed, "test"

//方法聲明為使用指向defaultMatcher類型值的指針作為接收者
func (m ×defaultMatcher) Search(feed *Feed, searchTerm string)

//聲明一個指向defaultMatcher類型的值
var dm defaultMatch

//編譯器會自動生成指針引用dm值,使用指針調(diào)用方法
dm.Search(feed, "test")

與直接通過值或者指針調(diào)用方法不同,如果通過接口類型的值調(diào)用方法,規(guī)則有很大不同,使用指針作為接收者聲明的方法,只能在接口類型的值是一個指針的時候被調(diào)用。使用值作為接收者聲明的方法,在接口類型的值為值或者指針時,都可以被調(diào)用。

接口方法調(diào)用所受限制的例子

//方法聲明為使用指向defaultMatcher類型值的指針作為接收者
func (m *defaultMatcher) Search(feed *Feed, searchTerm string)

//通過interface類型的值來調(diào)用方法
var dm defaultMatcher
var matcher Matcher = dm //將值賦值給接口類型
matcher.Search(feed, "test") //使用值來調(diào)用接口方法

> go build
cannot use dm (type defaultMatcher) as type Matcher in assignment

//方法聲明為使用defaultMatcher類型的值作為接收者
func (m defaultMatcher) Search(feed *Feed, searchTerm string)

//通過interface類型的值來調(diào)用方法
var dm defaultMatcher
var matcher Matcher = &dm //將指針賦值給接口類型
matcher.Search(feed, "test") //使用指針來調(diào)用接口方法

> go build
Build Successful

2.4 RSS 匹配器

matchers/rss.go

package matchers

import (
? ? "encoding/xml"
? ? "errors"
? ? "fmt"
? ? "log"
? ? "net/http"
? ? "regexp"
? ??
? ? "github.com/goinaction/code/chapter2/sample/search"
)

type (
? ? //item根據(jù)item字段的標(biāo)簽,將定義的字段
? ? //與rss文擋的字段關(guān)聯(lián)起來
? ? item struct {
? ? ? ? XMLName ? ? xml.Name ?’xml:"item"‘
? ? ? ? PubDate ? ? string ? ?’xml:"pubDate"‘
? ? ? ? Title ? ? ? string ? ?’xml:"title"‘
? ? ? ? Description string ? ?’xml:"description"‘
? ? ? ? Link ? ? ? ?string ? ?’xml:"link"‘
? ? ? ? GUID ? ? ? ?string ? ?’xml:"guid"‘
? ? ? ? GeoRssPoint string ? ?’xml:"georss:point"‘
? ? }
? ??
? ? //image根據(jù)image字段的標(biāo)簽,將定義的字段
? ? //與rss文檔的字段關(guān)聯(lián)起來
? ? image struct {
? ? ? ? XMLName xml.Name ?’xml:"image"‘
? ? ? ? URL ? ? string ? ?’xml:"url"‘
? ? ? ? Title ? string ? ?’xml:"title"‘
? ? ? ? Link ? ?string ? ?’xml:"link"‘
? ? }
? ??
? ? //channel根據(jù)channel字段的標(biāo)簽,將定義的字段
? ? //與rss文檔的字段關(guān)聯(lián)起來
? ? channel struct {
? ? ? ? XMLName ? ? ? ?xml.Name ?’xml:"channel"‘
? ? ? ? Title ? ? ? ? ?string ? ?’xml:"title"‘
? ? ? ? Description ? ?string ? ?’xml:"description"‘
? ? ? ? Link ? ? ? ? ? string ? ?’xml:"link"‘
? ? ? ? PubDate ? ? ? ?string ? ?’xml:"pubDate"‘
? ? ? ? LastBuildDate ?string ? ?’xml:"lastBuildDate"‘
? ? ? ? TTL ? ? ? ? ? ?string ? ?’xml:"ttl"‘
? ? ? ? Language ? ? ? string ? ?’xml:"language"‘
? ? ? ? ManagingEditor string ? ?’xml:"managingEditor"‘
? ? ? ? WebMaster ? ? ?string ? ?’xml:"webMaster"‘
? ? ? ? Image ? ? ? ? ?image ? ? 'xml:"image"'
? ? ? ? Item ? ? ? ? ? []item ? ?'xml:"item"'
? ? }
? ??
? ? //rssDocument定義了與rss文檔關(guān)聯(lián)的字段
? ? rssDocument struct {
? ? ? ? XMLName xml.Name ?’xml:"rss"‘?
? ? ? ? Channel channel ? ’xml:"channel"‘
? ? }
)

//rssMatcher實現(xiàn)了Matcher接口
type rssMaster struct{}

//init將匹配器注冊到程序里
func init() {
? ? var matcher rssMatcher
? ? search.Register("rss", matcher)
}

//Search在文檔中查找特定的搜索項
func (m rssMatcher) Search(feed *search.Feed, searchTerm string)
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ([]*search.Result, error) {

? ? //我們使用關(guān)鍵字 var 聲明了一個值為 nil 的切片,切片每一項都是指向 Result 類型值的指針。
? ? var result []*search.Result
? ? log.Printf("Search Feed Type[%s] Site[%s] For Uri[%s]\n",
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? feed.Type, feed.Name, feed.URI)
? ? //獲取要搜索的數(shù)據(jù)
? ? document, err := m.retrieve(feed)
? ? if err != nil {
? ? ? ? return nil, err
? ? }
? ??
? ? for _, channelItem := range document.Channel.Item {
? ? ? ? //檢查標(biāo)題部分是否包含搜索項
? ? ? ? matched, err := regexp.MatchString(searchTerm, channelItem.Title)
? ? ? ? if err != nil {
? ? ? ? ? ? return nil, err
? ? ? ? }
? ? ? ??
? ? ? ? //如果找到匹配的項,將其作為結(jié)果保存
? ? ? ? if matched {
? ? ? ? ? ? results = append(results, &search.Result){
? ? ? ? ? ? ? ? Field: "Title",
? ? ? ? ? ? ? ? Content: channelItem.Tile,
? ? ? ? ? ? })
? ? ? ? }
? ? ? ??
? ? ? ? //檢查描述部分是否包含搜索項
? ? ? ? matched, err = regexp.MatchString(searchTern, channelTerm.Description)
? ? ? ? if err != nil {
? ? ? ? ? ? return nil, err
? ? ? ? }
? ? ? ??
? ? ? ? ?//如果找到匹配的項, 將其作為結(jié)果保存
? ? ? ? ?if matched {
? ? ? ? ? ? ?results = append(results, &search.Result{
? ? ? ? ? ? ? ? ?Field: "Description",
? ? ? ? ? ? ? ? ?Content: channelItem.Description,
? ? ? ? ? ? ?})
? ? ? ? ?}
? ? }
? ??
? ? return results, nil
}

//retrieve發(fā)送HTTP Get請求獲取rss數(shù)據(jù)源并解碼
Func (m rssMatcher) retrieve(feed *search.Feed) (*rssDocument, error) {
? ? if feed.URI == "" {
? ? ? ? return nil, errors.New("No rss feed URI provided")
? ? }
? ??
? ? //從網(wǎng)絡(luò)獲得rss數(shù)據(jù)源文檔
? ? resp, err := http.Get(feed.URI)
? ? if err != nil {
? ? ? ? return nil, err
? ? }
? ??
? ? //一旦從函數(shù)返回,關(guān)閉返回的響應(yīng)鏈接
? ? defer resp.Body.Close()
? ??
? ? //檢查狀態(tài)碼是不是200,這樣就知道
? ? //是不是收到了正確的響應(yīng)
? ? if resp.StatusCode != 200 {
? ? ? ? return nil, fmt.Errorf("HTTP Response Error %d\n", resp.StatusCode)
? ? }
? ??
? ? //將rss數(shù)據(jù)源文檔解碼到我們定義的結(jié)構(gòu)類型里
? ? //不需要檢查錯誤,調(diào)用者會做這件事
? ? var document rssDocument
? ? err = xml.NewDecoder(resp.Body).Decode(&document)
? ? return &document, err
}

2.5 小結(jié)

每個代碼文件都屬于一個包,而包名應(yīng)該與代碼文件所在的文件夾同名;Go 語言提供了多種聲明和初始化變量的方式。如果變量的值沒有顯式初始化,編譯器會將變量初始化為零值;使用指針可以在函數(shù)間或者 goroutine 間共享數(shù)據(jù);通過啟動 goroutine 和使用通道完成并發(fā)和同步;Go 語言提供了內(nèi)置函數(shù)來支持 Go 語言內(nèi)部的數(shù)據(jù)結(jié)構(gòu);標(biāo)準(zhǔn)庫包含很多包,能做很多很有用的事情;使用 Go 接口可以編寫通用的代碼和框架。

第 3 章 打包和工具鏈

本章主要內(nèi)容:如何組織 Go 代碼;使用 Go 語言自帶的相關(guān)命令;使用其他開發(fā)者提供的工具;與其他開發(fā)者合作。

在 Go 語言里,包是個非常重要的概念。其設(shè)計理念是使用包來封裝不同語義單元的功能。這樣做,能夠更好地復(fù)用代碼,并對每個包內(nèi)的數(shù)據(jù)的使用有更好的控制。所有的.go 文件,除了空行和注釋,都應(yīng)該在第一行聲明自己所屬的包。每個包都在一個單獨的目錄里。不能把多個包放到同一個目錄中,也不能把同一個包的文件分拆到多個不同目錄中。這意味著,同一個目錄下的所有.go 文件必須聲明同一個包名。

3.1 main包

經(jīng)典的“Hello World!”程序

hello.go

package main

import "fmt"

func main() {
? ? fmt.Println("Hello World!"):
}

3.2 導(dǎo)入

如果需要導(dǎo)入多個包,習(xí)慣上是將import 語句包裝在一個導(dǎo)入塊中。strings 包提供了很多關(guān)于字符串的操作,如查找、替換或
者變換。

import (
? ? ? ? "fmt"
? ? ? ? "strings"

)

3.2.1 遠(yuǎn)程導(dǎo)入?

Go 工具鏈會使用導(dǎo)入路徑確定需要獲取的代碼在網(wǎng)絡(luò)的什么地方。
例如:import "github.com/spf13/viper

3.2.2 命名導(dǎo)入

重命名導(dǎo)入。有時,用戶可能需要導(dǎo)入一個包,但是不需要引用這個包的標(biāo)識符。在這種情況,可以使用空白標(biāo)識符_來重命名這個導(dǎo)入。

3.3 函數(shù) init

init 函數(shù)的用法

package postgres

import (
? ? "database/sql"
)

func init() {
? ? //創(chuàng)建一個 postgres 驅(qū)動的實例。這里為了展現(xiàn) init 的作用,沒有展現(xiàn)其定義細(xì)節(jié)。
? ? sql.Register("postgres", new(PostgresDriver))
}

在使用這個新的數(shù)據(jù)庫驅(qū)動寫程序時,我們使用空白標(biāo)識符來導(dǎo)入包,以便新的驅(qū)動會包含到 sql 包。如前所述,不能導(dǎo)入不使用的包,為此使用空白標(biāo)識符重命名這個導(dǎo)入可以讓 init函數(shù)發(fā)現(xiàn)并被調(diào)度運行,讓編譯器不會因為包未被使用而產(chǎn)生錯誤。我們可以調(diào)用 sql.Open 方法來使用這個驅(qū)動。

導(dǎo)入時使用空白標(biāo)識符作為包的別名:

package main

import (
? ? "database/sql"
? ? //使用空白標(biāo)識符導(dǎo)入包,避免編譯錯誤。
? ? _"github.com/goinaction/code/chapter3/dbdriver/postgres"
)
func main() {
? ? //調(diào)用sql包提供的open方法。該方法能工作的關(guān)鍵在于postgres驅(qū)動通過自己的init函數(shù)將自身注冊到了sql包。
? ? sql.Open("postgres","mydb")
}

3.4 使用 Go 的工具

build 和 clean 命令會執(zhí)行編譯和清理的工作。go build hello.go

調(diào)用 clean 后會刪除編譯生成的可執(zhí)行文件。go clean hello.go

使用 io 包的工作

package main
import (
? ? "fmt"
? ? "io/ioutil"
? ? "os"
? ??
? ? "github.com/goinaction/code/chapter3/words"
)
// main 是應(yīng)用程序的入口
func main() {
? ? filename := os.Args[1]
? ??
? ? contents, err := ioutil.ReadFile(filename)
? ? if err != nil {
? ? ? ? fmt.Println(err)
? ? ? ? return
? ? }
? ? text := string(contents)
? ? count := words.CountWords(text)
? ??
? ? fmt.Printf("There are %d words in your text.\n", count)
}

做開發(fā)會經(jīng)常使用 go build 和 go run 命令。go build wordcount.go

go run 命令會先構(gòu)建 wordcount.go 里包含的程序,然后執(zhí)行構(gòu)建后的程序。go run wordcount.go

3.5 進(jìn)一步介紹 Go 開發(fā)工具

vet 命令會幫開發(fā)人員檢測代碼的常見錯誤:Printf 類函數(shù)調(diào)用時,類型匹配錯誤的參數(shù);定義常用的方法時,方法簽名的錯誤;錯誤的結(jié)構(gòu)標(biāo)簽;沒有指定字段名的結(jié)構(gòu)字面量。

使用 go vet工具不能讓開發(fā)者避免嚴(yán)重的邏輯錯誤,或者避免編寫充滿小錯的代碼。

package main

import "fmt"
func main() {

? ? fmt.Printf("The quick brown fox jumped over lazy dogs", 3.14)
}

這個程序要輸出一個浮點數(shù) 3.14,但是在格式化字符串里并沒有對應(yīng)的格式化參數(shù)。

3.5.2 Go 代碼格式化

fmt 命令會自動格式化開發(fā)人員指定的源代碼文件并保存。

3.5.3 Go 語言的文檔

Go 語言有兩種方法為開發(fā)者生成文檔。如果開發(fā)人員使用命令行提示符工作,可以在終端上直接使用 go doc 命令來打印文檔。

1.從命令行獲取文檔go doc tar;2.瀏覽文檔godoc -http=:6060

3.6 與其他 Go 開發(fā)者合作

以分享為目的創(chuàng)建代碼庫

1.包應(yīng)該在代碼庫的根目錄中(使用 go get 的時候,開發(fā)人員指定了要導(dǎo)入包的全路徑);2.包可以非常小;

3.對代碼執(zhí)行 go fmt;4.給代碼寫文檔。

3.7 依賴管理

最流行的依賴管理工具有g(shù)odep、vender、gopkg.in 工具

3.7.1 第三方依賴

像 godep 和 vender 這種社區(qū)工具已經(jīng)使用第三方(verdoring)導(dǎo)入路徑重寫這種特性解決了依賴問題。其思想是把所有的依賴包復(fù)制到工程代碼庫中的目錄里,然后使用工程內(nèi)部的依賴包所在目錄來重寫所有的導(dǎo)入路徑。

3.7.2 對 gb 的介紹

gb 背后的原理源自理解到 Go 語言的 import 語句并沒有提供可重復(fù)構(gòu)建的能力。 import語句可以驅(qū)動 go get ,但是 import 本身并沒有包含足夠的信息來決定到底要獲取包的哪個修改的版本。 go get 無法定位待獲取代碼的問題,導(dǎo)致 Go 工具在解決重復(fù)構(gòu)建時,不得不使用復(fù)雜且難看的方法。

gb 的創(chuàng)建源于上述理解。gb 既不包裝 Go 工具鏈,也不使用 GOPATH 。gb 基于工程將 Go 工具鏈工作空間的元信息做替換。這種依賴管理的方法不需要重寫工程內(nèi)代碼的導(dǎo)入路徑。而且導(dǎo)入路徑依舊通過 go get 和 GOPATH 工作空間來管理。gb 工程與 Go 官方工具鏈(包括 go get )并不兼容。

3.8 小結(jié)
在 Go 語言中包是組織代碼的基本單位;?環(huán)境變量 GOPATH 決定了 Go 源代碼在磁盤上被保存、編譯和安裝的位置;可以為每個工程設(shè)置不同的 GOPATH ,以保持源代碼和依賴的隔離;go 工具是在命令行上工作的最好工具;開發(fā)人員可以使用 go get 來獲取別人的包并將其安裝到自己的 GOPATH 指定的目錄;想要為別人創(chuàng)建包很簡單,只要把源代碼放到公用代碼庫,并遵守一些簡單規(guī)則就可以了;Go 語言在設(shè)計時將分享代碼作為語言的核心特性和驅(qū)動力;推薦使用依賴管理工具來管理依賴;有很多社區(qū)開發(fā)的依賴管理工具,如 godep、vender 和 gb。

第4章 數(shù)組、切片、映射

本章主要內(nèi)容:數(shù)組的內(nèi)部實現(xiàn)和基礎(chǔ)功能;使用切片管理數(shù)據(jù)集合;使用映射管理鍵值對。

4.1 數(shù)組的內(nèi)部實現(xiàn)和基礎(chǔ)功能

數(shù)組是切片和映射的基礎(chǔ)數(shù)據(jù)結(jié)構(gòu)。

4.1.1 內(nèi)部實現(xiàn)

數(shù)組存儲的類型可以是內(nèi)置類型,如整型或者字符串,也可以是某種結(jié)構(gòu)類型。數(shù)組的類型信息可以提供每次訪問一個元素時需要在內(nèi)存中移動的距離。

4.1.2 聲明和初始化

聲明一個數(shù)組,并設(shè)置為零值;使用數(shù)組字面量聲明數(shù)組;讓 Go 自動計算聲明數(shù)組的長度;聲明數(shù)組并指定特定元素的值。

4.1.3 使用數(shù)組

訪問數(shù)組元素;訪問指針數(shù)組的元素;把同樣類型的一個數(shù)組賦值給另外一個數(shù)組;把一個指針數(shù)組賦值給另一個指針數(shù)組。

數(shù)組變量的類型包括數(shù)組長度和每個元素的類型。只有這兩部分都相同的數(shù)組,才是類型相同的數(shù)組,才能互相賦值。編譯器會阻止類型不同的數(shù)組互相賦值。

4.1.4 多維數(shù)組

聲明二維數(shù)組,訪問二維數(shù)組的元素,同樣類型的多維數(shù)組賦值,使用索引為多維數(shù)組賦值。

4.1.5 在函數(shù)間傳遞數(shù)組

使用值傳遞,在foo函數(shù)間傳遞大數(shù)組。每次函數(shù) foo 被調(diào)用時,必須在棧上分配 8 MB 的內(nèi)存。之后,整個數(shù)組的值(8 MB 的內(nèi)存)被復(fù)制到剛分配的內(nèi)存里。只傳入指向數(shù)組的指針,這樣只需要復(fù)制 8 字節(jié)的數(shù)據(jù)而不是8 MB 的內(nèi)存數(shù)據(jù)到棧上,使用指針在函數(shù)間傳遞大數(shù)組,這個操作會更有效地利用內(nèi)存,性能也更好。不過要意識到,因為現(xiàn)在傳遞的是指針,所以如果改變指針指向的值,會改變共享的內(nèi)存。如你所見,使用切片能更好地處理這類共享問題。

4.2 切片的內(nèi)部實現(xiàn)和基礎(chǔ)功能

切片是圍繞動態(tài)數(shù)組的概念構(gòu)建的,可以按需自動增長和縮小。切片的動態(tài)增長是通過內(nèi)置函數(shù) append 來實現(xiàn)的。切片的底層內(nèi)存也是在連續(xù)塊中分配的,所以切片還能獲得索引、迭代以及為垃圾回收優(yōu)化的好處。

切片有 3 個字段的數(shù)據(jù)結(jié)構(gòu),這些數(shù)據(jù)結(jié)構(gòu)包含 Go 語言需要操作底層數(shù)組的元數(shù)據(jù),這 3 個字段分別是指向底層數(shù)組的指針、切片訪問的元素的個數(shù)(即長度)和切片允許增長到的元素個數(shù)(即容量)。

4.2.2 創(chuàng)建和初始化

1.make 和切片字面量?

使用長度聲明一個字符串切片;使用長度和容量聲明整型切片。容量小于長度的切片會在編譯時報錯。

通過切片字面量來聲明切片;使用索引聲明切片;

2.nil 和空切片

只要在聲明時不做任何初始化,就會創(chuàng)建一個 nil 切片。nil 切片可以用于很多標(biāo)準(zhǔn)庫和內(nèi)置函數(shù)。在需要描述一個不存在的切片時, nil 切片會很好用。例如,函數(shù)要求返回一個切片但是發(fā)生異常的時候。

利用初始化,通過聲明一個切片可以創(chuàng)建一個空切片。空切片在底層數(shù)組包含 0 個元素,也沒有分配任何存儲空間。想表示空集合時空切片很有用,例如,數(shù)據(jù)庫查詢返回 0 個查詢結(jié)果時。

4.2.3 使用切片

1.賦值和切片

使用切片字面量來聲明切片;使用切片創(chuàng)建切片。

如何計算長度和容量:

對底層數(shù)組容量是 k 的切片 slice[i:j]來說
長度: j - i
容量: k - i

修改切片內(nèi)容可能導(dǎo)致的結(jié)果。與切片的容量相關(guān)聯(lián)的元素只能用于增長切片。在使用這部分元素前,必須將其合并到切片的長度里。

2.切片增長

使用 append 向切片增加元素;使用 append 同時增加切片的長度和容量;使用 append 同時增加切片的長度和容量。

3.創(chuàng)建切片時的 3 個索引

使用切片字面量聲明一個字符串切片;使用 3 個索引創(chuàng)建切片;

如果試圖設(shè)置的容量比可用的容量還大,就會得到一個語言運行時錯誤。內(nèi)置函數(shù) append 會首先使用可用容量。一旦沒有可用容量,會分配一個
新的底層數(shù)組。這導(dǎo)致很容易忘記切片間正在共享同一個底層數(shù)組。一旦發(fā)生這種情況,對切片
進(jìn)行修改,很可能會導(dǎo)致隨機且奇怪的問題。對切片內(nèi)容的修改會影響多個切片,卻很難找到問
題的原因。
如果在創(chuàng)建切片時設(shè)置切片的容量和長度一樣,就可以強制讓新切片的第一個 append 操作創(chuàng)建新的底層數(shù)組,與原有的底層數(shù)組分離。新切片與原有的底層數(shù)組分離后,可以安全地進(jìn)行后續(xù)修改。內(nèi)置函數(shù) append 也是一個可變參數(shù)的函數(shù)。這意味著可以在一次調(diào)用傳遞多個追加的值。如果使用 ... 運算符,可以將一個切片的所有元素追加到另一個切片里。

4.迭代切片

使用 for range 迭代切片,當(dāng)?shù)衅瑫r,關(guān)鍵字 range 會返回兩個值。第一個值是當(dāng)前迭代到的索引位置,第二個值是該位置對應(yīng)元素值的一份副本。range 創(chuàng)建了每個元素的副本,而不是直接返回對該元素的引用,如果使用該值變量的地址作為指向每個元素的指針,就會造成錯誤。

使用空白標(biāo)識符(下劃線)來忽略索引值。關(guān)鍵字 range 總是會從切片頭部開始迭代。如果想對迭代做更多的控制,依舊可以使用傳
統(tǒng)的 for 循環(huán),有兩個特殊的內(nèi)置函數(shù) len 和 cap ,可以用于處理數(shù)組、切片和通道。對于切片,函數(shù) len返回切片的長度,函數(shù) cap 返回切片的容量。

4.2.4 多維切片

組合切片的切片

4.2.5 在函數(shù)間傳遞切片

在函數(shù)間傳遞切片就是要在函數(shù)間以值的方式傳遞切片。由于切片的尺寸很小,在函數(shù)間復(fù)制和傳遞切片成本也很低。

4.3 映射的內(nèi)部實現(xiàn)和基礎(chǔ)功能

4.3.1 內(nèi)部實現(xiàn)

映射是無序的集合,意味著沒有辦法預(yù)測鍵值對被返回的順序。無序的原因是映射的實現(xiàn)使用了散列表。映射的散列表包含一組桶。在存儲、刪除或者查找鍵值對的時候,所有操作都要先選擇一個桶。把操作映射時指定的鍵傳給映射的散列函數(shù),就能選中對應(yīng)的桶。這個散列函數(shù)的目的是生成一個索引,這個索引最終將鍵值對分布到所有可用的桶里。

4.3.2 創(chuàng)建和初始化

使用 make 聲明映射,創(chuàng)建映射時,更常用的方法是使用映射字面量。映射的初始長度會根據(jù)初始化時指定的鍵值
對的數(shù)量來確定。
映射的鍵可以是任何值。這個值的類型可以是內(nèi)置的類型,也可以是結(jié)構(gòu)類型,只要這個值可以使用 == 運算符做比較。切片、函數(shù)以及包含切片的結(jié)構(gòu)類型這些類型由于具有引用語義,不能作為映射的鍵,使用這些類型會造成編譯錯誤。

使用映射字面量聲明空映射:

// 創(chuàng)建一個映射,使用字符串切片作為映射的鍵
dict := map[[ ]string]int{ };

聲明一個存儲字符串切片的映射:

// 創(chuàng)建一個映射,使用字符串切片作為值
dict := map[int][ ]string{ }

4.3.3 使用映射

從映射獲取值并判斷鍵是否存在:

// 獲取鍵 Blue 對應(yīng)的值
value, exists := colors["Blue"]

// 這個鍵存在嗎?
if exists {
fmt.Println(value)
}

從映射獲取值,并通過該值判斷鍵是否存在:

// 獲取鍵 Blue 對應(yīng)的值
value := colors["Blue"]
// 這個鍵存在嗎?
if value != "" {
? ? fmt.Println(value)
}

使用 range 迭代映射;

從映射中刪除一項:

// 刪除鍵為 Coral 的鍵值對
delete(colors, "Coral")
// 顯示映射里的所有顏色
for key, value := range colors {
? ? ?fmt.Printf("Key: %s Value: %s\n", key, value)
}

4.3.4 在函數(shù)間傳遞映射

當(dāng)傳遞映射給一個函數(shù),并對這個映射做了修改時,所有對這個映射的引用都會察覺到這個修改。

4.4 小結(jié)
數(shù)組是構(gòu)造切片和映射的基石;
Go 語言里切片經(jīng)常用來處理數(shù)據(jù)的集合,映射用來處理具有鍵值對結(jié)構(gòu)的數(shù)據(jù);內(nèi)置函數(shù) make 可以創(chuàng)建切片和映射,并指定原始的長度和容量。也可以直接使用切片和映射字面量,或者使用字面量作為變量的初始值;切片有容量限制,不過可以使用內(nèi)置的 append 函數(shù)擴展容量;映射的增長沒有容量或者任何限制;內(nèi)置函數(shù) len 可以用來獲取切片或者映射的長度;內(nèi)置函數(shù) cap 只能用于切片;通過組合,可以創(chuàng)建多維數(shù)組和多維切片。也可以使用切片或者其他映射作為映射的值;但是切片不能用作映射的鍵;?將切片或者映射傳遞給函數(shù)成本很小,并且不會復(fù)制底層的數(shù)據(jù)結(jié)構(gòu)。

第 5 章 Go 語言的類型系統(tǒng)

本章主要內(nèi)容:?聲明新的用戶定義的類型;使用方法,為類型增加新的行為;了解何時使用指針,何時使用值;通過接口實現(xiàn)多態(tài);通過組合來擴展或改變類型;公開或者未公開的標(biāo)識符。

Go 語言是一種靜態(tài)類型的編程語言。值的類型給編譯器提供兩部分信息:第一部分,需要分配多少內(nèi)存給這個值(即值的規(guī)模)
;第二部分,這段內(nèi)存表示什么。對于許多內(nèi)置類型的情況來說,規(guī)模和表示是類型名的一部分。

5.1 用戶定義的類型

//user 在程序里定義一個用戶類型
type user struct {
? ? name ? ?string
? ? email ? string
? ? ext ? ? int
? ? privileged, bool
}?

使用結(jié)構(gòu)類型聲明變量,并初始化為其零值

聲明user類型的變量

var bill user

任何時候,創(chuàng)建一個變量并初始化為其零值,習(xí)慣是使用關(guān)鍵字 var 。如果變量被初始化為某個非零值,就配合結(jié)構(gòu)字面量和短變量
聲明操作符來創(chuàng)建變量。一個短變量(:=)聲明操作符在一次操作中完成兩件事情:聲明一個變量,并初始化。短變量聲明操作符會使用右側(cè)給出的類型信息作為聲明變量的類型。

使用結(jié)構(gòu)字面量創(chuàng)建結(jié)構(gòu)類型的值;不使用字段名,創(chuàng)建結(jié)構(gòu)類型的值;

使用其他結(jié)構(gòu)類型聲明字段:

//admin需要一個user類型作為管理者,并附加權(quán)限
type admin struct {
? ? person user?
? ? level string
}

使用結(jié)構(gòu)字面量來創(chuàng)建字段的值:

// 聲明 admin 類型的變量
fred := admin{
? ? person: user{
? ? name: ? ? ? "Lisa",
? ? email: ? ? ?"lisa@email.com",
? ? ext: ? ? ? ?123,
? ? privileged: true,
? ? },
? ? level: "super",
}

另一種聲明用戶定義的類型的方法是,基于一個已有的類型,將其作為新類型的類型說明。

基于 int64 聲明一個新類型:

type Duration int64

Duration 是一種描述時間間隔的類型,單位是納秒(ns)。這個類型使用內(nèi)置的 int64 類型作為其表示。在 Duration類型的聲明中,我們把 int64 類型叫作 Duration 的基礎(chǔ)類型。不過,雖然 int64 是基礎(chǔ)類型,Go 并不認(rèn)為 Duration 和 int64 是同一種類型。這兩個類型是完全不同的有區(qū)別的類型。

給不同類型的變量賦值會產(chǎn)生編譯錯誤:

package main

type Duration int64

func mian() {
? ? var dur Duration
? ? dur = int64(1000)
}

5.2 方法

方法能給用戶定義的類型添加新的行為。方法實際上也是函數(shù),只是在聲明時,在關(guān)鍵字func 和方法名之間增加了一個參數(shù)

listing11.go

//這個示例程序展示如何聲明
//并使用方法
package main

import (
? ? "fmt"
)

//user在程序里定義一個用戶類型
type user struct {
? ? name string
? ? email string
}

//notify使用值接收者實現(xiàn)了一個方法
func (u user) notify() {
? ? fmt.Printf("Sending User Email To %s<%s>\n",
? ? u.name,
? ? u.email)
}

//changeEmail使用指針接收者實現(xiàn)了一個方法
func (u *user) changeEmail(email string) {
? ? u.email = email
}

//main是應(yīng)用程序的入口
func main() {
? ? //user類型的值可以用來調(diào)用
? ? //使用值接收者聲明的方法
? ? bill := user{"Bill", "bill@email.com"}
? ? bill.notify()
? ??
? ? //指向user類型值的指針也可以用來調(diào)用
? ? //使用值接收者聲明的方法
? ? lisa := &user{"Lisa", "lisa@email.com"}
? ? lisa.notify()
? ??
? ? //user類型的值可以用來調(diào)用
? ? //使用指針接收者聲明方法
? ? bill.changeEmail("bill@newdomain.com")
? ? bill.notify()
? ??
? ? //指向user類型值的指針也可以用來調(diào)用
? ? //使用指針接收者聲明的方法
? ? lisa.changeEmail("lisa@newdomain.com")
? ? lisa.notify()
}
Go 語言里有兩種類型的接收者:值接收者和指針接收者。notify 方法的接收者被聲明為 user 類型的值。如果使用值接收者聲明方法,調(diào)用時會使用這個值的一個副本來執(zhí)行。

使用變量來調(diào)用方法:

bill.notify()

這個語法與調(diào)用一個包里的函數(shù)看起來很類似。但在這個例子里, bill 不是包名,而是變量名。這段程序在調(diào)用 notify 方法時,使用 bill 的值作為接收者進(jìn)行調(diào)用,方法 notify會接收到 bill 的值的一個副本。

值接收者使用值的副本來調(diào)用方法,而指針接受者使用實際值來調(diào)用方法。

5.3 類型的本質(zhì)

在聲明一個新類型之后,聲明一個該類型的方法之前,果給這個類型增加或者刪除某個值,是要創(chuàng)建一個新值,還是要更改當(dāng)前的值?如果是要創(chuàng)建一個新值,該類型的方法就使用值接收者。如果是要修改當(dāng)前值,就使用指針接收者。

5.3.1 內(nèi)置類型

內(nèi)置類型是由語言提供的一組類型,分別是數(shù)值類型、字符串類型和布爾類型。當(dāng)對這些值進(jìn)行增加或者刪除的時候,會創(chuàng)建一個新值。當(dāng)把這些類型的值傳遞給方法或者函數(shù)時,應(yīng)該傳遞一個對應(yīng)值的副本。

5.3.2 引用類型

Go 語言里的引用類型有如下幾個:切片、映射、通道、接口和函數(shù)類型。當(dāng)聲明上述類型當(dāng)聲明上述類型的變量時,創(chuàng)建的變量被稱作標(biāo)頭(header)值。

5.3.3 結(jié)構(gòu)類型

type Time struct {
? ? // sec 給出自公元 1 年 1 月 1 日 00:00:00
? ? // 開始的秒數(shù)
? ? sec int64


? ? // nsec 指定了一秒內(nèi)的納秒偏移,
? ? // 這個值是非零值,
? ? // 必須在[0, 999999999]范圍內(nèi)
? ? nsec int32


? ? // loc 指定了一個 Location,
? ? // 用于決定該時間對應(yīng)的當(dāng)?shù)氐姆帧⑿r、
? ? // 天和年的值
? ? // 只有 Time 的零值,其 loc 的值是 nil
? ? // 這種情況下,認(rèn)為處于 UTC 時區(qū)
? ? loc *Location
}

Time 結(jié)構(gòu)選自 time 包。當(dāng)思考時間的值時,你應(yīng)該意識到給定的一個時間點的時間是不能修改的。

func Now() Time {
? ? sec, nsec := now()
? ? return Time{sec + unixToInternal, nsec, Local}
}

展示了 Now 函數(shù)的實現(xiàn)。這個函數(shù)創(chuàng)建了一個 Time 類型的值,并給調(diào)用者返回了 Time 值的副本。這個函數(shù)沒有使用指針來共享 Time 值。

func (t Time) Add(d Duration) Time {
? ? t.sec += int64(d / 1e9)
? ? nsec := int32(t.nsec) + int32(d%1e9)
? ? if nsec >= 1e9 {
? ? ? ? t.sec++
? ? ? ? nsec -= 1e9
? ? } else if nsec < 0 {
? ? ? ? t.sec--
? ? ? ? nsec += 1e9
? ? }
? ? t.nsec = nsec
? ? return t
}func (t Time) Add(d Duration) Time {
? ? t.sec += int64(d / 1e9)
? ? nsec := int32(t.nsec) + int32(d%1e9)
? ? if nsec >= 1e9 {
? ? ? ? t.sec++
? ? ? ? nsec -= 1e9
? ? } else if nsec < 0 {
? ? ? ? t.sec--
? ? ? ? nsec += 1e9
? ? }
? ? t.nsec = nsec
? ? return t
}

這個方法使用值接收者,并返回了一個新的 Time 值。該方法操作的是調(diào)用者傳入的 Time 值的副本,并且給調(diào)用者返回了一個方法內(nèi)的 Time 值的副本。至于是使用返回的值替換原來的 Time 值,還是創(chuàng)建一個新的 Time 變量來保存結(jié)果,是由調(diào)用者決定的事情。

golang.org/src/os/file_unix.go:

// File 表示一個打開的文件描述符
type File struct {
? ? *file
}
// file 是*File 的實際表示
// 額外的一層結(jié)構(gòu)保證沒有哪個 os 的客戶端
// 能夠覆蓋這些數(shù)據(jù)。如果覆蓋這些數(shù)據(jù),
// 可能在變量終結(jié)時關(guān)閉錯誤的文件描述符
type file struct {
? ? fd int
? ? name string
? ? dirinfo *dirInfo // 除了目錄結(jié)構(gòu),此字段為 nil
? ? nepipe int32 // Write 操作時遇到連續(xù) EPIPE 的次數(shù)
}

標(biāo)準(zhǔn)庫中聲明的 File 類型。這個類型的本質(zhì)是非原始的。這個類型的值實際上不能安全復(fù)制。對內(nèi)部未公開的類型的注釋,解釋了不安全的原因。因為沒有方法阻止程序員進(jìn)行復(fù)制,所以 File 類型的實現(xiàn)使用了一個嵌入的指針,指向一個未公開的類型。

golang.org/src/os/file.go:
func Open(name string) (file *File, err error) {
? ? return OpenFile(name, O_RDONLY, 0)
}

展示了 Open 函數(shù)的實現(xiàn),調(diào)用者得到的是一個指向 File 類型值的指針。Open 創(chuàng)建了 File 類型的值,并返回指向這個值的指針。如果一個創(chuàng)建用的工廠函數(shù)返回了一個指針,就表示這個被返回的值的本質(zhì)是非原始的。即便函數(shù)或者方法沒有直接改變非原始的值的狀態(tài),依舊應(yīng)該使用共享的方式傳遞。

func (f *File) Chdir() error {
? ? if f == nil {
? ? return ErrInvalid
? ? }
? ? if e := syscall.Fchdir(f.fd); e != nil {
? ? return &PathError{"chdir", f.name, e}
? ? }
? ? return nil
}

Chdir 方法展示了,即使沒有修改接收者的值,依然是用指針接收者來聲明的。因為 File 類型的值具備非原始的本質(zhì),所以總是應(yīng)該被共享,而不是被復(fù)制。是使用值接收者還是指針接收者,不應(yīng)該由該方法是否修改了接收到的值來決定。這個決策應(yīng)該基于該類型的本質(zhì)。這條規(guī)則的一個例外是,需要讓類型值符合某個接口的時候,即便類型的本質(zhì)是非原始本質(zhì)的,也可以選擇使用值接收者聲明方法。這樣做完全符合接口值調(diào)用方法的機制。

5.4 接口

多態(tài)是指代碼可以根據(jù)類型的具體實現(xiàn)采取不同行為的能力。如果一個類型實現(xiàn)了某個接口,所有使用這個接口的地方,都可以支持這種類型的值。

5.4.1 標(biāo)準(zhǔn)庫

示例程序?qū)崿F(xiàn)了流行程序 curl 的功能:

listing34.go

//這個示例程序展示如何使用io.Reader和io.Writer接口
//寫一個簡單版本的curl程序
package main

import (
? ? "fmt"
? ? "io"
? ? "net/http"
? ? "os"
)

//init在main函數(shù)之前調(diào)用
func init() {
? ? if len(os.Args) != 2 {
? ? ? ? fmt.Println("Usage: ./example2 <url>")
? ? ? ? os.Exit(-1)
? ? }
}

//main是應(yīng)用程序的入口
func main() {
? ? //從web服務(wù)器得到響應(yīng)
? ? r, err := http.Get(os.Args[1])
? ? if err != nil {
? ? ? ? fmt.Println(err)
? ? ? ? return
? ? }
? ??
? ? //從Body復(fù)制到Stdout
? ? io.Copy(os.Stdout, r.Body)
? ? if err := r.Body.close(); err != nil {
? ? ? ? fmt.Println(err)
? ? }
}

listing35.go

//這個示例程序展示bytes.Buffer也可以
//用于io.Copy函數(shù)
package main

import (
? ? "bytes"
? ? "fmt"
? ? "io"
? ? "os"
)

//main是應(yīng)用程序的入口
func main() {
? ? var b bytes.Buffer
? ??
? ? //將字符串寫入Buffer
? ? b.Writer([]byte("Hello"))
? ??
? ? //使用Fprintf將字符串拼接到Buffer
? ? fmt.Fprintf(&b, "world!")
? ??
? ? //將Buffer的內(nèi)容寫到Stdout
? ? io.Copy(os.Stdout, &b)
}

這個程序使用接口來拼接字符串,并將數(shù)據(jù)以流的方式輸出到標(biāo)準(zhǔn)輸出設(shè)備。

5.4.2 實現(xiàn)

接口是用來定義行為的類型。這些被定義的行為不由接口直接實現(xiàn),而是通過方法由用戶定義的類型實現(xiàn)。

圖 5-1 實體值賦值后接口值的簡圖

圖 5-2 實體指針賦值后接口值的簡圖

5.4.3 方法集
方法集定義了接口的接受規(guī)則。

listing36.go

//這個示例程序展示Go語言里如何使用接口

package main

import (
? ? "fmt"
)

//notifier是一個定義了
//通知類行為的接口
type notifier interface {
? ? notify()
}

//user在程序里定義一個用戶類型
type user struct {
? ? name string
? ? email string
}

//notify是使用指針接收者實現(xiàn)的方法
func (u *user) notify() {
? ? fmt.Printf("Sending user email to %s<%s>\n",
? ? ? ? u.name,
? ? ? ? u.email)
}

//main是應(yīng)用程序的入口
func main() {
? ? //創(chuàng)建一個user類型的值,并發(fā)送通知
? ? u := user{"Bill", "bill@email.com"}
? ??
? ? sendNotification(u)
? ??
? ? // ./listing36.go:32: 不能將u(類型是user)作為
? ? // ? ? ? ? ? ? ? ? ? ? ?sendNotification的參數(shù)類型notifier:
? ? // user類型并沒有實現(xiàn)notifier
? ? // ? ? ? ? ? ? ? ? ? ? ?(notifier方法使用指針接收者聲明)
}

//sendNotification接受一個實現(xiàn)了notifier接口的值
//并發(fā)送通知
func sendNotification(n notifier) {
? ? n.notify()
}
程序雖然看起來沒問題,但實際上卻無法通過編譯。用指針接收者來實現(xiàn)接口時為什么 user 類型的值無法實現(xiàn)該接口,需要先了解方法集。方法集定義了一組關(guān)聯(lián)到給定類型的值或者指針的方法。定義方法時使用的接收者的類型決定了這個方法是關(guān)聯(lián)到值,還是關(guān)聯(lián)到指針,還是兩個都關(guān)聯(lián)。

規(guī)范里描述的方法集

Values ? ? ?Methods Receivers
-----------------------------------------------
T ? ? ? ? ? (t T)
*T ? ? ? ? ?(t T) and (t *T)

從接收者類型的角度來看方法集

Methods Receivers ? Values
-----------------------------------------------
(t T) ? ? ? ? ? ? ? T and *T
(t *T) ? ? ? ? ? ? ?*T

編譯器并不是總能自動獲得一個值的地址

package main

import "fmt"

//duration是一個給予int類型的類型
type duration int

//duration是一個基于int類型的類型
type duration int

//使用更可讀的方式格式化duration值
func (d *duration) pretty() string {
? ? return fmt.Springf("Duration:%d", *d)
}

//main是應(yīng)用程序的入口
func main() {
? ? duration(42).pretty()
? ??
? ? // ./listing46.go:17: 不能通過指針調(diào)用duration(42)的方法
? ? // ./listing46.go:17: 不能獲取duration(42)的方法
}
5.4.4 多態(tài)

//這個示例程序使用接口展示多態(tài)行為
package main

import (
? ? "fmt"
)

//notifier是一個定義了
//通知類行為的接口
type notifier interface {
? ? notify()
}

//user在程序定義了一個用戶類型
type user struct {
? ? name string
? ? email string
}

//notify使用指針接收者實現(xiàn)了notifier接口
func (u *user) notify() {
? ? fmt.Printf("Sending user email to %s<%s>\n",
? ? ? ? u.name,
? ? ? ? u.email)
}

//admin定義了程序里的管理員
type admin struct {
? ? name string
? ? email string
}

//notify使用指針接收者實現(xiàn)了notifier接口
func (a *admin) notify() {
? ? fmt.Printf("Sending admin email to %s<%s>\n",
? ? ? ? a.name,
? ? ? ? a.email)
}

//main是應(yīng)用程序的入口
func main() {
? ? //創(chuàng)建一個user值并傳給sendNotificcation
? ? bill := user{"Bill", "bill@email.com"}
? ? sendNotification(&lisa)
}

//sendNotification接受一個實現(xiàn)了notifier接口的值
//并發(fā)送通知
func sendNotification(n notifier) {
? ? n.notify()
}

5.5 嵌入類型

Go 語言允許用戶擴展或者修改已有類型的行為。這個功能對代碼復(fù)用很重要,在修改已有類型以符合新類型的時候也很重要。嵌入類型是將已有的類型直接聲明在新的結(jié)構(gòu)類型里。被嵌入的類型被稱為新的外部類型的內(nèi)部類型。通過嵌入類型,與內(nèi)部類型相關(guān)的標(biāo)識符會提升到外部類型上。

//這個示例程序展示如何將一個類型嵌入另一個類型,以及
//內(nèi)部類型和外部類型之間的關(guān)系
package main

import (
? ? "fmt"
)

//user在程序里定義一個用戶類型
type user struct {
? ? name string
? ? email string
}

//notify實現(xiàn)了一個可以通過user類型值的指針
//調(diào)用的方法
func (u *user) notify() {
? ? fmt.Printf("Sending user email to %s<%s>\n",
? ? u.name,
? ? u.email)
}

//admin 代表一個擁有權(quán)限的管理員用戶
type admin struct {
? ? user //嵌入類型?
? ? level string?
}

//main是應(yīng)用程序的入口
func main() {
? ? //創(chuàng)建一個admin用戶
? ? ad := admin{
? ? ? ? user : user{
? ? ? ? ? ? name: "john smith",
? ? ? ? ? ? email: "john@yahoo.com",
? ? ? ? },
? ? ? ? level: "super",
? ? }
? ??
? ? //我們可以直接訪問內(nèi)部類型的方法
? ? ad.user.notify()
? ??
? ? //內(nèi)部類型的方法也被提升到外部類型
? ? ad.notify()
}

如何將嵌入類類型應(yīng)用于接口

package main

import (
? ? "fmt"
)

//notifier是一個定義了
//通知類行為的接口
type notifier interface {
? ? notify()
}

//user在程序里定義一個用戶類型
type user struct {
? ? name string
? ? email string
}

//通過user類型值的指針
//調(diào)用的方法
func (u *user) notify() {
? ? fmt.Printf("Sending user email to %s<%s>\n",
? ? u.name,
? ? u.email)
}

//admin代表一個擁有權(quán)限的管理員用戶
type admin struct {
? ? user
? ? level string
}

//main是應(yīng)用程序的入口
func main() {
? ? //創(chuàng)建一個admin用戶
? ? ad := admin{
? ? ? ? user: user{
? ? ? ? ? ? name: "john smith",
? ? ? ? ? ? email: "john@yahoo.com",
? ? ? ? },
? ? ? ? level: "super",
? ? }
? ??
? ? //給admin用戶發(fā)送一個通知
? ? //用于實現(xiàn)接口的內(nèi)部類型的方法,被提升到
? ? //外部類型
? ? sendNotification(&ad)
}

//sendNotification接受一個實現(xiàn)了notifier接口的值
//并發(fā)送通話
func sendNotification(n notifier) {
? ? n.notify()
}

示例程序展示當(dāng)內(nèi)部類型和外部類型要實現(xiàn)同一個接口時的做法

package main

import (
? ? "fmt"
)

//notifier是一個定義了
//通知類行為的接口
type notifier interface {
? ? notify()
}

//user在程序里定義一個用戶類型
type user struct {
? ? name string
? ? email string
}

//通過user類型值的指針
//調(diào)用的方法
func (u *user) notify() {
? ? fmt.Printf("Sending user email to %s<%s>\n",
? ? ? ? u.name,
? ? ? ? u.email)
}

//admin代表一個擁有權(quán)限的管理員用戶
type admin struct {
? ? user
? ? level string
}

//通過admin類型值的指針
//調(diào)用的方法
func (a *admin) notify() {
? ? fmt.Printf("Sending admin email to %s<%s>\n",
? ? ? ? a.name,
? ? ? ? a.email)
}

//main是應(yīng)用程序的入口
func main() {
? ? //創(chuàng)建一個admin用戶
? ? ad := admin{
? ? ? ? user: user{
? ? ? ? ? ? name: "john smith",
? ? ? ? ? ? email: "john@yahoo.com",
? ? ? ? },
? ? ? ? level: "super",
? ? }
? ??
? ? //給admin用戶發(fā)送一個通知
? ? //接口的嵌入的內(nèi)部類型實現(xiàn)并沒有替升到
? ? //外部類型
? ? sendNotification(&ad)
? ??
? ? //我們可以直接訪問內(nèi)部類型的方法
? ? ad.user.notify()
? ??
? ? //內(nèi)部類型的方法沒有被提升
? ? ad.notify()
}

//sendNotification接受一個實現(xiàn)了notifier接口的值
//并發(fā)送通知
func sendNotification(n notifier) {
? ? n.notify()
}

listing60.go 的輸出
Sending admin email to john smith<john@yahoo.com>
Sending user email to john smith<john@yahoo.com>
Sending admin email to john smith<john@yahoo.com>

這次我們看到了 admin 類型是如何實現(xiàn) notifier 接口的,以及如何由 sendNotification函數(shù)以及直接使用外部類型的變量 ad 來執(zhí)行 admin 類型實現(xiàn)的方法。這表明,如果外部類型實現(xiàn)了 notify 方法,內(nèi)部類型的實現(xiàn)就不會被提升。不過內(nèi)部類型的值一直存在,因此還可以通過直接訪問內(nèi)部類型的值,來調(diào)用沒有被提升的內(nèi)部類型實現(xiàn)的方法。

5.6 公開或未公開的標(biāo)識符

當(dāng)一個標(biāo)識符的名字以小寫字母開頭時,這個標(biāo)識符就是未公開的,即包外的代碼不可見。如果一個標(biāo)識符以大寫字母開頭,這個標(biāo)識符就是公開的,即被包外的代碼可見。

例子已經(jīng)修改為使用工廠函數(shù)來創(chuàng)建一個未公開的 alertCounter 類型的值。

// counters 包提供告警計數(shù)器的功能
package counters

// alertCounter 是一個未公開的類型05
// 這個類型用于保存告警計數(shù)
type alertCounter int

// New 創(chuàng)建并返回一個未公開的
// alertCounter 類型的值
func New(value int) alertCounter {
? ? return alertCounter(value)
}

將工廠函數(shù)命名為 New 是 Go 語言的一個習(xí)慣。這個 New 函數(shù)做了些有意思的事情:它創(chuàng)建了一個未公開的類型的值,并將這個值返回給調(diào)用者。

/ main 是應(yīng)用程序的入口
func main() {
? ? // 使用 counters 包公開的 New 函數(shù)來創(chuàng)建
? ? // 一個未公開的類型的變量
? ? counter := counters.New(10)

? ? fmt.Printf("Counter: %d\n", counter)
?}

這個 New 函數(shù)返回的值被賦給一個名為 counter 的變量。這個程序可以編譯并且運行,要讓這個行為可行,需要兩個理由。第一,公開或者未公開的標(biāo)識符,不是一個值。第二,短變量聲明操作符,有能力捕獲引用的類型,并創(chuàng)建一個未公開的類型的變量。永遠(yuǎn)不能顯式創(chuàng)建一個未公開的類型的變量,不過短變量聲明操作符可以這么做。

由于內(nèi)部類型 user 是未公開的,這段代碼無法直接通過結(jié)構(gòu)字面量的方式初始化該內(nèi)部類型。不過,即便內(nèi)部類型是未公開的,內(nèi)部類型里聲明的字段依舊是公開的。既然內(nèi)部類型的標(biāo)識符提升到了外部類型,這些公開的字段也可以通過外部類型的字段的值來訪問。

5.7 小結(jié)
使用關(guān)鍵字 struct 或者通過指定已經(jīng)存在的類型,可以聲明用戶定義的類型;方法提供了一種給用戶定義的類型增加行為的方式;設(shè)計類型時需要確認(rèn)類型的本質(zhì)是原始的,還是非原始的;接口是聲明了一組行為并支持多態(tài)的類型;嵌入類型提供了擴展類型的能力,而無需使用繼承;?標(biāo)識符要么是從包里公開的,要么是在包里未公開的。

第6章 并發(fā)

本章主要內(nèi)容:使用 goroutine 運行程序;檢測并修正競爭狀態(tài);利用通道共享數(shù)據(jù)。

Web 服務(wù)需要在各自獨立的套接字(socket)上同時接收多個數(shù)據(jù)請求。每個套接字請求都是獨立的,可以完全獨立于其他套接字進(jìn)行處理 。具有并行執(zhí)行多個請求的能力可以顯著提高這類系統(tǒng)的性能。Go 語言里的并發(fā)指的是能讓某個函數(shù)獨立于其他函數(shù)運行的能力。當(dāng)一個函數(shù)創(chuàng)建為 goroutine時,Go 會將其視為一個獨立的工作單元。

Go 語言里的并發(fā)指的是能讓某個函數(shù)獨立于其他函數(shù)運行的能力。Go 語言的并發(fā)同步模型來自一個叫作通信順序進(jìn)程(Communicating Sequential Processes, CSP)的范型(paradigm)。CSP 是一種消息傳遞模型,通過在 goroutine 之間傳遞數(shù)據(jù)來傳遞消息,而不是對數(shù)據(jù)進(jìn)行加鎖來實現(xiàn)同步訪問。用于在 goroutine 之間同步和傳遞數(shù)據(jù)的關(guān)鍵數(shù)據(jù)類型叫作通道(channel)。

6.1 并發(fā)與并行

什么是操作系統(tǒng)的線程(thread)和進(jìn)程(process)。一個線程是一個執(zhí)行空間,這個空間會被操作系統(tǒng)調(diào)度來運行函數(shù)中所寫的代碼。每個進(jìn)程至少包含一個線程,每個進(jìn)程的初始線程被稱作主線程。因為執(zhí)行這個線程的空間是應(yīng)用程序的本身的空間,所以當(dāng)主線程終止時,應(yīng)用程序也會終止。

FIFO:先進(jìn)先出調(diào)度算法LRU:最近最久未使用調(diào)度算法兩者都是緩存調(diào)度算法,經(jīng)常用作內(nèi)存的頁面置換算法。

線程調(diào)度算法:1、先來先服務(wù)(FCFS) ?2、最短作業(yè)優(yōu)先(SJF) 3、基于優(yōu)先權(quán)的調(diào)度算法(FPPS) 4、時間片輪轉(zhuǎn)(RR) 5、多級隊列調(diào)度(Multilevel feedback queue)

搶占式、非搶占式

圖 6-1 一個運行的應(yīng)用程序的進(jìn)程和線程的簡要描繪

在圖 6-2 中,可以看到操作系統(tǒng)線程、邏輯處理器和本地運行隊列之間的關(guān)系。

圖 6-2 Go 調(diào)度器如何管理 goroutine?

并發(fā)(concurrency)不是并行(parallelism)。并行是讓不同的代碼片段同時在不同的物理處理器上執(zhí)行。并行的關(guān)鍵是同時做很多事情,而并發(fā)是指同時管理很多事情,這些事情可能只做了一半就被暫停去做別的事情了。

6.2 goroutine

深入了解一下調(diào)度器的行為,以及調(diào)度器是如何創(chuàng)建 goroutine 并管理其壽命的。

//這個示例程序展示如何創(chuàng)建goroutine

//以及調(diào)度器的行為\

package main

import (

? ? "fmt"

????"runtime"

????"sync"

)

//main是所有Go程序的入口

func main() {

????//分配一個邏輯處理器給調(diào)度器使用

????runtime.GOMAXPROCS(1)

????

????//wg用來等待程序完成

????//計數(shù)加2,表示要等待兩個goroutine

????var wg sync.WaitGroup

????wg.add(2)

????

????fmt.Println("Start Goroutines")

????go func() {

????????//在函數(shù)退出時調(diào)用Done來通知main函數(shù)工作已經(jīng)完成

????????defer wg.Done()

????????

????????//顯示字母表3次

????????for count := 0; count < 3; count++ {

????????????for char := 'a'; char < 'a'+26; char++ {

????????????????fmt.Printf("%c, char")

????????????}

????????}

????}()

????

????//聲明一個匿名函數(shù),并創(chuàng)建一個goroutine

????go func() {

????????//在函數(shù)退出時調(diào)用Done來通知mian函數(shù)工作已經(jīng)完成

????????defer wg.Done()

????????

????????//顯示字母表3次

????????for count := 0; count < 3; count++ {

????????????for char := 'A'; char < 'A'+26; char++ {

????????????????fmt.Printf("%c", char)

????????????}

????????}

????}()

????

????//等待goroutine結(jié)束

????fmt.Println("Waiting to Finish")

????wg.Wait()

????

????fmt.Println("\nTerminating Program")

}

基于調(diào)度器的內(nèi)部算法,一個正運行的 goroutine 在工作結(jié)束前,可以被停止并重新調(diào)度。調(diào)度器這樣做的目的是防止某個 goroutine 長時間占用邏輯處理器。當(dāng) goroutine 占用時間過長時,調(diào)度器會停止當(dāng)前正運行的 goroutine,并給其他可運行的 goroutine 運行的機會。

//這個示例程序展示goroutine調(diào)度器是如何在單個程序上

//切分時間片的

package main

import (

????"fmt"

????"runtime"

????"sync"

)

//wg用來等待程序完成

var wg sync.WaitGroup

//main是所有Go程序的入口

func main() {

????//分配一個邏輯處理器給調(diào)度器使用

????runtime.GOMAXPROCX(1)

????//計數(shù)加2,表示要等待兩個goroutine

????wg.Add(2)

????

????//創(chuàng)建兩個goroutine

????fmt.Println("Create Goroutines")

????go printPrime("A")

????go printPrime("B")

????

????//等待goroutine結(jié)束

????fmt.Println("Waiting To Finish")

????wg.Wait()

????

????fmt.Println("Terminating Program")

}

//printPrime顯示5000以內(nèi)的素數(shù)值

func printPrime(prefix string) {

????//函數(shù)退出時調(diào)用Done來通知main函數(shù)工作已經(jīng)完成

????defer wg.Done()

????

next:

????for outer := 2; outer < 5000; outer++ {

????????for inner := 2; inner < outer; inner++ {

????????????if outer%inner == 0 {

????????????????continue next

????????????}

????????}

????????fmt.Printf("%s:%d", prefix, outer)

????}

????fmt.Println("Completed", prefix)

}

6.3 競爭狀態(tài)

如果兩個或者多個 goroutine 在沒有互相同步的情況下,訪問某個共享的資源,并試圖同時讀和寫這個資源,就處于相互競爭的狀態(tài),這種情況被稱作競爭狀態(tài)(race candition)。對一個共享資源的讀和寫操作必須是原子化的,換句話說,同一時刻只能有一個 goroutine 對共享資源進(jìn)行讀和寫操作。

// 這個示例程序展示如何在程序里造成競爭狀態(tài)

// 實際上不希望出現(xiàn)這種情況

package main

import (

????"fmt"

????"runtime"

????"sync"

)

var (

????//counter是所有g(shù)oroutine都要增加其值的變量

????counter int

????//wg用來等待程序結(jié)束

????wg.sync.WaitGroup

)

//main是所有Go程序的入口

func main() {

????//計數(shù)加2,表示要等待兩個goroutine

????wg.Add(2)

????

????//創(chuàng)建兩個goroutine

????go printPrime("A")

????go printPrime("B")

????

????//等待goroutine結(jié)束

????wg.Wait()

????

????fmt.Println("Final Counter:", counter)

}

//incCounter增加包里counter變量的值

func incCounter(id int) {

????//在函數(shù)退出時調(diào)用Done來通知main函數(shù)工作已經(jīng)完成

????defer wg.Done()

????

????for count := 0; count < 2; count++ {

????????//捕獲counter的值

????????value := counter

????????

????????//當(dāng)前goroutine從線程退出,并放回到隊列

????????runtime.Gosched()

????????

????????//增加本地value變量的值

????????value++

????????

????????//將改值保存回counter

????????counter = value

????}

}

go build -race // 用競爭檢測器標(biāo)志來編譯程序

./example ???// 運行程序

一種修正代碼、消除競爭狀態(tài)的辦法是,使用 Go 語言提供的鎖機制,來鎖住共享資源,從而保證 goroutine 的同步狀態(tài)。

6.4 鎖住共享資源

6.4.1 原子函數(shù)

原子函數(shù)能夠以很底層的加鎖機制來同步訪問整型變量和指針。我們可以用原子函數(shù)來修正代碼清單創(chuàng)建的競爭狀態(tài)。

// 這個示例程序展示如何使用 atomic 包來提供

// 對數(shù)值類型的安全訪問

package main

import (

????"fmt"

????"runtime"

????"sync"

????"sync/atomic"

)

var (

????//counter是所有g(shù)oroutine都要增加其值的變量

????counter int64

????//wg用來等待程序結(jié)束

????wg.sync.WaitGroup

)

//main是所有Go程序的入口

func main() {

????//計數(shù)加2,表示要等待兩個goroutine

????wg.Add(2)

????

????//創(chuàng)建兩個goroutine

????go incCounter(1)

????go incCounter(2)

????

????//等待goroutine結(jié)束

????wg.Wait()

????//現(xiàn)實最終的值

????fmt.Println("Final Counter:", counter)

}

//incCounter增加包里counter變量的值

func incCounter(id int) {

????//在函數(shù)退出時調(diào)用Done來通知main函數(shù)工作已經(jīng)完成

????defer wg.Done()

????

????for count := 0; count < 2; count++ {

????????//安全地對counter加1

????????atomic.AddInt64(&counter, 1)

????????

????????//當(dāng)前goroutine從線程退出,并放回到隊列

????????runtime.Gosched()

????}

}

Final Counter: 4

現(xiàn)在,程序的第 43 行使用了 atmoic 包的 AddInt64 函數(shù)。這個函數(shù)會同步整型值的加法,方法是強制同一時刻只能有一個 goroutine 運行并完成這個加法操作。當(dāng) goroutine 試圖去調(diào)用任何原子函數(shù)時,這些 goroutine 都會自動根據(jù)所引用的變量做同步處理。現(xiàn)在我們得到了正確的值 4。

另外兩個有用的原子函數(shù)是 LoadInt64 和 StoreInt64 。這兩個函數(shù)提供了一種安全地讀和寫一個整型值的方式。

// 這個示例程序展示如何使用 atomic 包里的

// Store 和 Load 類函數(shù)來提供對數(shù)值類型

package main

import (

????"fmt"

????"sync"

????"sync/atomic"

????"time"

)

var (

????//shutdown是通知正在執(zhí)行的goroutine停止工作的標(biāo)志

????shutdown int64

????

????//wg用來等待程序結(jié)束

????wg.sync.WaitGroup

)

//main是所有Go程序的入口

func main() {

????//計數(shù)加2,表示要等待兩個goroutine

????wg.Add(2)

????

????//創(chuàng)建兩個goroutine

????go doWork("A")

????go doWork("B")

????

????//給定goroutine執(zhí)行的時間

????time.Sleep(1 * time.Second)

????

????//該停止工作了,安全地設(shè)置shutdown標(biāo)志

????fmt.Println("Shutdown Now")

????atomic.StoreInt64(&shutdown, 1)

????

????//等待goroutine結(jié)束

????wg.Wait()

}

//doWork用來模擬執(zhí)行工作的goroutine,

//檢測之前的shutdown標(biāo)志來決定是否提前終止

func doWork(name string) {

????//在函數(shù)退出時調(diào)用Done來通知main函數(shù)工作已經(jīng)完成

????defer wg.Done()

????

????for {

????????fmt.Printf("Doing %s Work\n", name)

????????time.Sleep(250* time.Millisecod)

????????//要停止工作了嗎

????????if ?atomic.AddInt64(&counter, 1) == 1 {

????????????fmt.Printf("Shutting %sl0wen", name)

????????????break

????????}

????}

}

6.4.2 互斥鎖

另一種同步訪問共享資源的方式是使用互斥鎖( mutex )。互斥鎖用于在代碼上創(chuàng)建一個臨界區(qū),保證同一時間只有一個 goroutine 可以執(zhí)行這個臨界區(qū)代碼。

// 這個示例程序展示如何使用互斥鎖來

// 定義一段需要同步訪問的代碼臨界區(qū)

// 資源的同步訪問

package main

import (

????"fmt"

????"runtime"

????"sync"

)

var (

????//counter是所有g(shù)oroutine都要增加其值的變量

????counter int

????

????//wg用來等待程序結(jié)束

????wg.sync.WaitGroup

)

//main是所有Go程序的入口

func main() {

????//計數(shù)加2,表示要等待兩個goroutine

????wg.Add(2)

????

????//創(chuàng)建兩個goroutine

????go incCounter(1)

????go incCounter(2)

????

????//等待goroutine結(jié)束

????wg.Wait()

????fmt.Printf("Final Counter: %d\\n", counter)

}

//incCounter使用互斥鎖來同步并保證安全訪問

//增加包里counter變量的值

func incCounter(id int) {

????//在函數(shù)退出時調(diào)用Done來通知main函數(shù)工作已經(jīng)完成

????defer wg.Done()

????

????for count := 0; count < 2; counter++ {

????????//同一時刻只允許一個goroutine進(jìn)入

????????//這個臨界區(qū)

????????mutex.Lock()

????????{

????????????//捕獲counter的值

????????????value := counter

????????????

????????????//當(dāng)前goroutine從線程退出,并放回到隊列

????????????runtime.Gosched()

????????????

????????????//增加本地value變量的值

????????????value++

????????????

????????????//將該值保存回counter

????????????counter = value

????????}

????????mutex.Unlock()

????????//釋放鎖,允許其他正在等待的goroutine

????????//進(jìn)入臨界區(qū)

????}

}

6.5 通道

在 Go 語言里,你不僅可以使用原子函數(shù)和互斥鎖來保證對共享資源的安全訪

問以及消除競爭狀態(tài),還可以使用通道,通過發(fā)送和接收需要共享的資源,在 goroutine 之間做同步。

使用 make 創(chuàng)建通道

//無緩沖的整型通道

unbuffered := make(chan int)

//有緩沖的整型通道

buffered := make(chan string, 10)

//向通道發(fā)送值

// 通過通道發(fā)送一個字符串

buffered <- "Gopher"

// 從通道接收一個字符串

value := <-buffered

6.5.1 無緩沖的通道

無緩沖的通道(unbuffered channel)是指在接收前沒有能力保存任何值的通道。這種類型的通道要求發(fā)送 goroutine 和接收 goroutine 同時準(zhǔn)備好,才能完成發(fā)送和接收操作。如果兩個 goroutine沒有同時準(zhǔn)備好,通道會導(dǎo)致先執(zhí)行發(fā)送或接收操作的 goroutine 阻塞等待。這種對通道進(jìn)行發(fā)送和接收的交互行為本身就是同步的。其中任意一個操作都無法離開另一個操作單獨存在。

// 這個示例程序展示如何用無緩沖的通道來模擬

// 2 個 goroutine 間的網(wǎng)球比賽

package main

import (

????"fmt"

????"math/rand"

????"sync"

????"time"

)

//wg用來等待程序結(jié)束

var wg sync.WaitGroup

func init() {

????rand.Seed(time.Now().UnixNano())

}

//main是所有Go程序的入口

func main() {

????//創(chuàng)建一個無緩沖的通道

????court := make(chan int)

????

????//計數(shù)加2,表示要等待兩個goroutine

????wg.Add(2)

????

????//啟動兩個選手

????go player("Nadal", court)

????go player("Djokovic", court)

????

????//發(fā)球(將球發(fā)到通道里)

????court <- -1

????

????//等待游戲結(jié)束

????wg.Wait()

}

//player模擬一個選手在打網(wǎng)球

func player(name string, court chan int) {

????//在函數(shù)退出時調(diào)用Done來通知main函數(shù)工作已經(jīng)完成

????defer wg.Done()

????

????for {

????????//等待球被擊打過來

//goroutine 從通道接收數(shù)據(jù),用來表示等待接球。這個接收動作會鎖住//goroutine,直到有數(shù)據(jù)發(fā)送到通道里。

????????ball, ok := <-court

????????if !ok {

????????????//如果通道被關(guān)閉,我們就贏了

????????????fmt.Printf("Player %s Won\n", name)

????????????return

????????}

????????

????????//選隨機數(shù),然后用這個數(shù)來判斷我們是否丟球

????????n := rand.Intn(100)

????????if n%13 == 0 {

????????????fmt.Printf("Player %s Missed\n", name)

????????

????????????//關(guān)閉通道,表示我們輸了

????????????close(court)

????????????return

????????}

????????//顯示擊球數(shù),并將擊球數(shù)加1

????????fmt.Printf("Player %s Hit %d\n", name, ball)

????????ball++

????????

????????//將球打向?qū)κ?/p>

????????court <- ball

????}

}

// 這個示例程序展示如何用無緩沖的通道來模擬

// 4 個 goroutine 間的接力比賽

package main

import (

????"fmt"

????"sync"

????"time"

)

//wg用來等待程序結(jié)束

var wg sync.WaitGroup

//main是所有Go程序的入口

func main() {

????//創(chuàng)建一個無緩沖的通道

????court := make(chan int)

????

????//為最后一位跑步者將計數(shù)加1

????wg.Add(1)

????

????//第一位跑步者持有接力棒

????go Runner(baton)

????

????//開始比賽

????baton <- -1

????

????//等待比賽結(jié)束

????wg.Wait()

}

//Runner模擬一個選手在打網(wǎng)球

func Runner(baton chan int) {

????var newRunner int

????

????//等待接力棒

????runner := <-baton

????

????//開始繞著跑道跑步

????fmt.Printf("Runner %d Running With Baton\n", runner)

????

????//創(chuàng)建下一位跑步者

????if runner != 4 {

????????newRunner = runer + 100

????????fmt.Printf("Runner %d To The Line\n", newRunner)

????????go Runner(baton)

????}

????

????//圍繞跑到跑

????time.Sleep(100 * time.Millisecond)

????

????//比賽結(jié)束了嗎?

????if runner == 4 {

????????????fmt.Printf("Runner %d Finished, Race Over\n", runner)

????????????wg.Done()

????????????return

????????}

????????

????????//將接力棒交給下一位跑步者

????????fmt.Printf("Runner %d Exchange With Runner %d\n",

????????????runner,

????????????newRunner)

????????

????????baton <- newRunner

????}

}

6.5.2 有緩沖的通道

有緩沖的通道(buffered channel)是一種在被接收前能存儲一個或者多個值的通道。這種類型的通道并不強制要求 goroutine 之間必須同時完成發(fā)送和接收。通道會阻塞發(fā)送和接收動作的條件也會不同。只有在通道中沒有要接收的值時,接收動作才會阻塞。只有在通道沒有可用緩沖區(qū)容納被發(fā)送的值時,發(fā)送動作才會阻塞。這導(dǎo)致有緩沖的通道和無緩沖的通道之間的一個很大的不同:無緩沖的通道保證進(jìn)行發(fā)送和接收的 goroutine 會在同一時間進(jìn)行數(shù)據(jù)交換;有緩沖的通道沒有這種保證。

// 這個示例程序展示如何使用

// 有緩沖的通道和固定數(shù)目的

// goroutine 來處理一堆工作

package main

import (

????"fmt"

????"math/rand"

????"sync"

????"time"

)

const (

????numberGoroutines = 4 //要使用的goroutine的數(shù)量

????taskLoad ????????= 10 //要處理的工作的數(shù)量

)

//wg用來等待程序結(jié)束

var wg sync.WaitGroup

//init初始化包,Go語言運行時會在其他代碼執(zhí)行之前

//優(yōu)先執(zhí)行這個函數(shù)

func init() {

????//初始化隨機數(shù)種子

????rand.Seed(time.Now().Unix())

}

//main是所有Go程序的入口

func main() {

????//創(chuàng)建一個有緩沖的通道來管理工作

????task := make(chan string, taskLoad)

????

????//啟動goroutine來處理工作

????wg.Add(numberGoroutines)

????for gr := 1; gr <= numberGoroutines; gr++ {

????????go worker(tasks, gr)

????}

????

????//增加一組要完成的工作

????for post := 1; post <= taskLoad; post++ {

????????tasks <- fmt.Sprintf("Task : %d", post)

????}

????

????//當(dāng)所有工作都處理完時關(guān)閉通道

????//以便所有g(shù)oroutine退出

????close(tasks)

????

????

????//等待所有工作完成

????wg.Wait()

}

//worker作為goroutine啟動處理

//從有緩存的通道傳入的工作

func worker(tasks chan string, worker int) {

????//通知函數(shù)已經(jīng)返回

????defer wg.Done()

????

????for {

????????//等待分配工作

????????task, ok := <-tasks

????????if !ok {

????????????//這意味著通道已經(jīng)空了,并且已被關(guān)閉

????????????fmt.Printf("Worker: %d : Shutting Down\n", worker)

????????????return

????????}

????????

????????//顯示我們開始工作了

????????fmt.Printf("Worker: %d : Started %s\n", worker, task)

????????

????????//隨機等一段時間來模擬工作

????????sleep := rand.Int63n(100)

????????time.Sleep(time.Duration(sleep) * time.Millisecond)

????????

????????//顯示我們完成了工作

????????fmt.Printf("Worker: %d : Completed %s\n", worker, task)

????}

}

當(dāng)通道關(guān)閉后,goroutine 依舊可以從通道接收數(shù)據(jù),但是不能再向通道里發(fā)送數(shù)據(jù)。能夠從已經(jīng)關(guān)閉的通道接收數(shù)據(jù)這一點非常重要,因為這允許通道關(guān)閉后依舊能取出其中緩沖的全部值,而不會有數(shù)據(jù)丟失。從一個已經(jīng)關(guān)閉且沒有數(shù)據(jù)的通道

里獲取數(shù)據(jù),總會立刻返回,并返回一個通道類型的零值。如果在獲取通道時還加入了可選的標(biāo)志,就能得到通道的狀態(tài)信息。

6.6 小結(jié)

并發(fā)是指 goroutine 運行的時候是相互獨立的;使用關(guān)鍵字 go 創(chuàng)建 goroutine 來運行函數(shù);goroutine 在邏輯處理器上執(zhí)行,而邏輯處理器具有獨立的系統(tǒng)線程和運行隊列;競爭狀態(tài)是指兩個或者多個 goroutine 試圖訪問同一個資源;原子函數(shù)和互斥鎖提供了一種防止出現(xiàn)競爭狀態(tài)的辦法;通道提供了一種在兩個 goroutine 之間共享數(shù)據(jù)的簡單方法;無緩沖的通道保證同時交換數(shù)據(jù),而有緩沖的通道不做這種保證。

第7章 并發(fā)模式

本章主要內(nèi)容:控制程序的生命周期;管理可復(fù)用的資源池;創(chuàng)建可以處理任務(wù)的 goroutine 池。

7.1 runner

runner 包用于展示如何使用通道來監(jiān)視程序的執(zhí)行時間,如果程序運行時間太長,也可以用 runner 包來終止程序。

// Gabriel Aszalos 協(xié)助完成了這個示例

// runner 包管理處理任務(wù)的運行和生命周期

package runner

import (

????"errors"

????"os"

????"os/signal"

????"time"

)

//Runner在給定的超時間內(nèi)執(zhí)行一組任務(wù),

//并且在操作系統(tǒng)發(fā)送中斷信號時結(jié)束這些任務(wù)

type Runner struct {

????//interrupt通道報告從操作系統(tǒng)

????//發(fā)送的信號

????interrupt chan os.signal

????

????//complete通道報告處理任務(wù)已經(jīng)完成

????complete chan error

????

????//timeout報告處理任務(wù)已經(jīng)超時

????timeout <-chan time.time

????

????//task持有一組以索引順序依次執(zhí)行的

????//函數(shù)

????task []func(int)

}

//ErrTimeout會在任務(wù)執(zhí)行超時時返回

var ErrTimeout = errors.New("received timeout")

//ErrTimeout會在接收到操作系統(tǒng)的事件時返回

var ErrInterrupt = errors.New("received interrupt")

//New返回一個新的準(zhǔn)備使用的Runner

func New(d time.Duration) *Runner {

????return &Runner{

????????interrupt: make(chan os.Signal, 1),

????????complete: make(chan error),

????????timeout: time.After(d),

????}

}

//Add將一個任務(wù)附加到Runner上。這個任務(wù)是一個

//接收一個int類型的ID作為參數(shù)的函數(shù)

func (r *Runner) Add(tasks ...func(int)) {

????r.tasks = append(r.tasks, tasks...)

}

//Start執(zhí)行所有任務(wù),并監(jiān)視通道事件

func (r *Runner) Start() error {

????//我們希望接收所有中斷信號

????sinal.Notify(r.interrupt, os.Interrupt)

????

????//用不同的goroutine執(zhí)行不同的任務(wù)

????go func() {

????????r.complete <- r.run()

????}()

????

????select {

????//當(dāng)任務(wù)處理完成時發(fā)出的信號

????case err := <-r.complete:

????????return err

????//當(dāng)任務(wù)處理程序運行超時發(fā)出的信號

????case <-r.timeout:

????????return ErrTimeout

????}

}

//run執(zhí)行每一個已注冊的任務(wù)

func (r *Runner) run() error {

????for id, task := range r.tasks {

????????//檢測操作系統(tǒng)的中斷信號

????????if r.gotInterrupt() {

????????????return ErrInterrupt

????????}

????????

????????//執(zhí)行已注冊的新任務(wù)

????????task(id)

????}

????

????return nil

}

//gotInterrupt驗證是否接收到了中斷信號

func (r *Runner) gotInterrupt() bool {

????select {

????//當(dāng)中斷事件被觸發(fā)時發(fā)出的信號

????case <-r.interrupt:

????????//停止接收后續(xù)的任何信號

????????sinal.Stop(r.interrupt)

????????return true

????

????//繼續(xù)正常運行

????default:

????????return false

????}

}

程序展示了依據(jù)調(diào)度運行的無人值守的面向任務(wù)的程序,及其所使用的并發(fā)模式。在設(shè)計上,可支持以下終止點:程序可以在分配的時間內(nèi)完成工作,正常終止;程序沒有及時完成工作,“自殺”;接收到操作系統(tǒng)發(fā)送的中斷事件,程序立刻試圖清理狀態(tài)并停止工作。

// 這個示例程序演示如何使用通道來監(jiān)視
// 程序運行的時間,以在程序運行時間過長
// 時如何終止程序
package main

import (
???"log"
???"time"

???"github.com/goinaction/code/chapter7/patterns/runner"
???"os"
)

//timeout規(guī)定了必須在多少秒內(nèi)處理完成
const timeout = 3 * time.Second

//main是程序的入口
func main() {
???log.Println("Starting work.")

???//為本次執(zhí)行分配超時時間
???r := runner.New(timeout)

???//加入要執(zhí)行的任務(wù)
???r.Add(createTask(), createTask(), createTask())

???//執(zhí)行任務(wù)并處理結(jié)果
???if err := r.Start(); err != nil {
??????switch err {
??????case runner.ErrTimeout:
?????????log.Println("Terminating dur to timeout.")
?????????os.Exit(1)
??????case runner.ErrInterrupt:
?????????log.Println("Terminating dur to interrupt.")
?????????os.Exit(2)
??????}
???}
???
???log.Println("Process ended.")
}

//createTask返回一個根據(jù)id
//休眠指定秒數(shù)的示例任務(wù)
func createTask() func(int) {
???return func(id int) {
??????log.Printf("Processor - Task #%d.", id)
??????time.Sleep(time.Duration(id) * time.Second)
???}
}
7.2 pool

pool這個包用于展示如何使用有緩沖的通道實現(xiàn)資源池,來管理可以在任意數(shù)量的goroutine之間共享及獨立使用的資源。在 Go 1.6 及之后的版本中,標(biāo)準(zhǔn)庫里自帶了資源池的實現(xiàn)(sync.Pool)。

7.3 work

work 包的目的是展示如何使用無緩沖的通道來創(chuàng)建一個 goroutine 池,這些 goroutine 執(zhí)行并控制一組工作,讓其并發(fā)執(zhí)行。

// work 包管理一個?goroutine 池來完成工作
package work

import "sync"

//Worker必須滿足接口類型,
//才能使用工作池
type Worker interface {
???Task()
}

//Pool提供一個goroutine池,這個池可以完成
//任何已提交的Worker任務(wù)
type Pool struct {
???work chan Worker
???wg ??sync.WaitGroup
}

//New創(chuàng)建一個新工作池
func New(maxGoroutines int) *Pool {
???p := Pool{
??????work: make(chan Worker),
???}

???p.wg.Add(maxGoroutines)
???for i := 0; i < maxGoroutines; i++ {
??????go func() {
?????????for w := range p.work {
????????????w.Task()
?????????}
?????????p.wg.Done()
??????}()
???}

???return &p
}

//Run提交到工作池
func (p *Pool) Run(w Worker) {
???p.work <- w
}

//Shutdown等待所有goroutine停止工作
func (p *Pool) Shuntdown() {
???close(p.work)
???p.wg.Wait()
}

7.4 小結(jié)

可以使用通道來控制程序的生命周期; 帶 default 分支的 select 語句可以用來嘗試向通道發(fā)送或者接收數(shù)據(jù),而不會阻塞;有緩沖的通道可以用來管理一組可復(fù)用的資源;語言運行時會處理好通道的協(xié)作和同步;使用無緩沖的通道來創(chuàng)建完成工作的 goroutine 池;任何時間都可以用無緩沖的通道來讓兩個 goroutine 交換數(shù)據(jù),在通道操作完成時一定保證對方接收到了數(shù)據(jù)。

  • 標(biāo)準(zhǔn)庫
  • 本章主要內(nèi)容:輸出數(shù)據(jù)以及記錄日志;對 JSON 進(jìn)行編碼和解碼;處理輸入/輸出,并以流的方式處理數(shù)據(jù);讓標(biāo)準(zhǔn)庫里多個包協(xié)同工作。

    8.1 文檔與源代碼

    標(biāo)準(zhǔn)庫里總共有超過100 個包,這些包被分到 38 個類別里。標(biāo)準(zhǔn)庫里的頂級目錄和包:

    archive debug hash mime sort Time bufio encoding html net strconv unicode bytes errors image os strings unsafe compress expvar index path sync container flag io reflect syscall crypto fmt log regexp testing database go math runtime text

    8.2.1 log 包

    記錄日志的目的是跟蹤程序什么時候在什么位置做了什么。

    聲明 Ldate 常量

    // 日期: 2009/01/23

    Ldate = 1 << iota

    關(guān)鍵字 iota 在常量聲明區(qū)里有特殊的作用。這個關(guān)鍵字讓編譯器為每個常量復(fù)制相同的表達(dá)式,直到聲明區(qū)結(jié)束,或者遇到一個新的賦值語句。關(guān)鍵字 iota 的另一個功能是, iota 的初始值為 0,之后 iota 的值在每次處理為常量后,都會自增 1。

    初始完 log 包后,可以看一下 main() 函數(shù),看它是是如何寫消息的。

    func main() {
    ???//Println寫到標(biāo)準(zhǔn)日志記錄器
    ???log.Println("message")

    ???//Fatalln在調(diào)用Println()之后會接著調(diào)用os.Exit(1)
    ???log.Fatalln("fatal message")
    ???
    ???//Panicln在調(diào)用Println()之后會接著調(diào)用panic()
    ???log.Panicln("panic message")
    }

    8.2.2 定制的日志記錄器

    要想創(chuàng)建一個定制的日志記錄器,需要創(chuàng)建一個 Logger 類型值。可以給每個日志記錄器配置一個單獨的目的地,并獨立設(shè)置其前綴和標(biāo)志。

    // 這個示例程序展示如何創(chuàng)建定制的日志記錄器
    package main

    import (
    ???"io"
    ???"io/ioutil"
    ???"log"
    ???"os"
    ???"sync"
    )

    var (
    ???Trace ??*log.Logger // 記錄所有日志
    ???Info ???*log.Logger // 重要的信息
    ???Warning *log.Logger // 需要注意的信息
    ???Error ??*log.Logger // 非常嚴(yán)重的問題
    )

    func init() {
    ???filr, err := os.OpenFile("errors.txt",
    ??????os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0666)
    ???if err != nil {
    ??????log.Fatalln("Failed to open error log file:", err)
    ???}

    ???Trace = log.New(ioutil.Discard,
    ??????"TRACE:",
    ?????????log.Ldate|log.Ltime|log.Lshortfile)

    ???Info = log.New(os.Stdout,
    ??????"INFO:",
    ??????log.Ldate|log.Ltime|log.Lshortfile)
    ???Warning = log.New(os.Stdout,
    ??????"Warning:",
    ??????log.Ldate|log.Ltime|log.Lshortfile)
    ???Error = log.New(os.Stdout,
    ??????"Error:",
    ??????log.Ldate|log.Ltime|log.Lshortfile)
    }

    func main() {
    ???Trace.Println("I have something standard to say")
    ???Info.Println("Special Information")
    ???Warning.Println("There is something you need to know about")
    ???Error.Println("Something has failed")
    }

    8.3 編碼 / 解碼

    8.3.1 解碼 JSON

    使用 json 包的 NewDecoder 函數(shù)以及 Decode方法進(jìn)行解碼。如果要處理來自網(wǎng)絡(luò)響應(yīng)或者文件的 JSON,那么一定會用到這個函數(shù)及方法。

    // 這個示例程序展示如何使用?json 包和?NewDecoder 函數(shù)
    // 來解碼?JSON 響應(yīng)
    package main

    import (
    ???"net/http"
    ???"log"
    ???"encoding/json"
    ???"fmt"
    )

    type (
    ???//gResult映射從搜索拿到的結(jié)果文檔
    ???gResult struct {
    ??????GsearchResultClass string 'json:"GsearchResultClass"'
    ??????unescapedURL ?????string 'json:"unescapedURL"'
    ??????URL ?????????????string 'json:"url"'
    ??????VisibleURL ??????????string 'json:"VisibleUrl"'
    ??????CacheURl ??????????string 'json:"cacheUrl"'
    ??????Title ?????????????string 'json:"title"'
    ??????TitleNoFormatting ?string 'json:"titleNoFormatting"'
    ??????Content ???????????string 'json:"content"'
    ???}

    ???//gResponse包含頂級的文檔
    ???gResponse struct {
    ??????ResponseData struct {
    ?????????Results []gResult 'json:"results"'
    ??????} 'json:"responseData"'
    ???}
    )
    func main() {
    ???uri := "http://ajax.googleapis.com/ajax/services/search/web?v=1.0&rsz=8&q=golang"

    ???//Google發(fā)起搜索
    ???resp, err := http.Get(uri)
    ???if err != nil {
    ??????log.Println("ERROR:", err)
    ??????return
    ???}
    ???defer resp.Body.Close()
    ???
    ???//JSON響應(yīng)解碼到結(jié)構(gòu)類型
    ???var gr gResponse
    ???err = json.NewDecoder(resp.Body).Decode(&gr)
    ???if err != nil {
    ??????log.Println("ERROR", err)
    ??????return
    ???}
    ???
    ???fmt.Println(gr)
    }

    // 這個示例程序展示如何解碼?JSON 字符串
    package main

    import (
    ???"log"
    ???"encoding/json"
    ???"fmt"
    )

    //Contact結(jié)構(gòu)代表我們的JSON字符串
    type Contact struct {
    ????Name ???string 'json:"name"'
    ????Title ??string 'json:"title"'
    ????Contact struct {
    ??????Home string ‘json:"home"
    ??????Cell string 'json:"cell"'
    ?????} 'json:"contact"'
    }

    //JSON包含用于反序列化的演示字符串
    var JSON = '{
    ???"name": "Gopher",
    ???"title":"programmer",
    ???"contact":{
    ??????"home": "415.333.3333",
    ??????"cell": "415.555.5555"
    ???}
    }'

    func main() {
    ???//JSON字符串反序列化到變量
    ???var c Contact
    ???err := json.Unmarshal([]byte(JSON), &c)
    ???if err != nil {
    ??????log.Println("ERROR:", err)
    ??????return
    ???}

    ???fmt.Println(c)
    }

    有時,無法為 JSON 的格式聲明一個結(jié)構(gòu)類型,而是需要更加靈活的方式來處理 JSON 文檔。在這種情況下,可以將 JSON 文檔解碼到一個 map 變量中。

    // 這個示例程序展示如何解碼?JSON 字符串
    package main

    import (
    ???"log"
    ???"encoding/json"
    ???"fmt"
    )

    //JSON包含用于反序列化的演示字符串
    var JSON = '{
    ???"name": "Gopher",
    ???"title":"programmer",
    ???"contact":{
    ??????"home": "415.333.3333",
    ??????"cell": "415.555.5555"
    ???}
    }'

    func main() {
    ???//JSON字符串反序列化到map變量
    ???var c map[string]interface{}
    ???err := json.Unmarshal([]byte(JSON), &c)
    ???if err != nil {
    ??????log.Println("ERROR:", err)
    ??????return
    ???}

    ???fmt.Println("Name:", c["name"])
    ???fmt.Println("Title:", c["title"])
    ???fmt.Println("Contact")
    ???fmt.Println("H:", c["contact"].(map[string]interface{})["home"])
    ???fmt.Println("C:", c["contact"].(map[string]interface{})["cell"])
    }

    8.3.2 編碼 JSON

    我們要學(xué)習(xí)的處理 JSON 的第二個方面是,使用 json 包的 MarshalIndent 函數(shù)進(jìn)行編碼。這個函數(shù)可以很方便地將 Go 語言的 map 類型的值或者結(jié)構(gòu)類型的值轉(zhuǎn)換為易讀格式的 JSON 文檔。 序列化 (marshal)是指將數(shù)據(jù)轉(zhuǎn)換為 JSON 字符串的過程。

    // 這個示例程序展示如何序列化?JSON 字符串
    package main

    import (
    ???"encoding/json"
    ???"log"
    ???"fmt"
    )

    func main() {
    ???//創(chuàng)建一個保存鍵值對的映射
    ???c := make(map[string]interface{})
    ???c["name"] = "Gopher"
    ???c["title"] = "programmer"
    ???c["contact"] = map[string]interface{}{
    ??????"home": "415.333.3333"
    ??????"cell": "415.555.5555"
    ???}
    ???
    ???//將這個映射序列化到JSON字符串
    ???data, err := json.MarshalIndent(c, "", " ??")
    ???if err != nil {
    ??????log.Println("ERROR:", err)
    ??????return
    ???}
    ???
    ???fmt.Println(string(data))
    }

    // MarshalIndent 很像 Marshal,只是用縮進(jìn)對輸出進(jìn)行格式化

    func MarshalIndent(v interface{}, prefix, indent string) ([]byte, error) {

    在 MarshalIndent 函 數(shù) 里 再 一 次 看 到 使 用 了 空 接 口 類 型 interface{} 。 函 數(shù)MarshalIndent 會使用反射來確定如何將 map 類型轉(zhuǎn)換為 JSON 字符串。

    8.4 輸入和輸出

    8.4.1 Writer 和 Reader 接口

    // 這個示例程序展示來自不同標(biāo)準(zhǔn)庫的不同函數(shù)是如何
    // 使用?io.Writer 接口的
    package main

    import (
    ???"bytes"
    ???"fmt"
    ???"os"
    )

    //main是應(yīng)用程序的入口
    func main() {
    ???//創(chuàng)建一個Buffer值,并將一個字符串寫入Buffer
    ???//使用實現(xiàn)io.WriterWrite方法
    ???var b bytes.Buffer
    ???b.Write([]byte("Hello "))
    ???
    ???//使用Fprintf來將一個字符串拼接到Buffer
    ???//bytes.Buffer的地址作為io.Writer類型值傳入
    ???fmt.Fprintf(&b, "World!")
    ???
    ???//Buffer的內(nèi)容輸出到標(biāo)準(zhǔn)輸出設(shè)備
    ???//os.File值的地址作為io.Writer類型值傳入
    ???b.WriteTo(os.Stdout)
    }

    8.4.3?簡單的 curl

    curl這個工具可以對指定的 URL 發(fā)起 HTTP 請求,并保存返回的內(nèi)容。

    // 這個示例程序展示來自不同標(biāo)準(zhǔn)庫的不同函數(shù)是如何
    // 使用?io.Writer 接口的
    package main

    import (
    ???"net/http"
    ???"os"
    ???"log"
    ???"io"
    )
    //main是應(yīng)用程序的入口
    func main() {
    ???//這里的r是一個響應(yīng),r.Bodyio.Reader
    ???r, err := ?http.Get(os.Args[1])
    ???if err != nil {
    ??????log.Fatalln(err)
    ???}

    ???//創(chuàng)建文件來保存響應(yīng)內(nèi)容
    ???file, err := os.Create(os.Args[2])
    ???if err != nil {
    ??????log.Fatalln(err)
    ???}
    ???defer file.Close()

    ???//使用MultiWriter,這樣就可以同時向文件和標(biāo)準(zhǔn)輸出設(shè)備
    ???//進(jìn)行寫操作
    ???dest := io.MultiWriter(os.Stdout, file)

    ???//從響應(yīng)的結(jié)果讀出響應(yīng)的內(nèi)容,并寫道兩個目的地
    ???io.Copy(dest, r.Body)
    ???if err := r.Body.Close(); err != nil {
    ??????log.Println(err)
    ???}
    }

    8.5?小結(jié)

    標(biāo)準(zhǔn)庫有特殊的保證,并且被社區(qū)廣泛應(yīng)用;使用標(biāo)準(zhǔn)庫的包會讓你的代碼更易于管理,別人也會更信任你的代碼;100 余個包被合理組織,分布在 38 個類別里;標(biāo)準(zhǔn)庫里的 log 包擁有記錄日志所需的一切功能;標(biāo)準(zhǔn)庫里的 xml 和 json 包讓處理這兩種數(shù)據(jù)格式變得很簡單;io 包支持以流的方式高效處理數(shù)據(jù);接口允許你的代碼組合已有的功能;閱讀標(biāo)準(zhǔn)庫的代碼是熟悉 Go 語言習(xí)慣的好方法。

    第9章 測試和性能

    本章主要內(nèi)容:編寫單元測試來驗證代碼的正確性;使用 httptest 來模擬基于 HTTP 的請求和響應(yīng);使用示例代碼來給包寫文檔;通過基準(zhǔn)測試來檢查性能。

    9.1 單元測試

    單元測試是用來測試包或者程序的一部分代碼或者一組代碼的函數(shù)。測試的目的是確認(rèn)目標(biāo)代碼在給定的場景下,有沒有按照期望工作。另外一些單元測試可能會測試負(fù)向路徑的場景,保證代碼不僅會產(chǎn)生錯誤,而且是預(yù)期的錯誤。

    在 Go 語言里有幾種方法寫單元測試。基礎(chǔ)測試(basic test)只使用一組參數(shù)和結(jié)果來測試一段代碼。表組測試(table test)也會測試一段代碼,但是會使用多組參數(shù)和結(jié)果進(jìn)行測試。也可以使用一些方法來模仿(mock)測試代碼需要使用到的外部資源,如數(shù)據(jù)庫或者網(wǎng)絡(luò)服務(wù)器。

    // 這個示例程序展示如何寫基礎(chǔ)單元測試
    package listing01

    import (
    ???"net/http"
    ???"testing"
    )

    const chechMark = "\u2713"
    const ballotX = "\u2717"

    //TestDownload確認(rèn)http包的Get函數(shù)可以下載內(nèi)容
    func TestDownload(t *testing.T) {
    ???url := "http://www.goinggo.net/feeds/posts/default?alt=rss"
    ???statusCode := 200

    ???t.Log("Given the need to test downloadig content.")
    ???{
    ??????t.Logf("\tWhen checking \"%s\" for status code \"%d\"",
    ?????????url, statusCode)
    ??????{
    ?????????resp, err := http.Get(url)
    ?????????if err != nil {
    ????????????t.Fatal("\t\tShould be able to make the Get call.",
    ???????????????ballotX, err)
    ?????????}
    ?????????t.Log("\t\tShould be able to make the Get call.",
    ????????????checkMark)

    ?????????defer resp.Body.Close()

    ?????????if resp.StatusCode == statusCode {
    ????????????t.Logf("\t\tShould receive a \"%d\" status. %v",
    ???????????????statusCode, checkMark)
    ?????????} else {
    ????????????t.Errorf("\t\tShould receive a \"%d\" status. %v %v",
    ???????????????statusCode, ballotX, resp.StatusCode")
    ?????????}
    ??????}
    ???}
    }

    展示了測試 http 包的 Get 函數(shù)的單元測試。測試的內(nèi)容是確保可以從網(wǎng)絡(luò)正常下載 goinggo.net 的 RSS 列表。通過調(diào)用 go test -v 來運行這個測試( -v 表示提供冗余輸出)。

    9.1.2 表組測試

    如果測試可以接受一組不同的輸入并產(chǎn)生不同的輸出的代碼,那么應(yīng)該使用表組測試的方法進(jìn)行測試。

    9.1.3 模仿調(diào)用

    標(biāo)準(zhǔn)庫包含一個名為 httptest 的包,它讓開發(fā)人員可以模仿基于HTTP 的網(wǎng)絡(luò)調(diào)用。

    9.3 基準(zhǔn)測試

    基準(zhǔn)測試是一種測試代碼性能的方法。

    9.4 小結(jié)

    測試功能被內(nèi)置到 Go 語言中,Go 語言提供了必要的測試工具;go test 工具用來運行測試;測試文件總是以_test.go 作為文件名的結(jié)尾;表組測試是利用一個測試函數(shù)測試多組值的好辦法;包中的示例代碼,既能用于測試,也能用于文檔;基準(zhǔn)測試提供了探查代碼性能的機制。

    總結(jié)

    以上是生活随笔為你收集整理的《Go语言实战》William Kennedy中文版学习笔记的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    人妻与老人中文字幕 | 国产人妻人伦精品1国产丝袜 | 亚洲欧美日韩国产精品一区二区 | 亚洲高清偷拍一区二区三区 | 一本精品99久久精品77 | 亚洲国产欧美在线成人 | 免费无码肉片在线观看 | 无码人妻丰满熟妇区五十路百度 | av无码不卡在线观看免费 | 老子影院午夜精品无码 | 18禁止看的免费污网站 | 亚洲男人av天堂午夜在 | 未满成年国产在线观看 | 久久精品无码一区二区三区 | 国产农村妇女aaaaa视频 撕开奶罩揉吮奶头视频 | 最新版天堂资源中文官网 | 免费人成网站视频在线观看 | 福利一区二区三区视频在线观看 | 国产精品资源一区二区 | 国产激情无码一区二区app | 日本xxxx色视频在线观看免费 | 夜夜夜高潮夜夜爽夜夜爰爰 | 性做久久久久久久免费看 | 丰满少妇弄高潮了www | 97无码免费人妻超级碰碰夜夜 | 婷婷六月久久综合丁香 | 国产三级精品三级男人的天堂 | 日韩亚洲欧美中文高清在线 | 女人被男人爽到呻吟的视频 | 午夜福利试看120秒体验区 | 亚洲国产欧美国产综合一区 | 国产无遮挡吃胸膜奶免费看 | 日韩av无码中文无码电影 | 偷窥日本少妇撒尿chinese | 亚洲一区二区三区含羞草 | 日本www一道久久久免费榴莲 | 一二三四在线观看免费视频 | 中文字幕人成乱码熟女app | 天天做天天爱天天爽综合网 | 亚洲成a人片在线观看日本 | 国产精品美女久久久久av爽李琼 | 欧洲欧美人成视频在线 | 日韩亚洲欧美精品综合 | 国产办公室秘书无码精品99 | 国产精品香蕉在线观看 | 亚洲高清偷拍一区二区三区 | 久久久久久国产精品无码下载 | 乱人伦人妻中文字幕无码 | 欧美变态另类xxxx | 亚洲色偷偷男人的天堂 | 欧美黑人巨大xxxxx | 成人无码精品一区二区三区 | 欧美 日韩 亚洲 在线 | 国产另类ts人妖一区二区 | 三上悠亚人妻中文字幕在线 | 狠狠色色综合网站 | 国产精品va在线播放 | 国精品人妻无码一区二区三区蜜柚 | 亚洲日韩乱码中文无码蜜桃臀网站 | 最近的中文字幕在线看视频 | 欧美日韩一区二区综合 | 内射爽无广熟女亚洲 | 欧美 丝袜 自拍 制服 另类 | 日日天日日夜日日摸 | 婷婷色婷婷开心五月四房播播 | 亚洲va欧美va天堂v国产综合 | 日本一区二区三区免费高清 | 亚洲熟妇色xxxxx欧美老妇 | 少妇人妻大乳在线视频 | 欧美变态另类xxxx | 中文字幕乱码人妻二区三区 | 国产亚洲精品久久久久久久 | 人妻少妇精品无码专区二区 | 精品无码国产自产拍在线观看蜜 | 色综合久久久久综合一本到桃花网 | 水蜜桃亚洲一二三四在线 | 欧洲欧美人成视频在线 | 无码av免费一区二区三区试看 | 国产亚洲精品久久久久久久久动漫 | 粗大的内捧猛烈进出视频 | 又色又爽又黄的美女裸体网站 | 纯爱无遮挡h肉动漫在线播放 | 国产一精品一av一免费 | 最新国产乱人伦偷精品免费网站 | 免费看少妇作爱视频 | 亚洲国产成人av在线观看 | 日韩欧美中文字幕公布 | 色欲综合久久中文字幕网 | 波多野结衣 黑人 | 久久亚洲中文字幕精品一区 | 免费无码av一区二区 | 亚洲中文字幕在线无码一区二区 | 中文亚洲成a人片在线观看 | 亚洲中文字幕无码中字 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 色婷婷欧美在线播放内射 | 亚洲国精产品一二二线 | 国产精品久久久久7777 | 内射欧美老妇wbb | 5858s亚洲色大成网站www | 国产 精品 自在自线 | 老熟女重囗味hdxx69 | 久久久精品国产sm最大网站 | 亚洲狠狠婷婷综合久久 | 国产精品无码永久免费888 | 国产日产欧产精品精品app | 无遮挡国产高潮视频免费观看 | 国产乱子伦视频在线播放 | 美女张开腿让人桶 | 少妇久久久久久人妻无码 | 亚洲区小说区激情区图片区 | 天堂在线观看www | 久久久久免费看成人影片 | 成 人 免费观看网站 | 亚洲va欧美va天堂v国产综合 | 国产人成高清在线视频99最全资源 | 女人被男人爽到呻吟的视频 | 亚洲一区二区三区无码久久 | 国产成人久久精品流白浆 | 国产99久久精品一区二区 | 亚洲精品国偷拍自产在线麻豆 | 国产九九九九九九九a片 | 国产精品嫩草久久久久 | 无码国内精品人妻少妇 | av小次郎收藏 | 国産精品久久久久久久 | 啦啦啦www在线观看免费视频 | 亚洲成a人片在线观看无码 | 欧美精品国产综合久久 | 久久国产自偷自偷免费一区调 | 两性色午夜免费视频 | 少妇无码一区二区二三区 | 亚洲综合久久一区二区 | 青青草原综合久久大伊人精品 | 荫蒂添的好舒服视频囗交 | 中文字幕日产无线码一区 | 日本大乳高潮视频在线观看 | 任你躁在线精品免费 | 亚洲成av人影院在线观看 | 桃花色综合影院 | 日日鲁鲁鲁夜夜爽爽狠狠 | 亚洲成在人网站无码天堂 | 日韩亚洲欧美中文高清在线 | 亚洲一区二区三区无码久久 | 无套内谢老熟女 | 日韩视频 中文字幕 视频一区 | 色五月丁香五月综合五月 | 日日天干夜夜狠狠爱 | 啦啦啦www在线观看免费视频 | 黄网在线观看免费网站 | 性欧美大战久久久久久久 | 日日噜噜噜噜夜夜爽亚洲精品 | 亚洲第一网站男人都懂 | 国产乱人伦偷精品视频 | 亚洲人亚洲人成电影网站色 | 国产卡一卡二卡三 | 亚洲一区二区三区播放 | 男女爱爱好爽视频免费看 | 日韩在线不卡免费视频一区 | 午夜精品久久久久久久久 | 日日摸日日碰夜夜爽av | 午夜精品一区二区三区的区别 | 中文字幕色婷婷在线视频 | 精品人妻人人做人人爽夜夜爽 | 精品国产福利一区二区 | 国产精品理论片在线观看 | 欧美日韩一区二区综合 | 扒开双腿吃奶呻吟做受视频 | 午夜熟女插插xx免费视频 | 水蜜桃亚洲一二三四在线 | 四十如虎的丰满熟妇啪啪 | 麻豆成人精品国产免费 | 精品无人国产偷自产在线 | 99久久无码一区人妻 | 久久久精品成人免费观看 | 国产精品久久久久影院嫩草 | 内射白嫩少妇超碰 | 亚洲毛片av日韩av无码 | 欧美刺激性大交 | 永久黄网站色视频免费直播 | 国产免费久久久久久无码 | 亚洲乱码日产精品bd | 国产色xx群视频射精 | 亚洲精品中文字幕久久久久 | 中文字幕无码日韩欧毛 | 欧美乱妇无乱码大黄a片 | 亚洲欧美国产精品专区久久 | 欧美日本免费一区二区三区 | 性欧美videos高清精品 | 熟女俱乐部五十路六十路av | 久久国产精品_国产精品 | 露脸叫床粗话东北少妇 | 亚洲欧洲日本无在线码 | 久久99精品国产.久久久久 | 日本精品人妻无码77777 天堂一区人妻无码 | 国产精品沙发午睡系列 | 蜜桃无码一区二区三区 | 午夜无码人妻av大片色欲 | 日本在线高清不卡免费播放 | 亚洲国产午夜精品理论片 | 国产精品久久久久久无码 | 成熟人妻av无码专区 | 99久久久无码国产aaa精品 | 精品国精品国产自在久国产87 | 亚洲啪av永久无码精品放毛片 | 精品国产成人一区二区三区 | 久激情内射婷内射蜜桃人妖 | 人妻少妇精品久久 | 久久精品国产大片免费观看 | 亚洲色大成网站www | 亚洲综合在线一区二区三区 | 在线精品亚洲一区二区 | 亚洲综合无码久久精品综合 | 免费看男女做好爽好硬视频 | 亚洲中文字幕无码中文字在线 | 天天燥日日燥 | 动漫av一区二区在线观看 | 97精品人妻一区二区三区香蕉 | 天天摸天天碰天天添 | 亚洲 另类 在线 欧美 制服 | 日韩亚洲欧美中文高清在线 | 人人妻人人澡人人爽精品欧美 | 强辱丰满人妻hd中文字幕 | 图片小说视频一区二区 | 欧美人与禽猛交狂配 | 天天拍夜夜添久久精品 | 亚洲国产精品成人久久蜜臀 | 奇米影视7777久久精品 | 国产午夜视频在线观看 | 国产精品久久久av久久久 | 76少妇精品导航 | 在线 国产 欧美 亚洲 天堂 | 熟女体下毛毛黑森林 | 又粗又大又硬毛片免费看 | 国产成人无码区免费内射一片色欲 | 亚洲中文字幕在线观看 | 天堂一区人妻无码 | 国产农村乱对白刺激视频 | 欧美一区二区三区视频在线观看 | 精品国偷自产在线视频 | 欧洲欧美人成视频在线 | 久久综合九色综合97网 | 久久人人爽人人爽人人片ⅴ | 亚洲乱码国产乱码精品精 | 亚洲精品无码人妻无码 | 午夜嘿嘿嘿影院 | 真人与拘做受免费视频一 | 日韩欧美群交p片內射中文 | 精品无码一区二区三区爱欲 | 美女极度色诱视频国产 | 亚洲人成影院在线无码按摩店 | 熟女俱乐部五十路六十路av | 天下第一社区视频www日本 | 无码免费一区二区三区 | 亚洲成色www久久网站 | 精品无码一区二区三区爱欲 | 中文字幕无码热在线视频 | 国产无套内射久久久国产 | 色五月五月丁香亚洲综合网 | 色欲综合久久中文字幕网 | 久久久久免费精品国产 | 国产精品欧美成人 | 99精品视频在线观看免费 | 色噜噜亚洲男人的天堂 | 久久久久久久人妻无码中文字幕爆 | 亚洲国产日韩a在线播放 | √天堂资源地址中文在线 | 亚洲va中文字幕无码久久不卡 | 性史性农村dvd毛片 | 男人扒开女人内裤强吻桶进去 | 日日碰狠狠丁香久燥 | 在线а√天堂中文官网 | 精品熟女少妇av免费观看 | 日韩欧美中文字幕在线三区 | 久青草影院在线观看国产 | 5858s亚洲色大成网站www | 日本va欧美va欧美va精品 | 99riav国产精品视频 | 中文字幕 人妻熟女 | 国产精品亚洲一区二区三区喷水 | 亚洲gv猛男gv无码男同 | 人妻中文无码久热丝袜 | 亚洲精品国偷拍自产在线观看蜜桃 | 国产无套内射久久久国产 | 国产精品成人av在线观看 | 亚洲精品国产精品乱码视色 | 无码精品国产va在线观看dvd | 一本色道久久综合亚洲精品不卡 | 国产高清av在线播放 | 久久伊人色av天堂九九小黄鸭 | 亚洲无人区午夜福利码高清完整版 | 精品人妻中文字幕有码在线 | 最近中文2019字幕第二页 | 亚洲国产高清在线观看视频 | 久精品国产欧美亚洲色aⅴ大片 | 国产九九九九九九九a片 | 色综合天天综合狠狠爱 | 国产成人精品一区二区在线小狼 | 六月丁香婷婷色狠狠久久 | 男女下面进入的视频免费午夜 | 人妻有码中文字幕在线 | 亚洲小说春色综合另类 | 全黄性性激高免费视频 | 人妻夜夜爽天天爽三区 | 成人欧美一区二区三区黑人 | 午夜无码人妻av大片色欲 | 国产明星裸体无码xxxx视频 | 成人免费无码大片a毛片 | 99riav国产精品视频 | 狠狠色噜噜狠狠狠狠7777米奇 | 日韩少妇内射免费播放 | а天堂中文在线官网 | 中文字幕av无码一区二区三区电影 | 乌克兰少妇xxxx做受 | 午夜无码区在线观看 | 动漫av一区二区在线观看 | 日本一区二区三区免费播放 | 国内少妇偷人精品视频 | 色妞www精品免费视频 | 久久99精品国产.久久久久 | 性色欲网站人妻丰满中文久久不卡 | 午夜时刻免费入口 | 伊人久久婷婷五月综合97色 | 国产精品视频免费播放 | 亚洲国产综合无码一区 | 性欧美疯狂xxxxbbbb | 国产精品永久免费视频 | 51国偷自产一区二区三区 | 亚洲日韩乱码中文无码蜜桃臀网站 | 亚拍精品一区二区三区探花 | 日本成熟视频免费视频 | 亚洲成熟女人毛毛耸耸多 | 窝窝午夜理论片影院 | 在线a亚洲视频播放在线观看 | 国产真实夫妇视频 | 无码人妻av免费一区二区三区 | 亚洲成a人片在线观看无码3d | 亚洲成a人片在线观看无码3d | 性欧美疯狂xxxxbbbb | 欧美肥老太牲交大战 | 男人的天堂2018无码 | 亚洲成av人片天堂网无码】 | 少妇性l交大片 | 国产suv精品一区二区五 | www成人国产高清内射 | 亚洲一区二区三区国产精华液 | 精品无码成人片一区二区98 | 久久久久久久女国产乱让韩 | 狠狠噜狠狠狠狠丁香五月 | 亚洲精品一区二区三区四区五区 | 人妻与老人中文字幕 | 国产在线一区二区三区四区五区 | 国精产品一区二区三区 | 亚洲 a v无 码免 费 成 人 a v | 一个人免费观看的www视频 | 老司机亚洲精品影院 | 麻豆人妻少妇精品无码专区 | 亚洲а∨天堂久久精品2021 | 蜜桃臀无码内射一区二区三区 | 特黄特色大片免费播放器图片 | 男女猛烈xx00免费视频试看 | 少妇厨房愉情理9仑片视频 | 精品国产av色一区二区深夜久久 | 亚洲欧洲中文日韩av乱码 | 狠狠躁日日躁夜夜躁2020 | 中文字幕色婷婷在线视频 | 久久综合给合久久狠狠狠97色 | aⅴ亚洲 日韩 色 图网站 播放 | 亚洲国产精品无码一区二区三区 | 欧洲熟妇精品视频 | 国产精品igao视频网 | 樱花草在线播放免费中文 | аⅴ资源天堂资源库在线 | 国产精品久久国产精品99 | 人妻熟女一区 | 欧洲熟妇色 欧美 | 无码精品人妻一区二区三区av | 久久天天躁夜夜躁狠狠 | 亚洲va欧美va天堂v国产综合 | 99久久人妻精品免费一区 | 99麻豆久久久国产精品免费 | 国产人妻人伦精品1国产丝袜 | 波多野结衣av在线观看 | 亚洲gv猛男gv无码男同 | 樱花草在线播放免费中文 | 无码吃奶揉捏奶头高潮视频 | 国内精品人妻无码久久久影院蜜桃 | 99精品视频在线观看免费 | 麻豆av传媒蜜桃天美传媒 | 99国产欧美久久久精品 | 国产真实乱对白精彩久久 | 亚洲午夜久久久影院 | 精品一区二区三区无码免费视频 | 在线精品国产一区二区三区 | 国产精品.xx视频.xxtv | 激情人妻另类人妻伦 | 国产精华av午夜在线观看 | 天天摸天天透天天添 | 少妇无码一区二区二三区 | 大地资源网第二页免费观看 | 精品国产青草久久久久福利 | 亚洲 日韩 欧美 成人 在线观看 | 欧洲精品码一区二区三区免费看 | 97久久精品无码一区二区 | 国内少妇偷人精品视频 | 伊人久久大香线蕉午夜 | 鲁鲁鲁爽爽爽在线视频观看 | 欧美黑人性暴力猛交喷水 | 精品国产aⅴ无码一区二区 | 亚洲国产午夜精品理论片 | 亚洲精品综合一区二区三区在线 | 中文字幕无码乱人伦 | 夜精品a片一区二区三区无码白浆 | 影音先锋中文字幕无码 | 亚洲国产精品无码一区二区三区 | 国产精品高潮呻吟av久久4虎 | 精品国产av色一区二区深夜久久 | 久久久中文久久久无码 | 天天摸天天碰天天添 | 麻豆国产丝袜白领秘书在线观看 | 欧美激情内射喷水高潮 | 午夜福利一区二区三区在线观看 | 水蜜桃亚洲一二三四在线 | 久久精品国产精品国产精品污 | 国产无套内射久久久国产 | 青青久在线视频免费观看 | 亚洲人成无码网www | 日韩少妇内射免费播放 | 国产精品高潮呻吟av久久 | 国产乱人偷精品人妻a片 | 中文字幕无码av波多野吉衣 | 国产国产精品人在线视 | 7777奇米四色成人眼影 | 精品人妻人人做人人爽夜夜爽 | av无码电影一区二区三区 | 丰满少妇弄高潮了www | 又湿又紧又大又爽a视频国产 | 日本护士毛茸茸高潮 | 日本护士xxxxhd少妇 | 亚洲精品午夜国产va久久成人 | 国产激情综合五月久久 | www成人国产高清内射 | 又黄又爽又色的视频 | 久久亚洲精品成人无码 | 荫蒂被男人添的好舒服爽免费视频 | 亚洲另类伦春色综合小说 | 亚洲精品国产精品乱码视色 | 国产成人一区二区三区在线观看 | 在线观看国产午夜福利片 | 捆绑白丝粉色jk震动捧喷白浆 | 久久久久久av无码免费看大片 | 国产偷自视频区视频 | 玩弄人妻少妇500系列视频 | 99视频精品全部免费免费观看 | 国产色在线 | 国产 | 亚洲乱码国产乱码精品精 | 久久精品国产99久久6动漫 | 亚洲综合伊人久久大杳蕉 | 国产99久久精品一区二区 | 男人扒开女人内裤强吻桶进去 | 久久精品中文字幕一区 | 蜜臀aⅴ国产精品久久久国产老师 | www成人国产高清内射 | 一本一道久久综合久久 | 亚洲中文字幕av在天堂 | 亚洲精品一区三区三区在线观看 | 亚洲小说图区综合在线 | 国产97在线 | 亚洲 | 国产精品无码一区二区桃花视频 | 理论片87福利理论电影 | 亚洲色www成人永久网址 | 色婷婷综合中文久久一本 | 午夜肉伦伦影院 | 正在播放东北夫妻内射 | 国产热a欧美热a在线视频 | 精品亚洲成av人在线观看 | 亚洲精品无码国产 | 免费播放一区二区三区 | 亚洲中文字幕无码中字 | 在线看片无码永久免费视频 | 性生交大片免费看女人按摩摩 | 俺去俺来也在线www色官网 | 丁香花在线影院观看在线播放 | 又紧又大又爽精品一区二区 | 日韩亚洲欧美精品综合 | 九九久久精品国产免费看小说 | 欧美乱妇无乱码大黄a片 | 久久无码中文字幕免费影院蜜桃 | 国产在线一区二区三区四区五区 | 捆绑白丝粉色jk震动捧喷白浆 | 国产精品久免费的黄网站 | 国产内射爽爽大片视频社区在线 | 国产在线一区二区三区四区五区 | 亚洲国产高清在线观看视频 | 亚洲乱码中文字幕在线 | 国产av剧情md精品麻豆 | 给我免费的视频在线观看 | 久久久久久a亚洲欧洲av冫 | 强伦人妻一区二区三区视频18 | 免费播放一区二区三区 | 永久免费观看国产裸体美女 | 亚无码乱人伦一区二区 | 国产午夜福利100集发布 | 性开放的女人aaa片 | 内射老妇bbwx0c0ck | 国产福利视频一区二区 | 国产深夜福利视频在线 | 精品午夜福利在线观看 | 任你躁在线精品免费 | 欧洲熟妇色 欧美 | 荫蒂被男人添的好舒服爽免费视频 | 国产无套粉嫩白浆在线 | 亚洲va欧美va天堂v国产综合 | 99久久精品日本一区二区免费 | 丁香花在线影院观看在线播放 | 久久久久成人片免费观看蜜芽 | 日韩视频 中文字幕 视频一区 | 国产精品亚洲综合色区韩国 | 色五月五月丁香亚洲综合网 | 国产在线aaa片一区二区99 | 国产内射爽爽大片视频社区在线 | 久久综合狠狠综合久久综合88 | 亚洲经典千人经典日产 | 国产精品久久久久久亚洲毛片 | 中文久久乱码一区二区 | 亚洲另类伦春色综合小说 | 亚洲成色在线综合网站 | 精品国偷自产在线视频 | 强奷人妻日本中文字幕 | 99久久精品无码一区二区毛片 | 又湿又紧又大又爽a视频国产 | 国产尤物精品视频 | 国产综合久久久久鬼色 | 亚洲国产精华液网站w | 国产精品a成v人在线播放 | 国产真实伦对白全集 | 日本高清一区免费中文视频 | 欧美日本免费一区二区三区 | 亚洲精品久久久久久一区二区 | 国产人妻精品一区二区三区不卡 | 久久久久久a亚洲欧洲av冫 | 国产一区二区不卡老阿姨 | 自拍偷自拍亚洲精品被多人伦好爽 | 成人亚洲精品久久久久 | 国产真人无遮挡作爱免费视频 | 国产真实乱对白精彩久久 | 国产一区二区三区日韩精品 | 中文精品无码中文字幕无码专区 | 久久亚洲中文字幕无码 | 九九在线中文字幕无码 | 中文字幕日韩精品一区二区三区 | 亚洲色偷偷男人的天堂 | 亚洲精品午夜国产va久久成人 | 国产美女极度色诱视频www | 少妇高潮一区二区三区99 | 亚洲第一无码av无码专区 | 日本熟妇浓毛 | 人妻尝试又大又粗久久 | 欧美zoozzooz性欧美 | 啦啦啦www在线观看免费视频 | 鲁一鲁av2019在线 | av小次郎收藏 | 波多野42部无码喷潮在线 | 性生交大片免费看女人按摩摩 | 亚洲人成影院在线观看 | 国产精品无码一区二区三区不卡 | 国产精品手机免费 | 麻豆国产97在线 | 欧洲 | 天天摸天天碰天天添 | 国产九九九九九九九a片 | 真人与拘做受免费视频一 | 小鲜肉自慰网站xnxx | 成人av无码一区二区三区 | 正在播放东北夫妻内射 | 久在线观看福利视频 | 亚洲国产精品久久人人爱 | 免费观看又污又黄的网站 | 天天拍夜夜添久久精品 | 亚洲精品国产a久久久久久 | 国产免费久久久久久无码 | 欧美真人作爱免费视频 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | 国产激情无码一区二区app | 久久综合给合久久狠狠狠97色 | 色综合久久中文娱乐网 | 一本久久a久久精品亚洲 | 亚洲s色大片在线观看 | 四虎永久在线精品免费网址 | 亚洲国产精品无码久久久久高潮 | 国产精品久免费的黄网站 | 成熟女人特级毛片www免费 | 日韩精品无码一本二本三本色 | 日日天日日夜日日摸 | 午夜精品久久久内射近拍高清 | 日韩精品久久久肉伦网站 | 又大又紧又粉嫩18p少妇 | 国产精品久久久久久久影院 | 给我免费的视频在线观看 | 国产乱子伦视频在线播放 | 亚洲国产日韩a在线播放 | 国产精品久久久久久久9999 | 图片区 小说区 区 亚洲五月 | 中文字幕日韩精品一区二区三区 | 伊人久久大香线蕉av一区二区 | 女人高潮内射99精品 | 精品一区二区三区无码免费视频 | 人人澡人摸人人添 | 男人的天堂av网站 | 人人妻在人人 | 色综合久久88色综合天天 | 亚洲成a人片在线观看日本 | 亚洲国产欧美国产综合一区 | 性史性农村dvd毛片 | 亚洲码国产精品高潮在线 | 国产亚洲日韩欧美另类第八页 | 国产精品亚洲五月天高清 | 亚洲日韩av片在线观看 | 牛和人交xxxx欧美 | 亚洲国产欧美国产综合一区 | 中文字幕人妻无码一区二区三区 | 国产真实乱对白精彩久久 | 少妇无码一区二区二三区 | 国模大胆一区二区三区 | 日日摸天天摸爽爽狠狠97 | 欧美精品免费观看二区 | 日日摸日日碰夜夜爽av | 大屁股大乳丰满人妻 | 亚洲熟妇色xxxxx欧美老妇 | 波多野结衣av一区二区全免费观看 | 国产两女互慰高潮视频在线观看 | 亚洲国产精品久久久天堂 | 久久久精品国产sm最大网站 | 久久久精品国产sm最大网站 | 国产成人无码专区 | 东京热男人av天堂 | 国产又爽又黄又刺激的视频 | 又粗又大又硬毛片免费看 | 宝宝好涨水快流出来免费视频 | 妺妺窝人体色www在线小说 | 国产美女精品一区二区三区 | 人妻少妇精品无码专区二区 | 色综合久久久无码中文字幕 | 人妻与老人中文字幕 | 中文字幕无码乱人伦 | 国产综合色产在线精品 | 中文字幕 亚洲精品 第1页 | 色婷婷综合激情综在线播放 | 国产精品成人av在线观看 | 欧美三级a做爰在线观看 | 思思久久99热只有频精品66 | av无码久久久久不卡免费网站 | 国产九九九九九九九a片 | 日产国产精品亚洲系列 | 国产午夜亚洲精品不卡 | 天天躁日日躁狠狠躁免费麻豆 | 亚洲一区二区三区偷拍女厕 | 亚拍精品一区二区三区探花 | 5858s亚洲色大成网站www | 亚洲国产精品成人久久蜜臀 | 领导边摸边吃奶边做爽在线观看 | 亚洲中文字幕在线观看 | 欧美日韩一区二区免费视频 | 伊在人天堂亚洲香蕉精品区 | 97se亚洲精品一区 | 久久99久久99精品中文字幕 | 帮老师解开蕾丝奶罩吸乳网站 | 国产亚洲精品久久久久久久 | 亚洲国产欧美国产综合一区 | 久久久久国色av免费观看性色 | 黑人巨大精品欧美一区二区 | 国产精品美女久久久网av | 人妻少妇精品无码专区二区 | 亚洲伊人久久精品影院 | 久久精品人人做人人综合试看 | 亚洲日韩av一区二区三区四区 | 三级4级全黄60分钟 | 中文精品久久久久人妻不卡 | 国产黑色丝袜在线播放 | 久久97精品久久久久久久不卡 | 女人高潮内射99精品 | 欧洲欧美人成视频在线 | 欧美成人高清在线播放 | 午夜丰满少妇性开放视频 | 亚洲热妇无码av在线播放 | 欧美老人巨大xxxx做受 | 欧美乱妇无乱码大黄a片 | 妺妺窝人体色www婷婷 | 熟女体下毛毛黑森林 | 色 综合 欧美 亚洲 国产 | 久久久久久久人妻无码中文字幕爆 | 国产亚洲精品久久久久久久久动漫 | 国产精品无码一区二区三区不卡 | 久久无码专区国产精品s | 中文字幕av日韩精品一区二区 | 国产精品毛片一区二区 | 97精品国产97久久久久久免费 | 18禁黄网站男男禁片免费观看 | 午夜成人1000部免费视频 | 久久亚洲日韩精品一区二区三区 | 免费网站看v片在线18禁无码 | 小鲜肉自慰网站xnxx | 色综合久久久无码网中文 | 日韩精品a片一区二区三区妖精 | 国产女主播喷水视频在线观看 | 亚洲综合伊人久久大杳蕉 | 国产av无码专区亚洲awww | 国内揄拍国内精品人妻 | 高潮毛片无遮挡高清免费 | 久久久久久亚洲精品a片成人 | 亚洲精品久久久久久一区二区 | 乌克兰少妇性做爰 | 日产精品99久久久久久 | 国产又爽又黄又刺激的视频 | 中文字幕av日韩精品一区二区 | 亚洲人成无码网www | 久久久中文久久久无码 | 国产两女互慰高潮视频在线观看 | 亚洲人成影院在线无码按摩店 | 日本一区二区三区免费播放 | 中文字幕av日韩精品一区二区 | 欧美日韩色另类综合 | 亚洲一区二区观看播放 | 午夜肉伦伦影院 | 成人欧美一区二区三区黑人 | 国产精品无码一区二区桃花视频 | 少妇激情av一区二区 | 精品熟女少妇av免费观看 | 日本高清一区免费中文视频 | 国产成人精品三级麻豆 | 欧美野外疯狂做受xxxx高潮 | 国产午夜无码精品免费看 | 麻豆人妻少妇精品无码专区 | 丝袜人妻一区二区三区 | 日本乱偷人妻中文字幕 | a在线观看免费网站大全 | 日本乱人伦片中文三区 | 精品午夜福利在线观看 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | www成人国产高清内射 | 天干天干啦夜天干天2017 | 欧美三级a做爰在线观看 | 久久综合色之久久综合 | 天天拍夜夜添久久精品大 | 亚洲欧美综合区丁香五月小说 | 国产精品久久久av久久久 | 国产艳妇av在线观看果冻传媒 | 亚洲精品一区三区三区在线观看 | a片免费视频在线观看 | 欧美丰满熟妇xxxx性ppx人交 | 成在人线av无码免观看麻豆 | 99re在线播放 | 亚洲aⅴ无码成人网站国产app | 精品无人区无码乱码毛片国产 | 国产无套内射久久久国产 | 中文字幕 亚洲精品 第1页 | 清纯唯美经典一区二区 | 国产色xx群视频射精 | 大肉大捧一进一出视频出来呀 | 亚洲精品国产第一综合99久久 | 99麻豆久久久国产精品免费 | 波多野结衣乳巨码无在线观看 | 国产片av国语在线观看 | 乱码午夜-极国产极内射 | 日本精品人妻无码免费大全 | 欧美日本日韩 | 亚洲国产综合无码一区 | 欧美日本精品一区二区三区 | 日韩人妻无码一区二区三区久久99 | 亚洲日韩中文字幕在线播放 | 免费无码午夜福利片69 | 人人妻人人藻人人爽欧美一区 | 精品国产麻豆免费人成网站 | 久久婷婷五月综合色国产香蕉 | 亚洲最大成人网站 | 青青草原综合久久大伊人精品 | 久久久久久国产精品无码下载 | 少妇无码吹潮 | 午夜成人1000部免费视频 | 狠狠cao日日穞夜夜穞av | 久久久成人毛片无码 | 国产精品a成v人在线播放 | 国产国语老龄妇女a片 | 欧美人与物videos另类 | 精品国偷自产在线视频 | 国产三级精品三级男人的天堂 | 伊人久久大香线蕉av一区二区 | 亚洲精品一区二区三区在线观看 | 国产人妻大战黑人第1集 | 国产在热线精品视频 | 精品无码成人片一区二区98 | 无码人妻丰满熟妇区毛片18 | 久久亚洲日韩精品一区二区三区 | 九一九色国产 | 蜜桃视频插满18在线观看 | 美女极度色诱视频国产 | 中文字幕人妻无码一区二区三区 | 日韩人妻无码中文字幕视频 | 欧美xxxx黑人又粗又长 | 久久久久人妻一区精品色欧美 | 狠狠综合久久久久综合网 | 中文字幕无码乱人伦 | 福利一区二区三区视频在线观看 | 中文字幕无码免费久久99 | 日韩精品久久久肉伦网站 | 国产三级久久久精品麻豆三级 | 女人被男人爽到呻吟的视频 | 丰腴饱满的极品熟妇 | 熟女少妇人妻中文字幕 | 国产精品国产自线拍免费软件 | 55夜色66夜色国产精品视频 | 99久久久无码国产aaa精品 | 九一九色国产 | 中文久久乱码一区二区 | 日韩人妻少妇一区二区三区 | 亚洲成av人在线观看网址 | 国产日产欧产精品精品app | 在线 国产 欧美 亚洲 天堂 | 欧美日本精品一区二区三区 | 无码国产色欲xxxxx视频 | 在教室伦流澡到高潮hnp视频 | 日日摸天天摸爽爽狠狠97 | 天堂久久天堂av色综合 | 一本久道久久综合婷婷五月 | av小次郎收藏 | 国产一精品一av一免费 | 亚拍精品一区二区三区探花 | 久久久久久a亚洲欧洲av冫 | 国产亚av手机在线观看 | 日韩精品无码免费一区二区三区 | 玩弄少妇高潮ⅹxxxyw | 综合激情五月综合激情五月激情1 | 国产成人综合在线女婷五月99播放 | 99精品视频在线观看免费 | 97夜夜澡人人爽人人喊中国片 | 高清无码午夜福利视频 | 性啪啪chinese东北女人 | 日产国产精品亚洲系列 | 国产婷婷色一区二区三区在线 | 激情内射亚州一区二区三区爱妻 | 中文毛片无遮挡高清免费 | 亚洲国产精品一区二区美利坚 | 天海翼激烈高潮到腰振不止 | 成在人线av无码免观看麻豆 | 欧美三级不卡在线观看 | 久久精品国产亚洲精品 | 高清不卡一区二区三区 | 国内精品人妻无码久久久影院 | 国产99久久精品一区二区 | 国产精品久久久一区二区三区 | 久久午夜无码鲁丝片午夜精品 | 久久久久成人精品免费播放动漫 | 十八禁真人啪啪免费网站 | 免费无码午夜福利片69 | 中文字幕中文有码在线 | 国产精品久免费的黄网站 | 97色伦图片97综合影院 | 激情综合激情五月俺也去 | 成人免费无码大片a毛片 | 美女张开腿让人桶 | 国产午夜福利亚洲第一 | 免费视频欧美无人区码 | 99精品视频在线观看免费 | 2020久久香蕉国产线看观看 | 免费网站看v片在线18禁无码 | 国产精品无套呻吟在线 | 成年女人永久免费看片 | 国产精品久久国产三级国 | 在线亚洲高清揄拍自拍一品区 | 国产九九九九九九九a片 | 天天燥日日燥 | 欧美国产日韩久久mv | 精品国产精品久久一区免费式 | 日本精品久久久久中文字幕 | 中文字幕无码av激情不卡 | 纯爱无遮挡h肉动漫在线播放 | 全球成人中文在线 | 99久久精品国产一区二区蜜芽 | 影音先锋中文字幕无码 | 亚洲最大成人网站 | 熟妇人妻无码xxx视频 | 天天燥日日燥 | 中文亚洲成a人片在线观看 | 88国产精品欧美一区二区三区 | 在线观看国产一区二区三区 | 成人欧美一区二区三区 | 无人区乱码一区二区三区 | 亚洲综合无码一区二区三区 | 极品嫩模高潮叫床 | 久久国产精品精品国产色婷婷 | 久久久久久久久蜜桃 | 亚洲va欧美va天堂v国产综合 | 国产成人无码av一区二区 | 亚洲一区二区观看播放 | 国产精品久久精品三级 | 国产成人精品一区二区在线小狼 | 日韩精品无码免费一区二区三区 | 国产精品久久久久无码av色戒 | 亚洲色大成网站www | www国产精品内射老师 | 美女毛片一区二区三区四区 | 女人色极品影院 | 亚洲欧美色中文字幕在线 | 高清无码午夜福利视频 | 国产成人无码区免费内射一片色欲 | 国产乱人伦偷精品视频 | 成人欧美一区二区三区黑人 | 永久免费观看美女裸体的网站 | 内射欧美老妇wbb | 在线精品国产一区二区三区 | 色 综合 欧美 亚洲 国产 | 少妇性俱乐部纵欲狂欢电影 | 国产精品国产自线拍免费软件 | 久久精品国产99精品亚洲 | 免费无码一区二区三区蜜桃大 | 在线亚洲高清揄拍自拍一品区 | 精品水蜜桃久久久久久久 | 欧美35页视频在线观看 | 中文精品久久久久人妻不卡 | 老司机亚洲精品影院无码 | 国产在线一区二区三区四区五区 | 俺去俺来也www色官网 | 成人综合网亚洲伊人 | 中文字幕无码热在线视频 | 少女韩国电视剧在线观看完整 | 亚洲成a人片在线观看日本 | 色欲久久久天天天综合网精品 | 色一情一乱一伦一视频免费看 | 国产精品a成v人在线播放 | 丰满岳乱妇在线观看中字无码 | 99精品国产综合久久久久五月天 | 奇米影视7777久久精品人人爽 | 亚洲精品国偷拍自产在线观看蜜桃 | 亚洲日韩av一区二区三区四区 | 麻豆精产国品 | 成人欧美一区二区三区黑人免费 | 麻豆国产97在线 | 欧洲 | 少妇性荡欲午夜性开放视频剧场 | 精品久久久无码人妻字幂 | 人妻无码αv中文字幕久久琪琪布 | 女人高潮内射99精品 | 大乳丰满人妻中文字幕日本 | 欧美35页视频在线观看 | 久久久亚洲欧洲日产国码αv | 特大黑人娇小亚洲女 | 国产亚av手机在线观看 | 撕开奶罩揉吮奶头视频 | 中文无码精品a∨在线观看不卡 | 老头边吃奶边弄进去呻吟 | 亚洲最大成人网站 | 国内综合精品午夜久久资源 | 国产后入清纯学生妹 | 久久 国产 尿 小便 嘘嘘 | 久久午夜夜伦鲁鲁片无码免费 | 欧美日韩一区二区三区自拍 | 久久国语露脸国产精品电影 | 九九综合va免费看 | 色综合久久久无码中文字幕 | √8天堂资源地址中文在线 | 又紧又大又爽精品一区二区 | 波多野42部无码喷潮在线 | 波多野结衣av在线观看 | 久久精品人妻少妇一区二区三区 | 无遮无挡爽爽免费视频 | 国产综合在线观看 | 国产激情一区二区三区 | 99精品视频在线观看免费 | 老司机亚洲精品影院 | 宝宝好涨水快流出来免费视频 | 国产成人无码午夜视频在线观看 | 亚洲熟妇自偷自拍另类 | 蜜桃视频韩日免费播放 | 国产午夜无码精品免费看 | 亚洲精品综合一区二区三区在线 | 九九久久精品国产免费看小说 | 99视频精品全部免费免费观看 | 久久国产精品偷任你爽任你 | 国产人妻人伦精品 | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 亚洲国产欧美国产综合一区 | 精品偷自拍另类在线观看 | 中国女人内谢69xxxx | 性欧美牲交xxxxx视频 | 三上悠亚人妻中文字幕在线 | 亚洲人成影院在线无码按摩店 | 奇米影视7777久久精品人人爽 | 欧美性生交xxxxx久久久 | 未满小14洗澡无码视频网站 | 大地资源网第二页免费观看 | 久久久久免费精品国产 | 2020久久香蕉国产线看观看 | 99精品无人区乱码1区2区3区 | 日韩精品a片一区二区三区妖精 | 51国偷自产一区二区三区 | 麻豆av传媒蜜桃天美传媒 | 成人精品视频一区二区三区尤物 | 性欧美熟妇videofreesex | 正在播放老肥熟妇露脸 | 人妻少妇精品无码专区动漫 | 亚洲成a人片在线观看无码3d | 国产精品无码mv在线观看 | 日日天日日夜日日摸 | 人妻插b视频一区二区三区 | 欧美人与物videos另类 | 亚洲精品成人福利网站 | 国产97人人超碰caoprom | 亚洲国产成人a精品不卡在线 | 国产精品无码一区二区三区不卡 | 天天摸天天透天天添 | 国精品人妻无码一区二区三区蜜柚 | 成人免费无码大片a毛片 | 无码av免费一区二区三区试看 | 久久精品中文字幕一区 | 秋霞成人午夜鲁丝一区二区三区 | 国产特级毛片aaaaaaa高清 | 性做久久久久久久免费看 | 亚洲成a人片在线观看无码3d | 精品无人区无码乱码毛片国产 | 精品国产一区二区三区四区在线看 | 少妇无码一区二区二三区 | 桃花色综合影院 | 久久精品99久久香蕉国产色戒 | 红桃av一区二区三区在线无码av | 国产激情综合五月久久 | 中文字幕中文有码在线 | 国产精华av午夜在线观看 | 四虎国产精品一区二区 | 无码任你躁久久久久久久 | 日韩无码专区 | 午夜精品一区二区三区的区别 | 精品国偷自产在线视频 | 国产成人综合在线女婷五月99播放 | 日本成熟视频免费视频 | √天堂中文官网8在线 | 久久久久av无码免费网 | 无码av免费一区二区三区试看 | 国产亚洲精品久久久久久久 | www国产亚洲精品久久久日本 | 色综合久久久久综合一本到桃花网 | 精品国产av色一区二区深夜久久 | 狂野欧美性猛交免费视频 | 亚洲色偷偷偷综合网 | 亚洲精品成人福利网站 | 免费网站看v片在线18禁无码 | 久久熟妇人妻午夜寂寞影院 | 无码人妻丰满熟妇区毛片18 | 老头边吃奶边弄进去呻吟 | 亚洲国产精品一区二区第一页 | 欧美国产亚洲日韩在线二区 | 在线精品国产一区二区三区 | 久久久精品成人免费观看 | 国产三级精品三级男人的天堂 | 久久天天躁狠狠躁夜夜免费观看 | 久久精品中文闷骚内射 | 亚洲日韩av片在线观看 | 熟女俱乐部五十路六十路av | 丝袜美腿亚洲一区二区 | 国产热a欧美热a在线视频 | 久久久久久久女国产乱让韩 | 久久国产精品二国产精品 | 日韩 欧美 动漫 国产 制服 | 老太婆性杂交欧美肥老太 | 欧洲精品码一区二区三区免费看 | 国产亲子乱弄免费视频 | 帮老师解开蕾丝奶罩吸乳网站 | 成人无码精品1区2区3区免费看 | 久久国产精品_国产精品 | av香港经典三级级 在线 | 精品久久久中文字幕人妻 | 亚洲高清偷拍一区二区三区 | 欧美日韩亚洲国产精品 | 无码一区二区三区在线 | 成熟人妻av无码专区 | 好男人www社区 | 强伦人妻一区二区三区视频18 | 精品国产成人一区二区三区 | 国产性生大片免费观看性 | 精品久久久久久人妻无码中文字幕 | 一区二区传媒有限公司 | 国产精品自产拍在线观看 | 色一情一乱一伦 | 国内老熟妇对白xxxxhd | 欧美 亚洲 国产 另类 | 亚洲春色在线视频 | 色窝窝无码一区二区三区色欲 | 天天躁日日躁狠狠躁免费麻豆 | 妺妺窝人体色www婷婷 | 无码国内精品人妻少妇 | 中文字幕人妻无码一区二区三区 | 少妇太爽了在线观看 | 免费无码肉片在线观看 | 青青青手机频在线观看 | 欧美色就是色 | 日本乱人伦片中文三区 | 国产激情艳情在线看视频 | 国产精品久久国产精品99 | 亚洲欧美日韩国产精品一区二区 | 久精品国产欧美亚洲色aⅴ大片 | 蜜桃臀无码内射一区二区三区 | 欧美阿v高清资源不卡在线播放 | 欧美性黑人极品hd | 人人妻人人澡人人爽人人精品 | 人妻人人添人妻人人爱 | 国色天香社区在线视频 | 99riav国产精品视频 | 欧美性猛交xxxx富婆 | 国产网红无码精品视频 | 大肉大捧一进一出好爽视频 | 巨爆乳无码视频在线观看 | 久久亚洲中文字幕无码 | 欧美亚洲日韩国产人成在线播放 | 中国大陆精品视频xxxx | 亚欧洲精品在线视频免费观看 | 国产午夜手机精彩视频 | 国产成人无码一二三区视频 | 日韩av激情在线观看 | 国产精品-区区久久久狼 | 鲁鲁鲁爽爽爽在线视频观看 | 国产人妻人伦精品 | 女高中生第一次破苞av | 少妇高潮喷潮久久久影院 | 国产真人无遮挡作爱免费视频 | 国产97在线 | 亚洲 | 久久99精品久久久久婷婷 | 成年女人永久免费看片 | 中文亚洲成a人片在线观看 | 精品无人区无码乱码毛片国产 | 久久99精品国产麻豆蜜芽 | 免费国产成人高清在线观看网站 | 国产精品无码永久免费888 | 国产精品毛片一区二区 | 亚洲中文字幕乱码av波多ji | 日本饥渴人妻欲求不满 | 丰满少妇女裸体bbw | 乌克兰少妇性做爰 | 在线观看国产午夜福利片 | 亚洲精品国产精品乱码不卡 | 久激情内射婷内射蜜桃人妖 | 亚洲精品一区二区三区在线 | 亚拍精品一区二区三区探花 | 久久久国产一区二区三区 | 亚洲 激情 小说 另类 欧美 | 青青草原综合久久大伊人精品 | 中文字幕 亚洲精品 第1页 | 亚洲自偷精品视频自拍 | 婷婷六月久久综合丁香 | 久久精品人妻少妇一区二区三区 | 亚洲中文字幕va福利 | 欧美精品在线观看 | 沈阳熟女露脸对白视频 | 国产suv精品一区二区五 | 欧美丰满老熟妇xxxxx性 | 国产性猛交╳xxx乱大交 国产精品久久久久久无码 欧洲欧美人成视频在线 | 97夜夜澡人人爽人人喊中国片 | 久精品国产欧美亚洲色aⅴ大片 | 18无码粉嫩小泬无套在线观看 | 蜜桃臀无码内射一区二区三区 | 一本色道久久综合狠狠躁 | 伊人久久大香线蕉av一区二区 | 成人无码精品1区2区3区免费看 | 人人妻人人澡人人爽精品欧美 | 国产麻豆精品精东影业av网站 | 377p欧洲日本亚洲大胆 | 九九热爱视频精品 | 亚洲熟妇色xxxxx欧美老妇 | 日韩av无码一区二区三区 | 国产精品无码成人午夜电影 | 久久99久久99精品中文字幕 | 双乳奶水饱满少妇呻吟 | 亚洲a无码综合a国产av中文 | 国内精品人妻无码久久久影院蜜桃 | 99er热精品视频 | 美女极度色诱视频国产 | 久精品国产欧美亚洲色aⅴ大片 | 日本丰满熟妇videos | 午夜无码人妻av大片色欲 | 亚洲人亚洲人成电影网站色 | 国产av剧情md精品麻豆 | 国产亚洲tv在线观看 | 亚洲国产高清在线观看视频 | 中文字幕无码免费久久99 | 欧美乱妇无乱码大黄a片 | 婷婷丁香五月天综合东京热 | 午夜嘿嘿嘿影院 | 久久久久免费精品国产 | 亚洲中文字幕va福利 | 午夜精品久久久内射近拍高清 | 99久久精品午夜一区二区 | 少妇被黑人到高潮喷出白浆 | 亚洲s码欧洲m码国产av | 久久97精品久久久久久久不卡 | 日韩精品乱码av一区二区 | 小sao货水好多真紧h无码视频 | 蜜桃无码一区二区三区 | 中国女人内谢69xxxx | 亚洲色大成网站www | 欧美熟妇另类久久久久久多毛 | av无码电影一区二区三区 | 激情内射亚州一区二区三区爱妻 | 一个人免费观看的www视频 | 亚洲精品成人av在线 | 久久精品国产精品国产精品污 | 亚洲春色在线视频 | 亚洲精品久久久久久久久久久 | 国产凸凹视频一区二区 | 色一情一乱一伦一视频免费看 | 国产香蕉尹人视频在线 | 真人与拘做受免费视频一 | 美女极度色诱视频国产 | 露脸叫床粗话东北少妇 | 无码人妻av免费一区二区三区 | 中文无码伦av中文字幕 | 欧美精品免费观看二区 | 99久久人妻精品免费二区 | 最近免费中文字幕中文高清百度 | 成 人 网 站国产免费观看 | 人人妻人人澡人人爽人人精品 | 少妇高潮一区二区三区99 | 国产香蕉尹人视频在线 | 99久久亚洲精品无码毛片 | 午夜成人1000部免费视频 | 中文字幕+乱码+中文字幕一区 | 欧美一区二区三区视频在线观看 | 无码播放一区二区三区 | 欧美三级a做爰在线观看 | 波多野结衣av一区二区全免费观看 | 精品 日韩 国产 欧美 视频 | 午夜精品一区二区三区在线观看 | 亚洲精品久久久久久一区二区 | 久久精品国产日本波多野结衣 | 国产精品久久精品三级 | 在线а√天堂中文官网 | 国产黑色丝袜在线播放 | 免费无码肉片在线观看 | 国産精品久久久久久久 | 久久国语露脸国产精品电影 | 欧美老妇与禽交 | www成人国产高清内射 | 国产凸凹视频一区二区 | 亚洲欧美日韩综合久久久 | 粗大的内捧猛烈进出视频 | 国产精品久久久久久久9999 | 欧美性猛交内射兽交老熟妇 | 日韩人妻无码中文字幕视频 | 国产午夜精品一区二区三区嫩草 | 国产成人精品视频ⅴa片软件竹菊 | 男女爱爱好爽视频免费看 | 最新国产麻豆aⅴ精品无码 | 老司机亚洲精品影院 | 老熟女重囗味hdxx69 | 澳门永久av免费网站 | 日本高清一区免费中文视频 | 妺妺窝人体色www婷婷 | 免费观看的无遮挡av | 在线视频网站www色 | 久久天天躁夜夜躁狠狠 | 久久精品99久久香蕉国产色戒 | 老子影院午夜伦不卡 | 精品成在人线av无码免费看 | 午夜熟女插插xx免费视频 | 一区二区传媒有限公司 | 成人精品天堂一区二区三区 | 青青青爽视频在线观看 | 老子影院午夜精品无码 | 无人区乱码一区二区三区 | 国产香蕉97碰碰久久人人 | 青青草原综合久久大伊人精品 | 欧美人与禽zoz0性伦交 | 奇米影视888欧美在线观看 | 欧美变态另类xxxx | 乱人伦人妻中文字幕无码 | 久久久精品456亚洲影院 | 无码人妻出轨黑人中文字幕 | 成 人 免费观看网站 | 欧洲熟妇精品视频 | 国产精品va在线观看无码 | 永久免费观看美女裸体的网站 | 一本久久a久久精品vr综合 | 久久天天躁狠狠躁夜夜免费观看 | 狠狠躁日日躁夜夜躁2020 | 免费国产黄网站在线观看 | 日产精品高潮呻吟av久久 | 亚洲理论电影在线观看 | 日产精品99久久久久久 | 麻花豆传媒剧国产免费mv在线 | 夫妻免费无码v看片 | 免费观看又污又黄的网站 | 色综合久久久无码网中文 | 亚洲成色www久久网站 | 麻豆国产人妻欲求不满 | 2019nv天堂香蕉在线观看 | 一区二区传媒有限公司 | 国产免费观看黄av片 | 国产精品二区一区二区aⅴ污介绍 | 国产舌乚八伦偷品w中 | 免费网站看v片在线18禁无码 | 欧美 日韩 亚洲 在线 | 亚洲精品成人av在线 | 久久精品99久久香蕉国产色戒 | 亚洲男人av天堂午夜在 | 亚洲а∨天堂久久精品2021 | 久久综合给合久久狠狠狠97色 | 亚洲中文字幕在线无码一区二区 | 国产一区二区三区影院 | 野外少妇愉情中文字幕 | 无套内谢老熟女 | 永久黄网站色视频免费直播 | 国产熟妇高潮叫床视频播放 | 日日躁夜夜躁狠狠躁 | 国内揄拍国内精品少妇国语 | 好屌草这里只有精品 | 国产成人人人97超碰超爽8 | 国精产品一品二品国精品69xx | 天天做天天爱天天爽综合网 | 国产色xx群视频射精 | 国产精品永久免费视频 | 无码帝国www无码专区色综合 | 欧美人妻一区二区三区 | 免费网站看v片在线18禁无码 | 久久久无码中文字幕久... | 国产精品久久久午夜夜伦鲁鲁 | 国产精品沙发午睡系列 | 日韩成人一区二区三区在线观看 | 在线看片无码永久免费视频 | 在线视频网站www色 | 国产精品久久久久久亚洲影视内衣 | 伊人久久婷婷五月综合97色 | 亚洲精品中文字幕久久久久 | 国产精品亚洲lv粉色 | 日本乱偷人妻中文字幕 | 色一情一乱一伦一视频免费看 | 午夜福利电影 | 荫蒂添的好舒服视频囗交 | 在线视频网站www色 | 欧美色就是色 | 精品国产一区二区三区四区在线看 | 久久精品国产亚洲精品 | 国产成人精品三级麻豆 | 国产黄在线观看免费观看不卡 | 丰满少妇高潮惨叫视频 | 日本www一道久久久免费榴莲 | 18禁黄网站男男禁片免费观看 | 国产亚洲人成在线播放 | 久久人人爽人人爽人人片av高清 | 国产性生大片免费观看性 | 日韩欧美群交p片內射中文 | 人人妻人人澡人人爽欧美精品 | 欧美日本免费一区二区三区 | 国产性生大片免费观看性 | 国产又粗又硬又大爽黄老大爷视 | 亚洲一区二区三区播放 | 日韩亚洲欧美中文高清在线 | 国产成人精品一区二区在线小狼 | 高清不卡一区二区三区 | 精品久久综合1区2区3区激情 | av无码久久久久不卡免费网站 | 日韩无套无码精品 | 极品嫩模高潮叫床 | 国产无遮挡吃胸膜奶免费看 | 少妇无码吹潮 | 国产美女极度色诱视频www | 一本无码人妻在中文字幕免费 | 亚洲伊人久久精品影院 | 久久国产精品二国产精品 | 水蜜桃亚洲一二三四在线 | 日本一卡二卡不卡视频查询 | 亚洲人成网站免费播放 | 草草网站影院白丝内射 | 午夜精品一区二区三区在线观看 | 欧美成人高清在线播放 | 少妇高潮一区二区三区99 | 日本丰满熟妇videos | 又粗又大又硬毛片免费看 | 欧美乱妇无乱码大黄a片 | 色综合视频一区二区三区 | 国产人妻人伦精品 | 国产亚洲人成a在线v网站 | 丰满人妻被黑人猛烈进入 | 奇米影视7777久久精品 | 午夜无码区在线观看 | 成人无码精品1区2区3区免费看 | 欧美激情综合亚洲一二区 | 国产免费无码一区二区视频 | 亚洲一区二区三区偷拍女厕 | 2020最新国产自产精品 | 精品国产国产综合精品 | 最新国产乱人伦偷精品免费网站 | 亚欧洲精品在线视频免费观看 | 丰满少妇人妻久久久久久 | 无码帝国www无码专区色综合 | 亚洲欧美精品伊人久久 | 国产午夜亚洲精品不卡下载 | 玩弄人妻少妇500系列视频 | 红桃av一区二区三区在线无码av | av在线亚洲欧洲日产一区二区 | 国产精品人人爽人人做我的可爱 | 日本精品久久久久中文字幕 | 亚洲欧美国产精品久久 | 小鲜肉自慰网站xnxx | 伊人久久婷婷五月综合97色 | 野狼第一精品社区 | 无码毛片视频一区二区本码 | 国产特级毛片aaaaaa高潮流水 | 成人精品天堂一区二区三区 | 夜夜夜高潮夜夜爽夜夜爰爰 | 99久久人妻精品免费一区 | 天堂亚洲免费视频 | 久久综合久久自在自线精品自 | 亚洲国产精品久久人人爱 | 国产精品毛片一区二区 | 国产特级毛片aaaaaa高潮流水 | 又大又硬又爽免费视频 | 麻豆成人精品国产免费 | 国产精品久久国产精品99 | 美女扒开屁股让男人桶 | 国产欧美熟妇另类久久久 | 大肉大捧一进一出好爽视频 | 亚洲精品久久久久久一区二区 | 亚洲欧美日韩国产精品一区二区 | 天天做天天爱天天爽综合网 | 在线а√天堂中文官网 | 丰满人妻一区二区三区免费视频 | 77777熟女视频在线观看 а天堂中文在线官网 | 午夜无码区在线观看 | 久久综合九色综合97网 | 久久精品国产亚洲精品 | 日韩成人一区二区三区在线观看 | 国产肉丝袜在线观看 | 午夜福利不卡在线视频 | 国产在线aaa片一区二区99 | 久久精品国产99精品亚洲 | 少妇性荡欲午夜性开放视频剧场 | 日韩精品乱码av一区二区 | 领导边摸边吃奶边做爽在线观看 | 免费无码午夜福利片69 | 国产无套粉嫩白浆在线 | 欧美激情内射喷水高潮 | 久久精品女人的天堂av | 精品成在人线av无码免费看 | 成人精品视频一区二区三区尤物 | 国产一精品一av一免费 | 婷婷色婷婷开心五月四房播播 | 300部国产真实乱 | 亚洲熟妇色xxxxx欧美老妇 | 性生交片免费无码看人 | 亚洲春色在线视频 | 国产成人精品无码播放 | 麻豆国产丝袜白领秘书在线观看 | 成熟女人特级毛片www免费 | 人人妻人人澡人人爽人人精品 | 人妻aⅴ无码一区二区三区 | 亚洲欧美日韩综合久久久 | 麻豆蜜桃av蜜臀av色欲av | 日本熟妇人妻xxxxx人hd | 欧美日韩人成综合在线播放 | 嫩b人妻精品一区二区三区 | 97无码免费人妻超级碰碰夜夜 | 红桃av一区二区三区在线无码av | 未满成年国产在线观看 | 成人亚洲精品久久久久软件 | 中文字幕人成乱码熟女app | a国产一区二区免费入口 | 香蕉久久久久久av成人 | 亚洲精品午夜无码电影网 | 久久久久久av无码免费看大片 | 精品人人妻人人澡人人爽人人 | 久久久成人毛片无码 | 狂野欧美性猛xxxx乱大交 | 日韩av无码一区二区三区不卡 | 成人欧美一区二区三区黑人免费 | 亚洲成在人网站无码天堂 | 激情人妻另类人妻伦 | 中文字幕 亚洲精品 第1页 | 人妻少妇精品视频专区 | 亚洲一区二区三区 | 国产成人无码午夜视频在线观看 | 欧美人与物videos另类 | 乱码av麻豆丝袜熟女系列 | 中文字幕无码av激情不卡 | 一二三四社区在线中文视频 | 青青青手机频在线观看 | 国产九九九九九九九a片 | 日本一本二本三区免费 | 国产两女互慰高潮视频在线观看 | 天堂久久天堂av色综合 | 久在线观看福利视频 | 99久久99久久免费精品蜜桃 | 久久久久久a亚洲欧洲av冫 | 九九综合va免费看 | 成人无码视频免费播放 | 日韩精品久久久肉伦网站 | 无码国产乱人伦偷精品视频 | 自拍偷自拍亚洲精品被多人伦好爽 | 国产无av码在线观看 | 一本色道久久综合狠狠躁 | 兔费看少妇性l交大片免费 | 国产精品爱久久久久久久 | 18禁止看的免费污网站 | 高潮喷水的毛片 | 色婷婷av一区二区三区之红樱桃 | 久久综合久久自在自线精品自 | 全黄性性激高免费视频 | 国产精品美女久久久 | 精品无码国产自产拍在线观看蜜 | 人人超人人超碰超国产 | av在线亚洲欧洲日产一区二区 | 色婷婷欧美在线播放内射 | 夫妻免费无码v看片 | 麻豆国产人妻欲求不满 | 亚洲综合精品香蕉久久网 | 精品国精品国产自在久国产87 | 国产69精品久久久久app下载 | 狠狠噜狠狠狠狠丁香五月 | 久久久亚洲欧洲日产国码αv | 色婷婷av一区二区三区之红樱桃 | 精品成在人线av无码免费看 | 黑人巨大精品欧美一区二区 | 亚洲一区二区三区国产精华液 | 亚洲乱码中文字幕在线 | v一区无码内射国产 | 三级4级全黄60分钟 | 亚洲色偷偷偷综合网 | 日韩av无码一区二区三区不卡 | 国产偷抇久久精品a片69 | 一个人看的视频www在线 | 日本爽爽爽爽爽爽在线观看免 | 亚洲精品久久久久久一区二区 | 激情爆乳一区二区三区 | 玩弄人妻少妇500系列视频 | 亚洲精品欧美二区三区中文字幕 | 中文字幕日韩精品一区二区三区 | 丰满护士巨好爽好大乳 | 亚洲男人av天堂午夜在 | 黑人粗大猛烈进出高潮视频 | 清纯唯美经典一区二区 | 奇米影视888欧美在线观看 | 一区二区三区高清视频一 | 51国偷自产一区二区三区 | 波多野结衣av一区二区全免费观看 | 男人的天堂2018无码 | 无码人妻久久一区二区三区不卡 | 日韩人妻系列无码专区 | 国产一区二区不卡老阿姨 | 国产suv精品一区二区五 | 丰满少妇高潮惨叫视频 | 人妻无码久久精品人妻 | 99re在线播放 | 国产精品久久久久久无码 | 黑人巨大精品欧美一区二区 | 偷窥村妇洗澡毛毛多 | 精品国产av色一区二区深夜久久 | 久久97精品久久久久久久不卡 | 高潮喷水的毛片 | 76少妇精品导航 | 久久97精品久久久久久久不卡 | 婷婷五月综合激情中文字幕 | 无码中文字幕色专区 | 天堂无码人妻精品一区二区三区 | 老太婆性杂交欧美肥老太 | 欧美亚洲日韩国产人成在线播放 | 国产精品99爱免费视频 | 好男人社区资源 | 99久久精品午夜一区二区 | 日韩av无码一区二区三区 | 亚洲国产精华液网站w | 少妇高潮喷潮久久久影院 | 人人妻在人人 | 国产精品无码久久av | 婷婷丁香五月天综合东京热 | 人人澡人人妻人人爽人人蜜桃 | 免费无码一区二区三区蜜桃大 | 久久精品人人做人人综合试看 | 久久亚洲精品成人无码 | 久久国产精品萌白酱免费 | 亚洲精品美女久久久久久久 | 国内丰满熟女出轨videos | 黑人巨大精品欧美黑寡妇 | 国精品人妻无码一区二区三区蜜柚 | 久久伊人色av天堂九九小黄鸭 | 亚洲欧洲无卡二区视頻 | 伊人久久大香线蕉亚洲 | 久久久久久av无码免费看大片 | 精品久久久无码人妻字幂 | 国产精品久久久久影院嫩草 | 青青青手机频在线观看 | 精品国产乱码久久久久乱码 | 国产尤物精品视频 | 国产超碰人人爽人人做人人添 | 日韩欧美中文字幕公布 | 国产无套内射久久久国产 | 国产亚av手机在线观看 | 老太婆性杂交欧美肥老太 | 国产人妻精品午夜福利免费 | 宝宝好涨水快流出来免费视频 | 久久精品国产日本波多野结衣 | 四十如虎的丰满熟妇啪啪 | 美女毛片一区二区三区四区 | 久久久亚洲欧洲日产国码αv | 白嫩日本少妇做爰 | 东京一本一道一二三区 | 日本丰满护士爆乳xxxx | 无码人妻丰满熟妇区毛片18 | 精品乱码久久久久久久 | √天堂中文官网8在线 | 国产精品第一区揄拍无码 | 老司机亚洲精品影院 | 国内丰满熟女出轨videos | √天堂资源地址中文在线 | 国产成人无码av片在线观看不卡 | 欧美大屁股xxxxhd黑色 | 扒开双腿疯狂进出爽爽爽视频 | 中国女人内谢69xxxxxa片 | 亚洲精品成人福利网站 | 最近免费中文字幕中文高清百度 | 国产精品第一区揄拍无码 | 久久99热只有频精品8 | 国産精品久久久久久久 | aⅴ在线视频男人的天堂 | 亚洲毛片av日韩av无码 | 亚洲日韩中文字幕在线播放 | 大乳丰满人妻中文字幕日本 | 给我免费的视频在线观看 | 中国大陆精品视频xxxx | 丰满人妻一区二区三区免费视频 | 中文字幕无码av激情不卡 | 成年美女黄网站色大免费视频 | 亚洲综合另类小说色区 | 久久久中文字幕日本无吗 | 亚洲理论电影在线观看 | 美女扒开屁股让男人桶 | 国产精品国产自线拍免费软件 | 天天av天天av天天透 | 久久久精品欧美一区二区免费 | 色窝窝无码一区二区三区色欲 | 国产精品二区一区二区aⅴ污介绍 | 激情爆乳一区二区三区 | 麻豆国产丝袜白领秘书在线观看 | 漂亮人妻洗澡被公强 日日躁 | 全黄性性激高免费视频 | 色噜噜亚洲男人的天堂 | 国产精品亚洲一区二区三区喷水 | 女人被爽到呻吟gif动态图视看 | 亚洲精品国产精品乱码不卡 | 色综合久久88色综合天天 | 精品国精品国产自在久国产87 | 国产成人无码av一区二区 | 伦伦影院午夜理论片 | 无码任你躁久久久久久久 | 99麻豆久久久国产精品免费 | 亚洲国产精品久久人人爱 | 久久人妻内射无码一区三区 | 一本色道久久综合亚洲精品不卡 | 99精品无人区乱码1区2区3区 | 成 人 网 站国产免费观看 | 亚洲综合伊人久久大杳蕉 | 夜精品a片一区二区三区无码白浆 | 亚洲精品国产第一综合99久久 | 国产特级毛片aaaaaa高潮流水 | 一区二区三区乱码在线 | 欧洲 | 国产又粗又硬又大爽黄老大爷视 | 国产深夜福利视频在线 | 色欲综合久久中文字幕网 | 夫妻免费无码v看片 | 国产精品久久久久无码av色戒 | 天天躁日日躁狠狠躁免费麻豆 | 97精品人妻一区二区三区香蕉 | 对白脏话肉麻粗话av | 色五月五月丁香亚洲综合网 | 亚洲一区二区观看播放 | 午夜无码区在线观看 | 亚洲精品一区二区三区大桥未久 | 亚洲色在线无码国产精品不卡 | 99久久婷婷国产综合精品青草免费 | 国产女主播喷水视频在线观看 | 午夜精品一区二区三区在线观看 | 色综合久久久无码中文字幕 | 久久精品国产一区二区三区肥胖 |