<ruby id="bdb3f"></ruby>

    <p id="bdb3f"><cite id="bdb3f"></cite></p>

      <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
        <p id="bdb3f"><cite id="bdb3f"></cite></p>

          <pre id="bdb3f"></pre>
          <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

          <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
          <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

          <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                <ruby id="bdb3f"></ruby>

                ThinkChat2.0新版上線,更智能更精彩,支持會話、畫圖、視頻、閱讀、搜索等,送10W Token,即刻開啟你的AI之旅 廣告
                ## [](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/12.2.md#1221-讀文件)12.2.1 讀文件 在 Go 語言中,文件使用指向?`os.File`?類型的指針來表示的,也叫做文件句柄。我們在前面章節使用到過標準輸入`os.Stdin`?和標準輸出?`os.Stdout`,他們的類型都是?`*os.File`。讓我們來看看下面這個程序: 示例 12.4?[fileinput.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/examples/chapter_12/fileinput.go): ~~~ package main import ( "bufio" "fmt" "io" "os" ) func main() { inputFile, inputError := os.Open("input.dat") if inputError != nil { fmt.Printf("An error occurred on opening the inputfile\n" + "Does the file exist?\n" + "Have you got acces to it?\n") return // exit the function on error } defer inputFile.Close() inputReader := bufio.NewReader(inputFile) for { inputString, readerError := inputReader.ReadString('\n') if readerError == io.EOF { return } fmt.Printf("The input was: %s", inputString) } } ~~~ 變量?`inputFile`?是?`*os.File`?類型的。該類型是一個結構,表示一個打開文件的描述符(文件句柄)。然后,使用?`os`包里的?`Open`?函數來打開一個文件。該函數的參數是文件名,類型為?`string`。在上面的程序中,我們以只讀模式打開 input.dat 文件。 如果文件不存在或者程序沒有足夠的權限打開這個文件,Open函數會返回一個錯誤:`inputFile, inputError = os.Open("input.dat")`。如果文件打開正常,我們就使用?`defer.Close()`?語句確保在程序退出前關閉該文件。然后,我們使用?`bufio.NewReader`?來獲得一個讀取器變量。 通過使用?`bufio`?包提供的讀取器(寫入器也類似),如上面程序所示,我們可以很方便的操作相對高層的 string 對象,而避免了去操作比較底層的字節。 接著,我們在一個無限循環中使用?`ReadString('\n')`?或?`ReadBytes('\n')`?將文件的內容逐行(行結束符 '\n')讀取出來。 **注意:**?在之前的例子中,我們看到,Unix和Linux的行結束符是 \n,而Windows的行結束符是 \r\n。在使用?`ReadString`?和`ReadBytes`?方法的時候,我們不需要關心操作系統的類型,直接使用 \n 就可以了。另外,我們也可以使用?`ReadLine()`方法來實現相同的功能。 一旦讀取到文件末尾,變量?`readerError`?的值將變成非空(事實上,常亮?`io.EOF`?的值是 true),我們就會執行?`return`語句從而退出循環。 **其他類似函數:** **1) 將整個文件的內容讀到一個字符串里:** 如果您想這么做,可以使用?`io/ioutil`?包里的?`ioutil.ReadFile()`?方法,該方法第一個返回值的類型是?`[]byte`,里面存放讀取到的內容,第二個返回值是錯誤,如果沒有錯誤發生,第二個返回值為 nil。請看示例 12.5。類似的,函數`WriteFile()`?可以將?`[]byte`?的值寫入文件。 示例 12.5?[read_write_file1.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/examples/chapter_12/read_write_file1.go): ~~~ package main import ( "fmt" "io/ioutil" "os" ) func main() { inputFile := "products.txt" outputFile := "products_copy.txt" buf, err := ioutil.ReadFile(inputFile) if err != nil { fmt.Fprintf(os.Stderr, "File Error: %s\n", err) // panic(err.Error()) } fmt.Printf("%s\n", string(buf)) err = ioutil.WriteFile(outputFile, buf, 0x644) if err != nil { panic(err. Error()) } } ~~~ **2) 帶緩沖的讀取** 在很多情況下,文件的內容是不按行劃分的,或者干脆就是一個二進制文件。在這種情況下,`ReadString()`就無法使用了,我們可以使用?`bufio.Reader`?的?`Read()`,它只接收一個參數: ~~~ buf := make([]byte, 1024) ... n, err := inputReader.Read(buf) if (n == 0) { break} ~~~ 變量 n 的值表示讀取到的字節數. **3) 按列讀取文件中的數據** 如果數據是按列排列并用空格分隔的,你可以使用?`fmt`?包提供的以 FScan 開頭的一系列函數來讀取他們。請看以下程序,我們將 3 列的數據分別讀入變量 v1、v2 和 v3 內,然后分別把他們添加到切片的尾部。 示例 12.6?[read_file2.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/examples/chapter_12/read_file2.go): ~~~ package main import ( "fmt" "os" ) func main() { file, err := os.Open("products2.txt") if err != nil { panic(err) } defer file.Close() var col1, col2, col3 []string for { var v1, v2, v3 string _, err := fmt.Fscanln(file, &v1, &v2, &v3) // scans until newline if err != nil { break } col1 = append(col1, v1) col2 = append(col2, v2) col3 = append(col3, v3) } fmt.Println(col1) fmt.Println(col2) fmt.Println(col3) } ~~~ 輸出結果: ~~~ [ABC FUNC GO] [40 56 45] [150 280 356] ~~~ **注意:**?`path`?包里包含一個子包叫?`filepath`,這個子包提供了跨平臺的函數,用于處理文件名和路徑。例如 Base() 函數用于獲得路徑中的最后一個元素(不包含后面的分隔符): ~~~ import "path/filepath" filename := filepath.Base(path) ~~~ **練習 12.3**:[read_csv.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/exercises/chapter_12/read_csv.go) 文件 products.txt 的內容如下: ~~~ "The ABC of Go";25.5;1500 "Functional Programming with Go";56;280 "Go for It";45.9;356 "The Go Way";55;500 ~~~ 每行的第一個字段為 title,第二個字段為 price,第三個字段為 quantity。內容的格式基本與 示例 12.3c 的相同,除了分隔符改成了分號。請讀取出文件的內容,創建一個結構用于存取一行的數據,然后使用結構的切片,并把數據打印出來。 關于解析 CSV 文件,`encoding/csv`?包提供了相應的功能。具體請參考?[http://golang.org/pkg/encoding/csv/](http://golang.org/pkg/encoding/csv/) ## [](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/12.2.md#1222-compress包讀取壓縮文件)12.2.2?`compress`包:讀取壓縮文件 `compress`包提供了讀取壓縮文件的功能,支持的壓縮文件格式為:bzip2、flate、gzip、lzw 和 zlib。 下面的程序展示了如何讀取一個 gzip 文件。 示例 12.7?[gzipped.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/examples/chapter_12/gzipped.go): ~~~ package main import ( "fmt" "bufio" "os" "compress/gzip" ) func main() { fName := "MyFile.gz" var r *bufio.Reader fi, err := os.Open(fName) if err != nil { fmt.Fprintf(os.Stderr, "%v, Can't open %s: error: %s\n", os.Args[0], fName, err) os.Exit(1) } fz, err := gzip.NewReader(fi) if err != nil { r = bufio.NewReader(fi) } else { r = bufio.NewReader(fz) } for { line, err := r.ReadString('\n') if err != nil { fmt.Println("Done reading file") os.Exit(0) } fmt.Println(line) } } ~~~ ## [](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/12.2.md#1223-寫文件)12.2.3 寫文件 請看以下程序: 示例 12.8?[fileoutput.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/examples/chapter_12/fileoutput.go): ~~~ package main import ( "os" "bufio" "fmt" ) func main () { // var outputWriter *bufio.Writer // var outputFile *os.File // var outputError os.Error // var outputString string outputFile, outputError := os.OpenFile("output.dat", os.O_WRONLY|os.O_CREATE, 0666) if outputError != nil { fmt.Printf("An error occurred with file opening or creation\n") return } defer outputFile.Close() outputWriter := bufio.NewWriter(outputFile) outputString := "hello world!\n" for i:=0; i<10; i++ { outputWriter.WriteString(outputString) } outputWriter.Flush() } ~~~ 除了文件句柄,我們還需要?`bufio`?的寫入器。我們以只讀模式打開文件?`output.dat`,如果文件不存在則自動創建: ~~~ outputFile, outputError := os.OpenFile(“output.dat”, os.O_WRONLY|os.O_ CREATE, 0666) ~~~ 可以看到,`OpenFile`?函數有三個參數:文件名、一個或多個標志(使用邏輯運算符“|”連接),使用的文件權限。 我們通常會用到以下標志:?`os.O_RDONLY`:只讀 `os.WRONLY`:只寫 `os.O_CREATE`:創建:如果指定文件不存在,就創建該文件。 `os.O_TRUNC`:截斷:如果指定文件已存在,就將該文件的長度截為0。 在讀文件的時候,文件的權限是被忽略的,所以在使用?`OpenFile`?時傳入的第三個參數可以用0。而在寫文件時,不管是 Unix 還是 Windows,都需要使用 0666。 然后,我們創建一個寫入器(緩沖區)對象: ~~~ outputWriter := bufio.NewWriter(outputFile) ~~~ 接著,使用一個 for 循環,將字符串寫入緩沖區,寫 10 次:`outputWriter.WriteString(outputString)` 緩沖區的內容緊接著被完全寫入文件:`outputWriter.Flush()` 如果寫入的東西很簡單,我們可以使用?`fmt.Fprintf(outputFile, “Some test data.\n”)`?直接將內容寫入文件。`fmt`?包里的 F 開頭的 Print 函數可以直接寫入任何?`io.Writer`,包括文件(請參考[章節12.8](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/12.8.md)). 程序?`filewrite.go`?展示了不使用?`fmt.FPrintf`?函數,使用其他函數如何寫文件: 示例 12.8?[filewrite.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/examples/chapter_12/filewrite.go): ~~~ package main import "os" func main() { os.Stdout.WriteString("hello, world\n") f, _ := os.OpenFile("test", os.O_CREATE|os.O_WRONLY, 0) defer f.Close() f.WriteString("hello, world in a file\n") } ~~~ 使用?`os.Stdout.WriteString(“hello, world\n”)`,我們可以輸出到屏幕。 我們以只寫模式創建或打開文件“test”,并且忽略了可能發生的錯誤:`f, _ := os.OpenFile(“test”, os.O_CREATE|os.O_WRONLY, 0)` 我們不使用緩沖區,直接將內容寫入文件:`f.WriteString( )` **練習 12.4**:[wiki_part1.go](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/exercises/chapter_12/wiki_part1.go) (這是一個獨立的練習,但是同時也是為[章節15.4](https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/15.4.md)做準備) 程序中的數據結構如下,是一個包含以下字段的結構: ~~~ type Page struct { Title string Body []byte } ~~~ 請給這個結構編寫一個?`save`?方法,將 Title 作為文件名、Body作為文件內容,寫入到文本文件中。 再編寫一個?`load`?函數,接收的參數是字符串 title,該函數讀取出與 title 對應的文本文件。請使用 *Page 做為參數,因為這個結構可能相當巨大,我們不想在內存中拷貝它。請使用?`ioutil`?包里的函數(參考章節12.2.1)。
                  <ruby id="bdb3f"></ruby>

                  <p id="bdb3f"><cite id="bdb3f"></cite></p>

                    <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
                      <p id="bdb3f"><cite id="bdb3f"></cite></p>

                        <pre id="bdb3f"></pre>
                        <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

                        <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
                        <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

                        <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                              <ruby id="bdb3f"></ruby>

                              哎呀哎呀视频在线观看