一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

腳本之家,腳本語言編程技術(shù)及教程分享平臺(tái)!
分類導(dǎo)航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服務(wù)器之家 - 腳本之家 - Golang - golang日志框架之logrus的使用

golang日志框架之logrus的使用

2020-05-27 10:32jack-life Golang

這篇文章主要介紹了golang日志框架之logrus的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

golang日志

golang標(biāo)準(zhǔn)庫的日志框架非常簡單,僅僅提供了print,panic和fatal三個(gè)函數(shù)對于更精細(xì)的日志級(jí)別、日志文件分割以及日志分發(fā)等方面并沒有提供支持。所以催生了很多第三方的日志庫,但是在golang的世界里,沒有一個(gè)日志庫像slf4j那樣在Java中具有絕對統(tǒng)治地位。golang中,流行的日志框架包括logrus、zap、zerolog、seelog等。

logrus是目前Github上star數(shù)量最多的日志庫,目前(2018.08,下同)star數(shù)量為8119,fork數(shù)為1031。logrus功能強(qiáng)大,性能高效,而且具有高度靈活性,提供了自定義插件的功能。很多開源項(xiàng)目,如docker,prometheus等,都是用了logrus來記錄其日志。

zap是Uber推出的一個(gè)快速、結(jié)構(gòu)化的分級(jí)日志庫。具有強(qiáng)大的ad-hoc分析功能,并且具有靈活的儀表盤。zap目前在GitHub上的star數(shù)量約為4.3k。

seelog提供了靈活的異步調(diào)度、格式化和過濾功能。目前在GitHub上也有約1.1k。

logrus特性

logrus具有以下特性:

  • 完全兼容golang標(biāo)準(zhǔn)庫日志模塊:logrus擁有六種日志級(jí)別:debug、info、warn、error、fatal和panic,這是golang標(biāo)準(zhǔn)庫日志模塊的API的超集。如果您的項(xiàng)目使用標(biāo)準(zhǔn)庫日志模塊,完全可以以最低的代價(jià)遷移到logrus上。
  • 可擴(kuò)展的Hook機(jī)制:允許使用者通過hook的方式將日志分發(fā)到任意地方,如本地文件系統(tǒng)、標(biāo)準(zhǔn)輸出、logstash、elasticsearch或者mq等,或者通過hook定義日志內(nèi)容和格式等。
  • 可選的日志輸出格式:logrus內(nèi)置了兩種日志格式,JSONFormatter和TextFormatter,如果這兩個(gè)格式不滿足需求,可以自己動(dòng)手實(shí)現(xiàn)接口Formatter,來定義自己的日志格式。
  • Field機(jī)制:logrus鼓勵(lì)通過Field機(jī)制進(jìn)行精細(xì)化的、結(jié)構(gòu)化的日志記錄,而不是通過冗長的消息來記錄日志。
  • logrus是一個(gè)可插拔的、結(jié)構(gòu)化的日志框架。

logrus的使用

第一個(gè)示例

最簡單的使用logrus的示例如下:

?
1
2
3
4
5
6
7
8
9
10
11
package main
 
import (
 log "github.com/sirupsen/logrus"
)
 
func main() {
 log.WithFields(log.Fields{
  "animal": "walrus",
 }).Info("A walrus appears")
}

上面代碼執(zhí)行后,標(biāo)準(zhǔn)輸出上輸出如下:

?
1
time="2018-08-11T15:42:22+08:00" level=info msg="A walrus appears" animal=walrus

logrus與golang標(biāo)準(zhǔn)庫日志模塊完全兼容,因此您可以使用log“github.com/sirupsen/logrus”替換所有日志導(dǎo)入。
logrus可以通過簡單的配置,來定義輸出、格式或者日志級(jí)別等。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
package main
 
import (
  "os"
  log "github.com/sirupsen/logrus"
)
 
