1. 程式人生 > >go log包之Print, Fatal, Panic

go log包之Print, Fatal, Panic

      Golang's log模組主要提供了3類介面。分別是 “Print Panic Fatal ”,對每一類介面其提供了3中呼叫方式,分別是 "Xxxx 、 Xxxxln 、Xxxxf",基本和fmt中的相關函式類似,下面是一個Print的示例:

package main

import (
    "log"
)

func main(){
    arr := []int {2,3}
    log.Print("Print array ",arr,"\n")
    log.Println("Println array",arr)
    log.Printf("Printf array with item [%d,%d]\n",arr[0],arr[1])
}

       會得到如下結果:

2016/12/15 19:46:19 Print array [2 3]
2016/12/15 19:46:19 Println array [2 3]
2016/12/15 19:46:19 Printf array with item [2,3]

       對於 log.Fatal 介面,會先將日誌內容列印到標準輸出,接著呼叫系統的 os.exit(1) 介面,退出程式並返回狀態 1 。但是有一點需要注意,由於是直接呼叫系統介面退出,defer函式不會被呼叫,下面是一個Fatal的示例:

package main

import (
	"fmt"
	"log"
)

func test_deferfatal(){
	defer func() {
		fmt.Println("--first--")
	}()
	log.Fatalln("test for defer Fatal")
}

func main() {
	test_deferfatal()
}

        會得到如下結果:

2016/12/15 19:46:45 test for defer Fatal

        可以看到並沒有呼叫defer 函式。

        對於log.Panic介面,該函式把日誌內容刷到標準錯誤後呼叫 panic 函式,下面是一個Panic的示例:

package main

import (
	"fmt"
	"log"
)

func test_deferpanic(){
	defer func() {
		fmt.Println("--first--")
		if err := recover(); err != nil {
			fmt.Println(err)
		}
	}()
	log.Panicln("test for defer Panic")
	defer func() {
		fmt.Println("--second--")
	}()
}

func main() {
	test_deferpanic()
}

        會得到如下結果:

2016/12/15 19:59:30 test for defer Panic
--first--
test for defer Panic

       可以看到首先輸出了“test for defer Panic”,然後第一個defer函式被呼叫了並輸出了“--first--”,但是第二個defer 函式並沒有輸出,可見在Panic之後宣告的defer是不會執行的。

       你也可以自定義Logger型別, log.Logger提供了一個New方法用來建立物件:

func New(out io.Writer, prefix string, flag int) *Logger

       該函式一共有三個引數:  

     (1)輸出位置out,是一個io.Writer物件,該物件可以是一個檔案也可以是實現了該介面的物件。通常我們可以用這個來指定日誌輸出到哪個檔案。     (2)prefix 我們在前面已經看到,就是在日誌內容前面的東西。我們可以將其置為 "[Info]" 、 "[Warning]"等來幫助區分日誌級別。     (3) flags 是一個選項,顯示日誌開頭的東西,可選的值有:

Ldate         = 1 << iota     // 形如 2009/01/23 的日期
Ltime                         // 形如 01:23:23   的時間
Lmicroseconds                 // 形如 01:23:23.123123   的時間
Llongfile                     // 全路徑檔名和行號: /a/b/c/d.go:23 
Lshortfile                    // 檔名和行號: d.go:23
LstdFlags     = Ldate | Ltime // 日期和時間

       示例如下:

package main
import (
    "log"
    "os"
)
func main(){
    fileName := "Info_First.log"
    logFile,err  := os.Create(fileName)
    defer logFile.Close()
    if err != nil {
        log.Fatalln("open file error")
    }
    debugLog := log.New(logFile,"[Info]",log.Llongfile)
    debugLog.Println("A Info message here")
    debugLog.SetPrefix("[Debug]")
    debugLog.Println("A Debug Message here ")
}