func init() {
  // 設(shè)置日志格式為json格式
  log.SetFormatter(&log.JSONFormatter{})
 
  // 設(shè)置將日志輸出到標(biāo)準(zhǔn)輸出(默認(rèn)的輸出為stderr,標(biāo)準(zhǔn)錯(cuò)誤)
  // 日志消息輸出可以是任意的io.writer類型
  log.SetOutput(os.Stdout)
 
  // 設(shè)置日志級(jí)別為warn以上
  log.SetLevel(log.WarnLevel)
}
 
func main() {
  log.WithFields(log.Fields{
    "animal": "walrus",
    "size":  10,
  }).Info("A group of walrus emerges from the ocean")
 
  log.WithFields(log.Fields{
    "omg":  true,
    "number": 122,
  }).Warn("The group's number increased tremendously!")
 
  log.WithFields(log.Fields{
    "omg":  true,
    "number": 100,
  }).Fatal("The ice breaks!")
}

Logger

logger是一種相對高級(jí)的用法, 對于一個(gè)大型項(xiàng)目, 往往需要一個(gè)全局的logrus實(shí)例,即logger對象來記錄項(xiàng)目所有的日志。如:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package main
 
import (
  "github.com/sirupsen/logrus"
  "os"
)
 
// logrus提供了New()函數(shù)來創(chuàng)建一個(gè)logrus的實(shí)例。
// 項(xiàng)目中,可以創(chuàng)建任意數(shù)量的logrus實(shí)例。
var log = logrus.New()
 
func main() {
  // 為當(dāng)前l(fā)ogrus實(shí)例設(shè)置消息的輸出,同樣地,
  // 可以設(shè)置logrus實(shí)例的輸出到任意io.writer
  log.Out = os.Stdout
 
  // 為當(dāng)前l(fā)ogrus實(shí)例設(shè)置消息輸出格式為json格式。
  // 同樣地,也可以單獨(dú)為某個(gè)logrus實(shí)例設(shè)置日志級(jí)別和hook,這里不詳細(xì)敘述。
  log.Formatter = &logrus.JSONFormatter{}
 
  log.WithFields(logrus.Fields{
    "animal": "walrus",
    "size":  10,
  }).Info("A group of walrus emerges from the ocean")
}

Fields

前一章提到過,logrus不推薦使用冗長的消息來記錄運(yùn)行信息,它推薦使用Fields來進(jìn)行精細(xì)化的、結(jié)構(gòu)化的信息記錄。
例如下面的記錄日志的方式:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
log.Fatalf("Failed to send event %s to topic %s with key %d", event, topic, key)
````
 
在logrus中不太提倡,logrus鼓勵(lì)使用以下方式替代之:
 
 
 
 
 
<div class="se-preview-section-delimiter"></div>
 
```go
log.WithFields(log.Fields{
 "event": event,
 "topic": topic,
 "key": key,
}).Fatal("Failed to send event")

前面的WithFields API可以規(guī)范使用者按照其提倡的方式記錄日志。但是WithFields依然是可選的,因?yàn)槟承﹫鼍跋?,使用者確實(shí)只需要記錄儀一條簡單的消息。

通常,在一個(gè)應(yīng)用中、或者應(yīng)用的一部分中,都有一些固定的Field。比如在處理用戶http請求時(shí),上下文中,所有的日志都會(huì)有request_iduser_ip。為了避免每次記錄日志都要使用log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip}),我們可以創(chuàng)建一個(gè)logrus.Entry實(shí)例,為這個(gè)實(shí)例設(shè)置默認(rèn)Fields,在上下文中使用這個(gè)logrus.Entry實(shí)例記錄日志即可。

?
1
2
3
requestLogger := log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip})
requestLogger.Info("something happened on that request") # will log request_id and user_ip
requestLogger.Warn("something not great happened")

Hook

logrus最令人心動(dòng)的功能就是其可擴(kuò)展的HOOK機(jī)制了,通過在初始化時(shí)為logrus添加hook,logrus可以實(shí)現(xiàn)各種擴(kuò)展功能。

Hook接口

logrus的hook接口定義如下,其原理是每此寫入日志時(shí)攔截,修改logrus.Entry。

?
1
2
3
4
5
6
// logrus在記錄Levels()返回的日志級(jí)別的消息時(shí)會(huì)觸發(fā)HOOK,
// 按照Fire方法定義的內(nèi)容修改logrus.Entry。
type Hook interface {
  Levels() []Level
  Fire(*Entry) error
}

一個(gè)簡單自定義hook如下,DefaultFieldHook定義會(huì)在所有級(jí)別的日志消息中加入默認(rèn)字段appName="myAppName"。

?
1
2
3
4
5
6
7
8
9
10
11
type DefaultFieldHook struct {
}
 
func (hook *DefaultFieldHook) Fire(entry *log.Entry) error {
  entry.Data["appName"] = "MyAppName"
  return nil
}
 
func (hook *DefaultFieldHook) Levels() []log.Level {
  return log.AllLevels
}

hook的使用也很簡單,在初始化前調(diào)用log.AddHook(hook)添加相應(yīng)的hook即可。

logrus官方僅僅內(nèi)置了syslog的hook。

此外,但Github也有很多第三方的hook可供使用,文末將提供一些第三方HOOK的連接。

問題與解決方案

盡管logrus有諸多優(yōu)點(diǎn),但是為了靈活性和可擴(kuò)展性,官方也削減了很多實(shí)用的功能,例如:

  • 沒有提供行號(hào)和文件名的支持
  • 輸出到本地文件系統(tǒng)沒有提供日志分割功能
  • 官方?jīng)]有提供輸出到ELK等日志處理中心的功能

但是這些功能都可以通過自定義hook來實(shí)現(xiàn)。

記錄文件名和行號(hào)

logrus的一個(gè)很致命的問題就是沒有提供文件名和行號(hào),這在大型項(xiàng)目中通過日志定位問題時(shí)有諸多不便。Github上的logrus的issue#63:Log filename and line number創(chuàng)建于2014年,四年過去了仍是open狀態(tài)~~~
網(wǎng)上給出的解決方案分位兩類,一就是自己實(shí)現(xiàn)一個(gè)hook;二就是通過裝飾器包裝logrus.Entry。兩種方案網(wǎng)上都有很多代碼,但是大多無法正常工作。但總體來說,解決問題的思路都是對的:通過標(biāo)準(zhǔn)庫的runtime模塊獲取運(yùn)行時(shí)信息,并從中提取文件名,行號(hào)和調(diào)用函數(shù)名。

標(biāo)準(zhǔn)庫runtime模塊的Caller(skip int)函數(shù)可以返回當(dāng)前goroutine調(diào)用棧中的文件名,行號(hào),函數(shù)信息等,參數(shù)skip表示表示返回的棧幀的層次,0表示runtime.Caller的調(diào)用著。返回值包括響應(yīng)棧幀層次的pc(程序計(jì)數(shù)器),文件名和行號(hào)信息。為了提高效率,我們先通過跟蹤調(diào)用棧發(fā)現(xiàn),從runtime.Caller()的調(diào)用者開始,到記錄日志的生成代碼之間,大概有8到11層左右,所有我們在hook中循環(huán)第8到11層調(diào)用棧應(yīng)該可以找到日志記錄的生產(chǎn)代碼。

golang日志框架之logrus的使用

此外,runtime.FuncForPC(pc uintptr) *Func可以返回指定pc的函數(shù)信息。

所有我們要實(shí)現(xiàn)的hook也是基于以上原理,使用runtime.Caller()依次循環(huán)調(diào)用棧的第7~11層,過濾掉sirupsen包內(nèi)容,那么第一個(gè)非siupsenr包就認(rèn)為是我們的生產(chǎn)代碼了,并返回pc以便通過runtime.FuncForPC()獲取函數(shù)名稱。然后將文件名、行號(hào)和函數(shù)名組裝為source字段塞到logrus.Entry中即可。

time="2018-08-11T19:10:15+08:00" level=warning msg="postgres_exporter is ready for scraping on 0.0.0.0:9295..." source="postgres_exporter/main.go:60:main()"
time="2018-08-11T19:10:17+08:00" level=error msg="!!!msb info not found" source="postgres/postgres_query.go:63:QueryPostgresInfo()"
time="2018-08-11T19:10:17+08:00" level=error msg="get postgres instances info failed, scrape metrics failed, error:msb env not found" source="collector/exporter.go:71:Scrape()"

日志本地文件分割

logrus本身不帶日志本地文件分割功能,但是我們可以通過file-rotatelogs進(jìn)行日志本地文件分割。 每次當(dāng)我們寫入日志的時(shí)候,logrus都會(huì)調(diào)用file-rotatelogs來判斷日志是否要進(jìn)行切分。關(guān)于本地日志文件分割的例子網(wǎng)上很多,這里不再詳細(xì)介紹,奉上代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
import (
  "github.com/lestrrat-go/file-rotatelogs"
  "github.com/rifflock/lfshook"
  log "github.com/sirupsen/logrus"
  "time"
)
 
func newLfsHook(logLevel *string, maxRemainCnt uint) log.Hook {
  writer, err := rotatelogs.New(
    logName+".%Y%m%d%H",
    // WithLinkName為最新的日志建立軟連接,以方便隨著找到當(dāng)前日志文件
    rotatelogs.WithLinkName(logName),
 
    // WithRotationTime設(shè)置日志分割的時(shí)間,這里設(shè)置為一小時(shí)分割一次
    rotatelogs.WithRotationTime(time.Hour),
 
    // WithMaxAge和WithRotationCount二者只能設(shè)置一個(gè),
    // WithMaxAge設(shè)置文件清理前的最長保存時(shí)間,
    // WithRotationCount設(shè)置文件清理前最多保存的個(gè)數(shù)。
    //rotatelogs.WithMaxAge(time.Hour*24),
    rotatelogs.WithRotationCount(maxRemainCnt),
  )
 
  if err != nil {
    log.Errorf("config local file system for logger error: %v", err)
  }
 
  level, ok := logLevels[*logLevel]
 
  if ok {
    log.SetLevel(level)
  } else {
    log.SetLevel(log.WarnLevel)
  }
 
  lfsHook := lfshook.NewHook(lfshook.WriterMap{
    log.DebugLevel: writer,
    log.InfoLevel: writer,
    log.WarnLevel: writer,
    log.ErrorLevel: writer,
    log.FatalLevel: writer,
    log.PanicLevel: writer,
  }, &log.TextFormatter{DisableColors: true})
 
  return lfsHook
}

使用上述本地日志文件切割的效果如下:

golang日志框架之logrus的使用

將日志發(fā)送到elasticsearch

將日志發(fā)送到elasticsearch是很多日志監(jiān)控系統(tǒng)的選擇,將logrus日志發(fā)送到elasticsearch的原理是在hook的每次fire調(diào)用時(shí),使用golang的es客戶端將日志信息寫到elasticsearch。elasticsearch官方?jīng)]有提供golang客戶端,但是有很多第三方的go語言客戶端可供使用,我們選擇elastic。elastic提供了豐富的文檔,以及Java中的流式接口,使用起來非常方便。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
client, err := elastic.NewClient(elastic.SetURL("http://localhost:9200"))
  if err != nil {
    log.Panic(err)
  }
 
// Index a tweet (using JSON serialization)
tweet1 := Tweet{User: "olivere", Message: "Take Five", Retweets: 0}
put1, err := client.Index().
  Index("twitter").
  Type("tweet").
  Id("1").
  BodyJson(tweet1).
  Do(context.Background())

考慮到logrus的Fields機(jī)制,可以實(shí)現(xiàn)如下數(shù)據(jù)格式:

?
1
2
3
4
5
6
7
msg := struct {
  Host   string
  Timestamp string `json:"@timestamp"`
  Message  string
  Data   logrus.Fields
  Level   string
}

其中Host記錄產(chǎn)生日志主機(jī)信息,在創(chuàng)建hook是指定。其他數(shù)據(jù)需要從logrus.Entry中取得。測試過程我們選擇按照此原理實(shí)現(xiàn)的第三方HOOK:elogrus。其使用如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
import (
  "github.com/olivere/elastic"
  "gopkg.in/sohlich/elogrus"
)
 
func initLog() {
  client, err := elastic.NewClient(elastic.SetURL("http://localhost:9200"))
  if err != nil {
    log.Panic(err)
  }
  hook, err := elogrus.NewElasticHook(client, "localhost", log.DebugLevel, "mylog")
  if err != nil {
    log.Panic(err)
  }
  log.AddHook(hook)
}

從Elasticsearch查詢得到日志存儲(chǔ),效果如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
 
HTTP/1.1 200 OK
content-type: application/json; charset=UTF-8
transfer-encoding: chunked
 
{
 "took": 1,
 "timed_out": false,
 "_shards": {
  "total": 5,
  "successful": 5,
  "failed": 0
 },
 "hits": {
  "total": 2474,
  "max_score": 1.0,
  "hits": [
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw13jWnMZReb-jHQup",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:12:32.212818666Z",
     "Message": "!!!msb info not found",
     "Data": {},
     "Level": "ERROR"
    }
   },
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw13jgnMZReb-jHQuq",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:12:32.223103348Z",
     "Message": "get postgres instances info failed, scrape metrics failed, error:msb env not found",
     "Data": {
      "source": "collector/exporter.go:71:Scrape()"
     },
     "Level": "ERROR"
    }
   },
   //...
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw2f1enMZReb-jHQu_",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:15:17.212546892Z",
     "Message": "!!!msb info not found",
     "Data": {
      "source": "collector/exporter.go:71:Scrape()"
     },
     "Level": "ERROR"
    }
   },
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw2NhmnMZReb-jHQu1",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:14:02.21276903Z",
     "Message": "!!!msb info not found",
     "Data": {},
     "Level": "ERROR"
    }
   }
  ]
 }
}

將日志發(fā)送到其他位置

將日志發(fā)送到日志中心也是logrus所提倡的,雖然沒有提供官方支持,但是目前Github上有很多第三方hook可供使用:

logrus_amqp:Logrus hook for Activemq。

logrus-logstash-hook:Logstash hook for logrus。

mgorus:Mongodb Hooks for Logrus。

logrus_influxdb:InfluxDB Hook for Logrus。

logrus-redis-hook:Hook for Logrus which enables logging to RELK stack (Redis, Elasticsearch, Logstash and Kibana)。

等等,上述第三方hook我這里沒有具體驗(yàn)證,大家可以根據(jù)需要自行嘗試。

其他注意事項(xiàng)

Fatal處理

和很多日志框架一樣,logrus的Fatal系列函數(shù)會(huì)執(zhí)行os.Exit(1)。但是logrus提供可以注冊一個(gè)或多個(gè)fatal handler函數(shù)的接口logrus.RegisterExitHandler(handler func() {} ),讓logrus在執(zhí)行os.Exit(1)之前進(jìn)行相應(yīng)的處理。fatal handler可以在系統(tǒng)異常時(shí)調(diào)用一些資源釋放api等,讓應(yīng)用正確的關(guān)閉。

線程安全

默認(rèn)情況下,logrus的api都是線程安全的,其內(nèi)部通過互斥鎖來保護(hù)并發(fā)寫?;コ怄i工作于調(diào)用hooks或者寫日志的時(shí)候,如果不需要鎖,可以調(diào)用logger.SetNoLock()來關(guān)閉之??梢躁P(guān)閉logrus互斥鎖的情形包括:

  • 沒有設(shè)置hook,或者所有的hook都是線程安全的實(shí)現(xiàn)。
  • 寫日志到logger.Out已經(jīng)是線程安全的了,如logger.Out已經(jīng)被鎖保護(hù),或者寫文件時(shí),文件是以O_APPEND方式打開的,并且每次寫操作都小于4k。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持服務(wù)器之家。

原文鏈接:https://blog.csdn.net/wslyk606/article/details/81670713

延伸 · 閱讀

精彩推薦
  • Golanggolang 通過ssh代理連接mysql的操作

    golang 通過ssh代理連接mysql的操作

    這篇文章主要介紹了golang 通過ssh代理連接mysql的操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    a165861639710342021-03-08
  • Golanggolang json.Marshal 特殊html字符被轉(zhuǎn)義的解決方法

    golang json.Marshal 特殊html字符被轉(zhuǎn)義的解決方法

    今天小編就為大家分享一篇golang json.Marshal 特殊html字符被轉(zhuǎn)義的解決方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧 ...

    李浩的life12792020-05-27
  • Golanggo日志系統(tǒng)logrus顯示文件和行號(hào)的操作

    go日志系統(tǒng)logrus顯示文件和行號(hào)的操作

    這篇文章主要介紹了go日志系統(tǒng)logrus顯示文件和行號(hào)的操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    SmallQinYan12302021-02-02
  • GolangGolang通脈之?dāng)?shù)據(jù)類型詳情

    Golang通脈之?dāng)?shù)據(jù)類型詳情

    這篇文章主要介紹了Golang通脈之?dāng)?shù)據(jù)類型,在編程語言中標(biāo)識(shí)符就是定義的具有某種意義的詞,比如變量名、常量名、函數(shù)名等等,Go語言中標(biāo)識(shí)符允許由...

    4272021-11-24
  • Golanggo語言制作端口掃描器

    go語言制作端口掃描器

    本文給大家分享的是使用go語言編寫的TCP端口掃描器,可以選擇IP范圍,掃描的端口,以及多線程,有需要的小伙伴可以參考下。 ...

    腳本之家3642020-04-25
  • GolangGolang中Bit數(shù)組的實(shí)現(xiàn)方式

    Golang中Bit數(shù)組的實(shí)現(xiàn)方式

    這篇文章主要介紹了Golang中Bit數(shù)組的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    天易獨(dú)尊11682021-06-09
  • Golanggolang的httpserver優(yōu)雅重啟方法詳解

    golang的httpserver優(yōu)雅重啟方法詳解

    這篇文章主要給大家介紹了關(guān)于golang的httpserver優(yōu)雅重啟的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,...

    helight2992020-05-14
  • Golanggolang如何使用struct的tag屬性的詳細(xì)介紹

    golang如何使用struct的tag屬性的詳細(xì)介紹

    這篇文章主要介紹了golang如何使用struct的tag屬性的詳細(xì)介紹,從例子說起,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看...

    Go語言中文網(wǎng)11352020-05-21
主站蜘蛛池模板: 亚洲成人91 | 免费刷10000名片赞网站 | 99热这里只有精品在线 | 美女被狂揉下部羞羞动漫 | 精品在线免费观看 | 亚洲精品中文字幕久久久久久 | 亚洲 欧美 另类 中文 在线 | 成人欧美一区二区三区白人 | 久久亚洲精品AV无码四区 | 国产精品嫩草影院一二三区 | 糖心vlog网页版 | 久久er国产免费精品 | 我要看免费毛片 | 我与岳乱短篇小说 | 精品国产成人a区在线观看 精品国产91久久久久久久 | 精品国产爱久久 | 亚洲欧美日韩成人 | 久久永久免费视频 | 国产日韩欧美在线一二三四 | 国产成人福利色视频 | 久久国产乱子伦免费精品 | 天天操夜夜操狠狠操 | 男男gaygays国内 | 毛片视频在线免费观看 | 久草在在线免视频在线观看 | 国产精品亚洲专区在线播放 | 日韩精品一区二区三区老鸭窝 | 好紧好爽再叫浪一点点潘金莲 | 99九九国产精品免费视频 | 国产欧美日韩亚洲精品区2345 | 欧美在线视频免费播放 | 精品国产免费久久久久久 | 欧美三级不卡在线观线看高清 | 日本最大的黄色网站 | 动漫jk美女被爆羞羞漫画 | 香蕉tv国产在线永久播放 | 国产香蕉一区二区精品视频 | 性欧美video 性满足久久久久久久久 | caonila国产在线观看 | 99av涩导航 | 精品午夜视频 |