標(biāo)準(zhǔn)庫 - fmt/scan.go 解讀

// Copyright 2010 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

// go/src/fmt/scan.go
// version 1.7

// 格式化輸入輸出的用法請參考:http://www.cnblogs.com/golove/p/3284304.html

package fmt

import (
    "errors"
    "io"
    "math"
    "os"
    "reflect"
    "strconv"
    "sync"
    "unicode/utf8"
)

// ScanState 將掃描器的狀態(tài)報告給自定義類型的 Scan 方法。
type ScanState interface {
    // ReadRune 從輸入端讀取一個字符仙蚜,如果用在 Scanln 類的掃描器中侈贷,
    // 則該方法會在讀到第一個換行符之后或讀到指定寬度之后返回 EOF。
    // r   :讀取的字符
    // size:字符所占用的字節(jié)數(shù)
    // err :遇到的錯誤信息
    ReadRune() (r rune, size int, err error)
    // UnreadRune 撤消最后一次的 ReadRune 操作,
    // 使下次的 ReadRune 操作得到與前一次 ReadRune 相同的結(jié)果卧抗。
    // 返回:遇到的錯誤信息
    UnreadRune() error
    // SkipSpace 為自定義的 Scan 方法提供跳過開頭空白的能力读跷。
    // 根據(jù)掃描器的不同(Scan 或 Scanln)決定是否跳過換行符。
    SkipSpace()
    // Token 用于從輸入端讀取符合要求的字符串蠢络,準(zhǔn)備解析衰猛。
    // Token 從輸入端讀取連續(xù)的符合 f(c) 的字符 c。如果 f 為 nil刹孔,則使用
    // !unicode.IsSpace(c) 代替 f(c)啡省。
    // skipSpace:是否跳過輸入端開頭的連續(xù)空白(通過 SkipSpace 方法)。
    // token    :存放讀取到的數(shù)據(jù)髓霞。
    // err      :遇到的錯誤信息卦睹。
    // 注意:token 指向共享的數(shù)據(jù),下次的 Token 操作可能會覆蓋本次的結(jié)果酸茴。
    Token(skipSpace bool, f func(rune) bool) (token []byte, err error)
    // Width 返回占位符中指定的寬度值(寬度值是字符個數(shù)分预,不是字節(jié)個數(shù))。
    // wid:獲取到的寬度值
    // ok :是否指定了寬度值
    Width() (wid int, ok bool)
    // 因為上面實現(xiàn)了 ReadRune 方法薪捍,所以 Read 方法永遠(yuǎn)不應(yīng)該被 Scan 方法調(diào)用笼痹。
    // 一個好的 ScanState 應(yīng)該讓 Read 直接返回相應(yīng)的錯誤信息。
    Read(buf []byte) (n int, err error)
}

// Scanner 用于讓自定義類型實現(xiàn)自己的掃描過程酪穿。
// Scan 方法會從輸入端讀取數(shù)據(jù)并將處理結(jié)果存入接收端凳干,接收端必須是有效的指針。
// Scan 方法會被掃描器調(diào)用被济,只要對應(yīng)的 arg 實現(xiàn)了該方法救赐。
type Scanner interface {
    Scan(state ScanState, verb rune) error
}

// Scan 從標(biāo)準(zhǔn)輸入中讀取字符串(以空白分隔的值的序列)并解析為具體的值,
// 存入?yún)?shù) a 所提供的變量中(參數(shù) a 必須提供變量的地址)。換行視為空白经磅。
// 當(dāng)讀到 EOF 或所有變量都填寫完畢則停止掃描泌绣。
// n  :成功解析的參數(shù)數(shù)量
// err:解析過程中遇到的錯誤信息
func Scan(a ...interface{}) (n int, err error) {
    return Fscan(os.Stdin, a...)
}

// Scanln 和 Scan 類似,只不過遇到換行符就停止掃描预厌。
func Scanln(a ...interface{}) (n int, err error) {
    return Fscanln(os.Stdin, a...)
}

// Scanf 從標(biāo)準(zhǔn)輸入中讀取字符串阿迈,并根據(jù)格式字符串 format 對讀取的數(shù)據(jù)進(jìn)行解析,
// 存入?yún)?shù) a 所提供的變量中(參數(shù) a 必須提供變量的地址)轧叽。
// 輸入端的換行符必須和格式字符串中的換行符相對應(yīng)(如果格式字符串中有換行符苗沧,則
// 輸入端必須輸入相應(yīng)的換行符)。
// 占位符 %c 總是匹配下一個字符炭晒,包括空白待逞,比如空格符、制表符网严、換行符识樱。
// n  :成功解析的參數(shù)數(shù)量
// err:解析過程中遇到的錯誤信息
func Scanf(format string, a ...interface{}) (n int, err error) {
    return Fscanf(os.Stdin, format, a...)
}

// 實現(xiàn)了 Reader 接口的字符串類型
type stringReader string

func (r *stringReader) Read(b []byte) (n int, err error) {
    n = copy(b, *r)
    *r = (*r)[n:]
    if n == 0 {
        err = io.EOF
    }
    return
}

// Sscan 和 Scan 類似,只不過從 str 中讀取數(shù)據(jù)屿笼。
func Sscan(str string, a ...interface{}) (n int, err error) {
    return Fscan((*stringReader)(&str), a...)
}

// Sscanln 和 Scanln 類似牺荠,只不過從 str 中讀取數(shù)據(jù)。
func Sscanln(str string, a ...interface{}) (n int, err error) {
    return Fscanln((*stringReader)(&str), a...)
}

// Sscanf 和 Scanf 類似驴一,只不過從 str 中讀取數(shù)據(jù)休雌。
func Sscanf(str string, format string, a ...interface{}) (n int, err error) {
    return Fscanf((*stringReader)(&str), format, a...)
}

// Fscan 和 Scan 類似,只不過從 r 中讀取數(shù)據(jù)肝断。
func Fscan(r io.Reader, a ...interface{}) (n int, err error) {
    s, old := newScanState(r, true, false) // 創(chuàng)建掃描器
    n, err = s.doScan(a)                   // 開始掃描
    s.free(old)                            // 回收掃描器
    return
}

// Fscanln 和 Fcanln 類似杈曲,只不過從 r 中讀取數(shù)據(jù)。
func Fscanln(r io.Reader, a ...interface{}) (n int, err error) {
    s, old := newScanState(r, false, true) // 創(chuàng)建掃描器
    n, err = s.doScan(a)                   // 開始掃描
    s.free(old)                            // 回收掃描器
    return
}

// Fscanf 和 Scanf 類似胸懈,只不過從 r 中讀取數(shù)據(jù)担扑。
func Fscanf(r io.Reader, format string, a ...interface{}) (n int, err error) {
    s, old := newScanState(r, false, false) // 創(chuàng)建掃描器
    n, err = s.doScanf(format, a)           // 開始掃描
    s.free(old)                             // 回收掃描器
    return
}

// scanError 聲明本地錯誤類型,用于 recover 時辨別 panic 是否由本地代碼產(chǎn)生的趣钱。
type scanError struct {
    err error
}

// 本地代碼用 -1 表示遇到 EOF
const eof = -1

// ss 是掃描器涌献,整個掃描過程都是由它完成的。
// 它從 rs 中讀取數(shù)據(jù)并進(jìn)行解析首有。
type ss struct {
    rs    io.RuneScanner // 輸入端
    buf   buffer         // 緩沖區(qū)
    count int            // 已讀取的字符數(shù)
    atEOF bool           // 是否讀到 EOF
    ssave                // 一些需要經(jīng)常復(fù)位的字段
}

// ssave 是 ss 的一部分燕垃,存儲一些需要經(jīng)常復(fù)位的字段
type ssave struct {
    validSave bool // 平時用不上,遞歸時使用
    nlIsEnd   bool // 是否在換行符之后停止讀取
    nlIsSpace bool // 是否將換行符視為空白
    argLimit  int  // 已讀的字符數(shù)不能超過 argLimit(argLimit <= limit)
    limit     int  // 已讀的字符數(shù)不能超過 limit(好像就當(dāng)做常量在使用井联,用于復(fù)位 argLimit)
    maxWid    int  // 存儲占位符中指定的寬度值
}

// 實現(xiàn) ScanState 接口
// Read 方法僅用于 ScanState 以滿足 io.Reader 接口卜壕。
// 在內(nèi)部永遠(yuǎn)不會調(diào)用它,所以沒有必要讓它有任何動作烙常。
func (s *ss) Read(buf []byte) (n int, err error) {
    return 0, errors.New("ScanState's Read should not be called. Use ReadRune")
}

// 實現(xiàn) ScanState 接口
func (s *ss) ReadRune() (r rune, size int, err error) {
    // 讀到 EOF 或超出讀取限制轴捎,則返回 0 0 nil
    if s.atEOF || s.count >= s.argLimit {
        err = io.EOF
        return
    }
    r, size, err = s.rs.ReadRune()
    if err == nil {
        s.count++ // 統(tǒng)計被讀出的字符數(shù)
        if s.nlIsEnd && r == '\n' {
            s.atEOF = true // 拒絕再次 ReadRune
        }
    } else if err == io.EOF {
        s.atEOF = true // 拒絕再次 ReadRune
    }
    return
}

// 實現(xiàn) ScanState 接口
func (s *ss) Width() (wid int, ok bool) {
    if s.maxWid == hugeWid { // hugeWid 是常量 1 << 30
        return 0, false
    }
    return s.maxWid, true
}

// 讀取一個字符,如果遇到 EOF 則返回 eof(即 -1)
// 如果遇到其它錯誤,則中止整個掃描過程侦副,返回 err侦锯。
func (s *ss) getRune() (r rune) {
    r, _, err := s.ReadRune()
    if err != nil {
        if err == io.EOF {
            return eof
        }
        s.error(err)
    }
    return
}

// 功能同 getRune,只不過遇到 EOF 也中止整個掃描過程秦驯,返回 err率触。
func (s *ss) mustReadRune() (r rune) {
    r = s.getRune()
    if r == eof {
        s.error(io.ErrUnexpectedEOF)
    }
    return
}

// 實現(xiàn) ScanState 接口
func (s *ss) UnreadRune() error {
    s.rs.UnreadRune()
    s.atEOF = false // 允許再次 ReadRune
    s.count--       // 統(tǒng)計被讀出的字符數(shù)
    return nil      // UnreadRune 可以反復(fù)調(diào)用,不返回錯誤信息汇竭。
}

// 將錯誤信息轉(zhuǎn)換為 panic。
// 用于配合 recover 快速結(jié)束函數(shù)調(diào)用鏈穴张,避免過多的返回值判斷细燎。
// 類似于 break label 的用法。
func (s *ss) error(err error) {
    panic(scanError{err})
}

// 作用同上面的 error 方法
func (s *ss) errorString(err string) {
    panic(scanError{errors.New(err)})
}

// 實現(xiàn) ScanState 接口
func (s *ss) Token(skipSpace bool, f func(rune) bool) (tok []byte, err error) {
    // 遇到本地錯誤則僅僅返回 err 信息皂甘。
    // 遇到其它錯誤則 panic玻驻。
    defer func() {
        if e := recover(); e != nil {
            if se, ok := e.(scanError); ok {
                err = se.err
            } else {
                panic(e)
            }
        }
    }()
    // 確定審查函數(shù)
    if f == nil {
        f = notSpace
    }
    // 準(zhǔn)備緩沖區(qū)給 s.token 用
    s.buf = s.buf[:0]
    tok = s.token(skipSpace, f)
    return
}

// space 是 unicode.White_Space 的拷貝,避免包的深度依賴偿枕。
// 這些都是空白字符的 Unicode 碼點范圍
var space = [][2]uint16{
    {0x0009, 0x000d},
    {0x0020, 0x0020},
    {0x0085, 0x0085},
    {0x00a0, 0x00a0},
    {0x1680, 0x1680},
    {0x2000, 0x200a},
    {0x2028, 0x2029},
    {0x202f, 0x202f},
    {0x205f, 0x205f},
    {0x3000, 0x3000},
}

// 判斷 r 是否為空白字符
func isSpace(r rune) bool {
    // 空白字符的碼點不會超過 2 個字節(jié)
    if r >= 1<<16 {
        return false
    }
    // 開始判斷
    rx := uint16(r)
    for _, rng := range space {
        if rx < rng[0] {
            return false
        }
        if rx <= rng[1] {
            return true
        }
    }
    return false
}

// notSpace 是 Token 中的默認(rèn)審查函數(shù)璧瞬。
func notSpace(r rune) bool {
    return !isSpace(r)
}

// 實現(xiàn) ScanState 接口
func (s *ss) SkipSpace() {
    s.skipSpace(false)
}

// readRune 用于將 io.Reader 包裝成 io.RuneScanner
type readRune struct {
    reader   io.Reader         // 被包裝的 io.Reader
    buf      [utf8.UTFMax]byte // 僅在 ReadRune 方法中使用
    pending  int               // pendBuf 中存放的字節(jié)數(shù),遇到無效 UTF8 編碼時使用渐夸。
    pendBuf  [utf8.UTFMax]byte // 存放讀取的無效 UTF-8 編碼嗤锉,一次處理不完,留著下次處理
    peekRune rune              // 用于 UnreadRune 存放撤銷的字符墓塌。
}

// readByte 讀取一個字節(jié)
// 它可能是上次 ReadRune 時未處理完的不完整 UTF8 編碼瘟忱。
func (r *readRune) readByte() (b byte, err error) {
    // 如果之前的 ReadRune 有未處理完的字節(jié),則重新讀出這些字節(jié)苫幢。
    if r.pending > 0 {
        // 讀出一個字節(jié)
        b = r.pendBuf[0]
        // 剩下的字節(jié)向前移動一格
        copy(r.pendBuf[0:], r.pendBuf[1:])
        r.pending--
        return
    }
    // 如果沒有未處理的字節(jié)访诱,則從輸入端讀出一個字節(jié)
    n, err := io.ReadFull(r.reader, r.pendBuf[:1])
    if n != 1 {
        return 0, err
    }
    return r.pendBuf[0], err
}

// 實現(xiàn) io.RuneScanner 接口
func (r *readRune) ReadRune() (rr rune, size int, err error) {
    // 之前 UnreadRune 撤銷的字符,存在 peekRune 中韩肝,有則直接取出触菜。
    if r.peekRune >= 0 {
        rr = r.peekRune
        // 將 peekRune 取反為負(fù)數(shù),表示允許 UnreadRune 執(zhí)行撤銷操作
        r.peekRune = ^r.peekRune
        size = utf8.RuneLen(rr)
        return
    }
    // 沒有撤銷的字符哀峻,則從輸入端讀取一個
    r.buf[0], err = r.readByte()
    if err != nil {
        return
    }
    // 如果讀出的是一個單字節(jié)字符涡相,則讀取完畢。
    if r.buf[0] < utf8.RuneSelf {
        rr = rune(r.buf[0])
        size = 1
        // 將讀出的內(nèi)容寫入 peekRune 后取反谜诫,以便 UnreadRune 可以撤銷漾峡。
        r.peekRune = ^rr
        return
    }
    // 讀出的不是單字節(jié)字符
    var n int
    // FullRune 的功能不太好理解,總的來說喻旷,就是判斷首字符的編碼長度是否完整生逸,
    // 如果不完整則返回 false,其它情況都返回 true(包括無效編碼)。
    // 循環(huán)直到 r.buf[:n] 是完整的 UTF-8 編碼(或無效編碼)
    for n = 1; !utf8.FullRune(r.buf[:n]); n++ {
        // 如果字符編碼長度不夠槽袄,則再讀出一個字節(jié)烙无,繼續(xù)判斷
        r.buf[n], err = r.readByte()
        if err != nil {
            if err == io.EOF {
                err = nil // 之前有讀出的字節(jié)未處理,跳出去處理
                break
            }
            return
        }
    }
    // 解碼剛讀出的 UTF-8 序列
    rr, size = utf8.DecodeRune(r.buf[:n])
    if size < n {
        // 遇到錯誤遍尺,保存未處理的字節(jié)截酷,用于下一次讀取。
        copy(r.pendBuf[r.pending:], r.buf[size:n])
        r.pending += n - size
    }
    // 將讀出的內(nèi)容寫入 peekRune 后取反乾戏,以便 UnreadRune 可以撤銷迂苛。
    r.peekRune = ^rr
    return
}

// 實現(xiàn) io.RuneScanner 接口
func (r *readRune) UnreadRune() error {
    // 之前執(zhí)行過 UnreadRune,不能重復(fù)執(zhí)行鼓择。
    // 只有 UnreadRune 才能讓 peekRune 大于 0三幻。
    if r.peekRune >= 0 {
        return errors.New("fmt: scanning called UnreadRune with no rune available")
    }
    // 開始撤銷
    // 反轉(zhuǎn) peekRune 中的二進(jìn)制位,使其成為有效的字符呐能。
    r.peekRune = ^r.peekRune
    return nil
}

// 臨時對象池
var ssFree = sync.Pool{
    New: func() interface{} { return new(ss) },
}

// 創(chuàng)建掃描器念搬,或從臨時對象池中獲取一個。
func newScanState(r io.Reader, nlIsSpace, nlIsEnd bool) (s *ss, old ssave) {
    // 從臨時對象池中獲取一個掃描器
    s = ssFree.Get().(*ss)
    // 如果參數(shù) r 不是一個 RuneScanner摆出,則將其包裝成 RuneScanner 再賦值給 s.rs
    if rs, ok := r.(io.RuneScanner); ok {
        s.rs = rs
    } else {
        // 注意:r 只有 Read 方法朗徊,沒有撤銷方法,所以這里包裝的 readRune 對象
        // 無法將 UnreadRune 所撤銷的內(nèi)容返回到 r 中偎漫。也就是說爷恳,盡量為 r 提供
        // RuneScanner,否則可能造成 r 中的數(shù)據(jù)丟失骑丸。
        s.rs = &readRune{reader: r, peekRune: -1}
    }
    // 復(fù)位參數(shù)
    s.nlIsSpace = nlIsSpace
    s.nlIsEnd = nlIsEnd
    s.atEOF = false
    s.limit = hugeWid
    s.argLimit = hugeWid
    s.maxWid = hugeWid
    s.validSave = true
    s.count = 0
    return
}

// 回收掃描器舌仍,避免再次分配。
func (s *ss) free(old ssave) {
    // 如果掃描器被遞歸使用通危,則只需要恢復(fù)舊狀態(tài)铸豁,然后繼續(xù)使用。
    if old.validSave {
        s.ssave = old
        return
    }
    // 不回收緩沖區(qū)太大的掃描器菊碟,避免內(nèi)存浪費节芥。
    if cap(s.buf) > 1024 {
        return
    }
    // 復(fù)位并存入
    s.buf = s.buf[:0]
    s.rs = nil
    ssFree.Put(s)
}

// 用于實現(xiàn) ScanState 接口
func (s *ss) skipSpace(stopAtNewline bool) {
    for {
        r := s.getRune()
        if r == eof {
            return
        }
        // \r\n 當(dāng) \n 處理
        // peek 判斷即將讀取的字符是否在字符串 "\n" 中(只判斷不讀取)逆害。
        if r == '\r' && s.peek("\n") {
            continue
        }
        if r == '\n' {
            if stopAtNewline { // 換行符之后停止讀取
                break
            }
            if s.nlIsSpace { // 換行符當(dāng)空白處理
                continue
            }
            // 換行符當(dāng)非空白字符處理
            // 在這里不允許头镊,所以中止整個掃描過程,返回 err魄幕。
            s.errorString("unexpected newline")
            return
        }
        // 非空白字符相艇,撤銷讀取并返回。
        if !isSpace(r) {
            s.UnreadRune()
            break
        }
    }
}

// 用于實現(xiàn) ScanState 接口
func (s *ss) token(skipSpace bool, f func(rune) bool) []byte {
    if skipSpace {
        s.skipSpace(false)
    }
    // 循環(huán)讀取直到不滿足 f(r) 或遇到 EOF
    for {
        r := s.getRune()
        if r == eof {
            break
        }
        if !f(r) {
            s.UnreadRune()
            break
        }
        s.buf.WriteRune(r)
    }
    return s.buf
}

var complexError = errors.New("syntax error scanning complex number")
var boolError = errors.New("syntax error scanning boolean")

// 返回 r 在 s 中的字符序號(不是字節(jié)下標(biāo))
func indexRune(s string, r rune) int {
    for i, c := range s {
        if c == r {
            return i
        }
    }
    return -1
}

// 判斷即將讀取的字符是否在 ok 中纯陨。
// 如果 accept 為 flase 則讀取并丟棄該字符坛芽,無論結(jié)果如何留储。
// 如果 accept 為 true,則根據(jù)結(jié)果做進(jìn)一步處理:
// 結(jié)果為 true :將字符讀入 s.buf 中
// 結(jié)果為 false:不讀取該字符
func (s *ss) consume(ok string, accept bool) bool {
    r := s.getRune()
    if r == eof {
        return false
    }
    // r 在 ok 中
    if indexRune(ok, r) >= 0 {
        if accept {
            s.buf.WriteRune(r)
        }
        return true
    }
    // r 不在 ok 中(上面已經(jīng)判斷過 r == eof咙轩,所以這里沒必要再次判斷)
    if r != eof && accept {
        s.UnreadRune() // 不讀取該字符
    }
    return false
}

// 判斷即將讀取的字符是否在 ok 中获讳,但不讀取該字符。
func (s *ss) peek(ok string) bool {
    r := s.getRune()
    if r != eof {
        s.UnreadRune()
    }
    // 在 ok 中查找 r 的下標(biāo)活喊,判斷您是否 >= 0
    return indexRune(ok, r) >= 0
}

// 判斷輸入端是否有數(shù)據(jù)可讀
// 如果沒有數(shù)據(jù)可讀丐膝,則中止整個掃描過程,返回 err钾菊。
func (s *ss) notEOF() {
    if r := s.getRune(); r == eof {
        panic(io.EOF)
    }
    s.UnreadRune()
}

// 判斷即將讀取的字符是否在 ok 中帅矗,如果在,則將其讀入 s.buf 中煞烫,
// 并返回 true损晤,否則不讀取,并返回 false红竭。
func (s *ss) accept(ok string) bool {
    return s.consume(ok, true)
}

// 判斷 verb 是否在 okVerbs 中,
// 如果在喘落,則返回 true茵宪。如果不在,則中止整個掃描過程瘦棋,返回 err稀火。
// 沒有返回 false 的情況。typ 用于在 err 中指示類型信息赌朋。
func (s *ss) okVerb(verb rune, okVerbs, typ string) bool {
    for _, v := range okVerbs {
        if v == verb {
            return true
        }
    }
    s.errorString("bad verb '%" + string(verb) + "' for " + typ)
    return false
}

// 從輸入端讀取一個布爾值凰狞,verb 必須為 t 或 v,否則讀取失敗沛慢。
// 可探測 0赡若、1、t团甲、f逾冬、true、false躺苦,忽略大小寫身腻。
func (s *ss) scanBool(verb rune) bool {
    // 跳過行首空白(包括換行符)
    s.skipSpace(false)
    // 輸入端必須有內(nèi)容可讀
    s.notEOF()
    // 動詞不是 t 或 v,不符合布爾型的要求
    if !s.okVerb(verb, "tv", "boolean") {
        return false
    }
    // 布爾型的語法檢測很討厭匹厘,我們不做嚴(yán)格要求嘀趟。
    // 如果遇到不完整的 tr、tru 或 fa愈诚、fal她按、fals 則中止整個掃描過程牛隅,返回 err。
    switch s.getRune() {
    case '0':
        return false
    case '1':
        return true
    case 't', 'T':
        if s.accept("rR") && (!s.accept("uU") || !s.accept("eE")) {
            s.error(boolError)
        }
        return true
    case 'f', 'F':
        if s.accept("aA") && (!s.accept("lL") || !s.accept("sS") || !s.accept("eE")) {
            s.error(boolError)
        }
        return false
    }
    return false
}

// 數(shù)值元素
const (
    binaryDigits      = "01"
    octalDigits       = "01234567"
    decimalDigits     = "0123456789"
    hexadecimalDigits = "0123456789aAbBcCdDeEfF"
    sign              = "+-"
    period            = "."
    exponent          = "eEp"
)

// 返回 verb 所代表的進(jìn)位制尤溜,及其字符范圍(即上面的常量)
func (s *ss) getBase(verb rune) (base int, digits string) {
    // 判斷 verb 是否符合整型要求倔叼。
    // 如果不符合,則中止整個掃描過程宫莱,返回 err丈攒。
    s.okVerb(verb, "bdoUxXv", "integer")
    base = 10 // 默認(rèn)為十進(jìn)制
    digits = decimalDigits
    switch verb {
    case 'b': // 二進(jìn)制
        base = 2
        digits = binaryDigits
    case 'o': // 八進(jìn)制
        base = 8
        digits = octalDigits
    case 'x', 'X', 'U': // 十六進(jìn)制
        base = 16
        digits = hexadecimalDigits
    }
    return
}

// 從輸入端讀取數(shù)值字符串到 s.buf 中。
// digits 是可接收的字符范圍(不同進(jìn)位制有不同的字符范圍)
// haveDigits 表示 s.buf 中是否已經(jīng)有數(shù)值存在授霸,
// 如果沒有巡验,則本方法必須讀出數(shù)值,否則中止整個掃描過程碘耳,返回 err显设。
func (s *ss) scanNumber(digits string, haveDigits bool) string {
    if !haveDigits {
        // 輸入端必須有內(nèi)容可讀
        s.notEOF()
        if !s.accept(digits) {
            // 如果沒有讀到指定進(jìn)制的字符,則中止整個掃描過程辛辨,返回 err捕捂。
            s.errorString("expected integer")
        }
    }
    // 繼續(xù)讀取合格的字符,存入 s.buf 中
    for s.accept(digits) {
    }
    // 返回讀出的字符串
    return string(s.buf)
}

// 功能同 ReadRune斗搞,只不過通過 bitSize 限制讀取字符的位寬指攒。
// 如果讀出的字符在指定位寬內(nèi),則返回僻焚,否則中止整個掃描過程允悦,返回 err。
func (s *ss) scanRune(bitSize int) int64 {
    s.notEOF()
    r := int64(s.getRune())
    n := uint(bitSize)
    // 位寬判斷
    x := (r << (64 - n)) >> (64 - n)
    if x != r {
        s.errorString("overflow on character value " + string(r))
    }
    return r
}

// 根據(jù)輸入端的前導(dǎo)符 0 或 0x 判斷進(jìn)位制并返回虑啤,同時返回字符范圍隙弛。
// found 表示檢測到前導(dǎo)符。只有當(dāng)動詞是 %v 的時候才會被調(diào)用狞山。
func (s *ss) scanBasePrefix() (base int, digits string, found bool) {
    // 如果不是以 0 開頭全闷,表示是十進(jìn)制數(shù)
    if !s.peek("0") {
        return 10, decimalDigits, false
    }
    // 如果是 0 開頭,則將其讀入 s.buf 中
    s.accept("0")
    found = true // 已經(jīng)讀出一個 0萍启,如果前導(dǎo)符后面沒有數(shù)值室埋,將使用該 0 值。
    // 繼續(xù)判斷是八進(jìn)制還是十六進(jìn)制
    base, digits = 8, octalDigits
    if s.peek("xX") {
        s.consume("xX", false) // 丟棄匹配的 x 或 X 字符
        base, digits = 16, hexadecimalDigits
    }
    return
}

// 讀取一個 int64 整數(shù)伊约。bitSize 用于限制整數(shù)的位寬姚淆。
// 如果讀出的整數(shù)在指定位寬內(nèi),則返回屡律,否則中止整個掃描過程腌逢,返回 err。
func (s *ss) scanInt(verb rune, bitSize int) int64 {
    // 只需要讀取一個字符
    if verb == 'c' {
        return s.scanRune(bitSize)
    }
    s.skipSpace(false)
    s.notEOF()
    // 根據(jù)不同的動詞獲取進(jìn)位制信息
    base, digits := s.getBase(verb)
    haveDigits := false // 是否已經(jīng)讀出數(shù)值
    if verb == 'U' {
        // 丟棄前導(dǎo)符 U+
        // 如果沒有讀取到 U+ 則中止整個掃描過程超埋,返回 err搏讶。
        if !s.consume("U", false) || !s.consume("+", false) {
            s.errorString("bad unicode format ")
        }
    } else {
        // sign 是常量 +-佳鳖,如果能讀取到符號,則將其存入 s.buf 中媒惕。
        s.accept(sign)
        if verb == 'v' {
            // 根據(jù)輸入端的前導(dǎo)符 0 或 0x 確定進(jìn)位制
            // 如果有前導(dǎo)符录粱,則已經(jīng)讀出一個 0赶促,前導(dǎo)符后面可以沒有數(shù)值宣决。
            base, digits, haveDigits = s.scanBasePrefix()
        }
    }
    // 讀出數(shù)值字符串残揉,如果讀取失敗,則中止整個掃描過程肴盏,返回 err科盛。
    tok := s.scanNumber(digits, haveDigits)
    // 轉(zhuǎn)換為整型
    i, err := strconv.ParseInt(tok, base, 64)
    if err != nil {
        s.error(err)
    }
    // 位寬判斷
    n := uint(bitSize)
    x := (i << (64 - n)) >> (64 - n)
    if x != i {
        s.errorString("integer overflow on token " + tok)
    }
    return i
}

// 功能同 scanInt,只不過返回的是無符號整數(shù)菜皂。
func (s *ss) scanUint(verb rune, bitSize int) uint64 {
    if verb == 'c' {
        return uint64(s.scanRune(bitSize))
    }
    s.skipSpace(false)
    s.notEOF()
    base, digits := s.getBase(verb)
    haveDigits := false
    if verb == 'U' {
        if !s.consume("U", false) || !s.consume("+", false) {
            s.errorString("bad unicode format ")
        }
    } else if verb == 'v' {
        base, digits, haveDigits = s.scanBasePrefix()
    }
    tok := s.scanNumber(digits, haveDigits)
    i, err := strconv.ParseUint(tok, base, 64)
    if err != nil {
        s.error(err)
    }
    n := uint(bitSize)
    x := (i << (64 - n)) >> (64 - n)
    if x != i {
        s.errorString("unsigned integer overflow on token " + tok)
    }
    return i
}

// 讀取一個浮點數(shù)贞绵,如果指定了寬度值,則不會超過寬度值恍飘。
// 沒有檢查“只有指數(shù)沒有小數(shù)”的情況榨崩,但是 Atof 會進(jìn)行檢查。
func (s *ss) floatToken() string {
    s.buf = s.buf[:0]
    // 非數(shù)值 NAN
    if s.accept("nN") && s.accept("aA") && s.accept("nN") {
        return string(s.buf)
    }
    // 符號 +-
    s.accept(sign)
    // 無窮大 INF
    if s.accept("iI") && s.accept("nN") && s.accept("fF") {
        return string(s.buf)
    }
    // 整數(shù)部分
    for s.accept(decimalDigits) {
    }
    // 小數(shù)點
    if s.accept(period) {
        // 小數(shù)部分
        for s.accept(decimalDigits) {
        }
    }
    // 指數(shù)標(biāo)志
    if s.accept(exponent) {
        // 指數(shù)符號
        s.accept(sign)
        // 指數(shù)值
        for s.accept(decimalDigits) {
        }
    }
    return string(s.buf)
}

// 讀出一個虛數(shù)的實部和虛部章母。
// 虛數(shù)可以加上括號蜡饵,虛數(shù)格式必須為 N+Ni,N 必須是浮點數(shù)胳施,中間不能有空格。
func (s *ss) complexTokens() (real, imag string) {
    // TODO: 未實現(xiàn)純實部和純虛部的讀取
    parens := s.accept("(") // 登記是否以 "(" 開頭
    real = s.floatToken()   // 讀取實部
    s.buf = s.buf[:0]
    // 虛部必須有符號
    if !s.accept("+-") { // 讀取符號到 s.buf
        s.error(complexError)
    }
    imagSign := string(s.buf) // 取出符號
    imag = s.floatToken()     // 讀取虛部
    if !s.accept("i") {       // 虛部后面必須為 i
        s.error(complexError)
    }
    // 如果以 "(" 開頭肢专,則必須以 ")" 結(jié)尾舞肆。
    if parens && !s.accept(")") {
        s.error(complexError)
    }
    return real, imagSign + imag
}

// 將一個字符串轉(zhuǎn)換為 float64 類型的值。
// str 要轉(zhuǎn)換的字符串博杖,n:要轉(zhuǎn)換出的浮點數(shù)類型(32 或 64)
// 如果轉(zhuǎn)換失敗椿胯,則中止整個掃描過程,返回 err剃根。
func (s *ss) convertFloat(str string, n int) float64 {
    // Atof 不處理以 2 為底的指數(shù)哩盲,但是它們很容易計算。
    if p := indexRune(str, 'p'); p >= 0 {
        // 獲取小數(shù)部分
        f, err := strconv.ParseFloat(str[:p], n)
        if err != nil {
            if e, ok := err.(*strconv.NumError); ok {
                e.Num = str
            }
            s.error(err)
        }
        // 獲取指數(shù)部分
        m, err := strconv.Atoi(str[p+1:])
        if err != nil {
            //
            if e, ok := err.(*strconv.NumError); ok {
                e.Num = str
            }
            s.error(err)
        }
        // 算出結(jié)果:f * (2 的 m 次方)
        return math.Ldexp(f, m)
    }
    // 普通浮點數(shù)直接轉(zhuǎn)換
    f, err := strconv.ParseFloat(str, n)
    if err != nil {
        s.error(err)
    }
    return f
}

// 讀取一個 complex128 類型的值狈醉。
func (s *ss) scanComplex(verb rune, n int) complex128 {
    // 檢查 verb 的有效性(floatVerbs 是常量 "beEfFgGv")
    if !s.okVerb(verb, floatVerbs, "complex") {
        return 0
    }
    s.skipSpace(false)
    s.notEOF()
    // 讀取實部和虛部
    sreal, simag := s.complexTokens()
    real := s.convertFloat(sreal, n/2)
    imag := s.convertFloat(simag, n/2)
    return complex(real, imag)
}

// 讀取一個字符串廉油。
func (s *ss) convertString(verb rune) (str string) {
    // 檢查 verb 的有效性
    if !s.okVerb(verb, "svqxX", "string") {
        return ""
    }
    s.skipSpace(false)
    s.notEOF()
    switch verb {
    case 'q': // 帶引號字符串
        str = s.quotedString()
    case 'x', 'X': // 十六進(jìn)制格式的字符串
        str = s.hexString()
    default:
        // %s 和 %v 僅返回連續(xù)的非空白字符
        str = string(s.token(true, notSpace))
    }
    return
}

// 讀取雙引號或反引號字符串。
func (s *ss) quotedString() string {
    s.notEOF()
    quote := s.getRune()
    switch quote {
    case '`':
        // 讀取直到遇到下一個反引號或 EOF
        for {
            r := s.mustReadRune()
            if r == quote {
                break
            }
            s.buf.WriteRune(r)
        }
        return string(s.buf)
    case '"':
        s.buf.WriteByte('"')
        for {
            r := s.mustReadRune()
            s.buf.WriteRune(r)
            if r == '\\' {
                // strconv.Unquote 會處理轉(zhuǎn)義字符苗傅,這里只需要寫入抒线。
                s.buf.WriteRune(s.mustReadRune())
            } else if r == '"' {
                break
            }
        }
        result, err := strconv.Unquote(string(s.buf))
        if err != nil {
            s.error(err)
        }
        return result
    default:
        s.errorString("expected quoted string")
    }
    return ""
}

// hexDigit 返回十六進(jìn)制字符所代表的十進(jìn)制值
func hexDigit(d rune) (int, bool) {
    digit := int(d)
    switch digit {
    case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
        return digit - '0', true
    case 'a', 'b', 'c', 'd', 'e', 'f':
        return 10 + digit - 'a', true
    case 'A', 'B', 'C', 'D', 'E', 'F':
        return 10 + digit - 'A', true
    }
    return -1, false
}

// 讀取兩個十六進(jìn)制字符,并返回其所表示的字節(jié)渣慕。
// b :讀取的字節(jié)
// ok:是否讀取成功
// 如果缺少后一個字符嘶炭,則中止整個掃描過程抱慌,返回 err。
func (s *ss) hexByte() (b byte, ok bool) {
    // 處理第一個字符
    rune1 := s.getRune()
    if rune1 == eof {
        return
    }
    value1, ok := hexDigit(rune1)
    if !ok {
        s.UnreadRune()
        return
    }
    // 處理第二個字符
    value2, ok := hexDigit(s.mustReadRune())
    if !ok {
        s.errorString("illegal hex digit")
        return
    }
    // 轉(zhuǎn)換為十進(jìn)制數(shù)值
    return byte(value1<<4 | value2), true
}

// 讀取十六進(jìn)制字符串眨猎,并返回其所表示的內(nèi)容
// 兩個十六進(jìn)制字符表示一個字節(jié)
// 讀取失敗則中止整個掃描過程抑进,返回 err。
func (s *ss) hexString() string {
    s.notEOF()
    for {
        // 讀取兩個十六進(jìn)制字符所表示的一個字節(jié)
        b, ok := s.hexByte()
        if !ok {
            break
        }
        s.buf.WriteByte(b)
    }
    if len(s.buf) == 0 {
        s.errorString("no hex data for %x string")
        return ""
    }
    return string(s.buf)
}

const (
    floatVerbs = "beEfFgGv"

    hugeWid = 1 << 30

    intBits     = 32 << (^uint(0) >> 63)
    uintptrBits = 32 << (^uintptr(0) >> 63)
)

// 處理一個 arg睡陪。
// 遇到錯誤則中止整個掃描過程寺渗,返回 err。
func (s *ss) scanOne(verb rune, arg interface{}) {
    s.buf = s.buf[:0]
    var err error
    // 如果參數(shù)有它自己的 Scan 方法宝穗,則調(diào)用它户秤。
    if v, ok := arg.(Scanner); ok {
        err = v.Scan(s, verb)
        if err != nil {
            if err == io.EOF {
                err = io.ErrUnexpectedEOF
            }
            s.error(err)
        }
        return
    }

    // 根據(jù)不同的 arg 類型選擇不同的解析方法。
    switch v := arg.(type) {
    case *bool:
        *v = s.scanBool(verb)
    case *complex64:
        *v = complex64(s.scanComplex(verb, 64))
    case *complex128:
        *v = s.scanComplex(verb, 128)
    case *int:
        *v = int(s.scanInt(verb, intBits))
    case *int8:
        *v = int8(s.scanInt(verb, 8))
    case *int16:
        *v = int16(s.scanInt(verb, 16))
    case *int32:
        *v = int32(s.scanInt(verb, 32))
    case *int64:
        *v = s.scanInt(verb, 64)
    case *uint:
        *v = uint(s.scanUint(verb, intBits))
    case *uint8:
        *v = uint8(s.scanUint(verb, 8))
    case *uint16:
        *v = uint16(s.scanUint(verb, 16))
    case *uint32:
        *v = uint32(s.scanUint(verb, 32))
    case *uint64:
        *v = s.scanUint(verb, 64)
    case *uintptr:
        *v = uintptr(s.scanUint(verb, uintptrBits))
    case *float32:
        if s.okVerb(verb, floatVerbs, "float32") {
            s.skipSpace(false)
            s.notEOF()
            *v = float32(s.convertFloat(s.floatToken(), 32))
        }
    case *float64:
        if s.okVerb(verb, floatVerbs, "float64") {
            s.skipSpace(false)
            s.notEOF()
            *v = s.convertFloat(s.floatToken(), 64)
        }
    case *string:
        *v = s.convertString(verb)
    case *[]byte:
        // 先掃描成字符串逮矛,然后再轉(zhuǎn)換為 []byte鸡号,所以得到的是一個副本,
        // 如果我們掃描成 []byte须鼎,那么結(jié)果將指向緩沖區(qū)鲸伴。
        *v = []byte(s.convertString(verb))
    default:
        val := reflect.ValueOf(v)
        ptr := val
        // arg 必須是一個指針,就像其它 arg 那樣
        if ptr.Kind() != reflect.Ptr {
            s.errorString("type not a pointer: " + val.Type().String())
            return
        }
        // 接下來的流程和上面一樣
        switch v := ptr.Elem(); v.Kind() {
        case reflect.Bool:
            v.SetBool(s.scanBool(verb))
        case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
            v.SetInt(s.scanInt(verb, v.Type().Bits()))
        case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
            v.SetUint(s.scanUint(verb, v.Type().Bits()))
        case reflect.String:
            v.SetString(s.convertString(verb))
        case reflect.Slice:
            typ := v.Type()
            // 對于切片晋控,只能處理 []byte 的別名類型汞窗。
            if typ.Elem().Kind() != reflect.Uint8 {
                s.errorString("can't scan type: " + val.Type().String())
            }
            // 解析出字符串
            str := s.convertString(verb)
            // 轉(zhuǎn)換為字節(jié)切片返回
            v.Set(reflect.MakeSlice(typ, len(str), len(str)))
            for i := 0; i < len(str); i++ {
                v.Index(i).SetUint(uint64(str[i]))
            }
        case reflect.Float32, reflect.Float64:
            s.skipSpace(false)
            s.notEOF()
            v.SetFloat(s.convertFloat(s.floatToken(), v.Type().Bits()))
        case reflect.Complex64, reflect.Complex128:
            v.SetComplex(s.scanComplex(verb, v.Type().Bits()))
        default:
            s.errorString("can't scan type: " + val.Type().String())
        }
    }
}

// 將本地引發(fā)的 panic(scanError 類型)和 EOF panic 轉(zhuǎn)換為 error。
func errorHandler(errp *error) {
    if e := recover(); e != nil {
        // 本地 panic 轉(zhuǎn)換為 error
        if se, ok := e.(scanError); ok {
            *errp = se.err
            // EOF panic 也轉(zhuǎn)換為 error
        } else if eof, ok := e.(error); ok && eof == io.EOF {
            *errp = eof
        } else {
            panic(e)
        }
    }
}

// 掃描器的掃描過程
func (s *ss) doScan(a []interface{}) (numProcessed int, err error) {
    defer errorHandler(&err)

    // 循環(huán)處理所有 arg
    for _, arg := range a {
        s.scanOne('v', arg)
        numProcessed++
    }

    // 所有參數(shù)掃描結(jié)束

    // 檢查是否以換行符或 EOF 結(jié)尾(Scanln 等需要這個錯誤信息)
    if s.nlIsEnd {
        for {
            r := s.getRune()
            if r == '\n' || r == eof {
                break
            }
            // 跳過空白字符后再次判斷
            if !isSpace(r) {
                s.errorString("expected newline")
                break
            }
        }
    }
    return
}

// 處理非占位字符串赡译,返回已處理的字節(jié)數(shù)仲吏。處理結(jié)果分為以下幾種情況:
// 遇到占位符               :返回 % 之前的字節(jié)數(shù)
// 不匹配                   :返回 -1
// 完全匹配(format 被讀完):返回 len(foramt)
// 輸入端被讀完             :強行中止掃描
// advance 的邏輯比較復(fù)雜,要配合 doScanf 理解蝌焚,很難完全理解裹唆。
func (s *ss) advance(format string) (i int) {
    // 這里的 format 不是完整的格式字符串,而是由 doScanf 提供的
    // 未處理部分的格式字符串只洒。doScanf 處理完一個占位符后许帐,就把
    // 剩下的格式字符串交給 advance 處理。
    for i < len(format) {
        // 解碼一個待處理字符
        fmtc, w := utf8.DecodeRuneInString(format[i:])

        // 1毕谴、處理遇到的 % 號

        if fmtc == '%' {
            // 不能以 % 結(jié)尾
            if i+w == len(format) {
                s.errorString("missing verb: % at end of format string")
            }
            nextc, _ := utf8.DecodeRuneInString(format[i+w:])
            // 遇到單獨的 %(占位符)則返回 % 的下標(biāo) i(即 % 之前已處理的字節(jié)數(shù))
            if nextc != '%' {
                return
            }
            // %% 被解析為一個 %成畦,當(dāng)做普通字符,交給后面處理
            i += w // 跳過 %% 中的前一個 %
        }

        // 2涝开、處理 format 中的連續(xù)空白字符

        sawSpace := false   // 是否遇到連續(xù)的空白字符(包括換行符)
        wasNewline := false // 是否遇到換行符

        // 跳過連續(xù)的空白符
        for isSpace(fmtc) && i < len(format) {
            if fmtc == '\n' {
                if wasNewline {
                    // 一次只處理一個換行符循帐,之后的換行符交給后面處理
                    break
                }
                // 登記遇到換行符
                wasNewline = true
            }
            // 登記遇到空白字符
            sawSpace = true
            i += w // 跳過已處理的空白字符
            // 更新待處理字符
            fmtc, w = utf8.DecodeRuneInString(format[i:])
        }

        // 到此,表示沒有連續(xù)空白或已跳過連續(xù)空白舀武,
        // 此時 i 指向非空白字符或換行符(即前面遇到的未處理的換行符)惧浴。

        // 3、對比輸入端的連續(xù)空白字符

        if sawSpace {
            inputc := s.getRune()
            if inputc == eof {
                // 輸入端被讀空奕剃,返回已處理的字節(jié)數(shù)衷旅。
                // 返回后捐腿,在 doScanf 中繼續(xù)判斷 format 是否也被讀完。
                return
            }
            // 輸入端未遇到空白字符柿顶,匹配失敗茄袖,中止整個掃描過程,返回 err嘁锯。
            if !isSpace(inputc) {
                s.errorString("expected space in input to match format")
            }
            // 輸入端也遇到空白字符宪祥,跳過空白部分。
            for inputc != '\n' && isSpace(inputc) {
                inputc = s.getRune()
            }
            // 此時 inputc 有可能為 eof
            // 輸入端遇到換行符
            if inputc == '\n' {
                // format 中未遇到換行符家乘,匹配失敗蝗羊,中止整個掃描過程,返回 err仁锯。
                if !wasNewline {
                    s.errorString("newline in input does not match format")
                }

                // 到此耀找,輸入端和 format 中都遇到換行符,匹配成功业崖。

                // 輸入端換行符之后的空白沒有繼續(xù)處理野芒,而 format 中卻處理了,
                // 這將導(dǎo)致 "\n a %d" 無法匹配 "\n a 1"双炕。使用的時候要注意狞悲。

                // 匹配完畢,返回已處理的字節(jié)數(shù)
                // 這里把 \n 當(dāng)做一次掃描結(jié)束妇斤,這種行為類似于 Scanln摇锋。
                return
            }

            // 輸入端空白字符處理完畢,未遇到換行符站超,則讀取的應(yīng)該是非空白字符荸恕。

            // 撤銷對非空白字符的讀取,交給下一輪去處理顷编。
            // 如果之前讀取的是 eof 則 UnreadRune 不會撤銷任何內(nèi)容。
            s.UnreadRune()
            // format 中遇到換行符剑刑,與輸入端不匹配
            if wasNewline {
                s.errorString("newline in format does not match input")
            }
            // 空白部分(第二個換行符之前的)全部匹配成功媳纬,繼續(xù)下一輪,處理后面的字符施掏。
            continue
        }

        // 到此钮惠,表示 format 中沒遇到空白字符或空白字符已經(jīng)處理完畢。

        // 4七芭、處理 format 中的非空白字符

        // 使用 mustReadRune 而不是 getRune 表示如果讀取失敗(EOF)狸驳,
        // 則中止整個掃描過程缩赛,返回 err撰糠。
        inputc := s.mustReadRune()

        // 非空白字符匹配失敗,撤銷對 input 的讀取阅酪,并返回 -1
        if fmtc != inputc {
            // 匹配失敗,應(yīng)該不需要再做什么了砚尽,不過 advance 作為一個獨立的功能函數(shù),
            // 還是要嚴(yán)謹(jǐn)一些必孤,執(zhí)行 s.UnreadRune 是為了保證輸入端中已處理的內(nèi)容與 
            // format 中 i 的位置對齊。
            s.UnreadRune()
            return -1
        }
        // 非空白字符匹配成功较屿,繼續(xù)處理下一個字符隧魄。
        i += w
    }
    // 全部處理完畢,返回 len(format)
    return
}

// 掃描器的格式化掃描過程
func (s *ss) doScanf(format string, a []interface{}) (numProcessed int, err error) {
    // 消化本地 panic隘蝎,結(jié)束整個掃描過程购啄。
    defer errorHandler(&err)
    end := len(format) - 1
    for i := 0; i <= end; {
        // 先處理 format 中的非占位符部分。
        w := s.advance(format[i:])
        // 循環(huán)直到遇到 % 字符
        if w > 0 {
            i += w
            continue
            // 這里有一個作用嘱么,就是當(dāng) i == len(format) 時狮含,
            // 會終止循環(huán),不會繼續(xù)在后面訪問 format[i]
        }

        // 到這里曼振,表示 format 沒有處理完几迄,而且應(yīng)該處理占位符了。

        // 沒有遇找到占位符冰评,看看是什么原因
        if format[i] != '%' {
            // 非空白字符匹配失敗
            if w < 0 {
                s.errorString("input does not match format")
            }
            // 到此映胁,表示遇到 EOF

            // 不過代碼不會執(zhí)行到這里,因為在 advance 中 EOF 會引發(fā) panic
            // 為了邏輯的嚴(yán)謹(jǐn)甲雅,這里還是需要添加一個 break解孙,以防 advance 發(fā)生改變。
            break
        }

        i++ // 跳過 % 號

        // 讀取占位符中的寬度信息
        var widPresent bool
        s.maxWid, widPresent, i = parsenum(format, i, end)
        // 如果沒有設(shè)置寬度信息抛人,則將寬度設(shè)置為默認(rèn)值 hugeWid
        if !widPresent {
            s.maxWid = hugeWid // hugeWid 是常量 1 << 30
        }
        // 獲取動詞
        c, w := utf8.DecodeRuneInString(format[i:])
        i += w // 跳過動詞

        // 如果動詞不是 c弛姜,則跳過輸入端開頭的空白
        if c != 'c' {
            s.SkipSpace()
        }

        // 默認(rèn)讀取限制
        s.argLimit = s.limit
        // 根據(jù)占位符中的寬度信息設(shè)置輸入端允許讀出的最大字符數(shù)
        if f := s.count + s.maxWid; f < s.argLimit {
            s.argLimit = f
        }

        // arg 太少,占位符太多妖枚,數(shù)量不匹配廷臼。
        if numProcessed >= len(a) {
            s.errorString("too few operands for format '%" + format[i-w:] + "'")
            break
        }
        arg := a[numProcessed]

        s.scanOne(c, arg) // 處理一個 arg
        numProcessed++    // 跳過已處理的 arg
        // 恢復(fù)默認(rèn)讀取限制
        s.argLimit = s.limit
    }
    // arg 太多,占位符太少,數(shù)量不匹配荠商。
    if numProcessed < len(a) {
        s.errorString("too many operands")
    }
    return
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末寂恬,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子结啼,更是在濱河造成了極大的恐慌掠剑,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,627評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件朴译,死亡現(xiàn)場離奇詭異眠寿,居然都是意外死亡盯拱,警方通過查閱死者的電腦和手機(jī)狡逢,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評論 3 399
  • 文/潘曉璐 我一進(jìn)店門奢浑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來雀彼,“玉大人徊哑,你說我怎么就攤上這事莺丑∩颐В” “怎么了蟹漓?”我有些...
    開封第一講書人閱讀 169,346評論 0 362
  • 文/不壞的土叔 我叫張陵源内,是天一觀的道長嗽交。 經(jīng)常有香客問我颂斜,道長沃疮,這世上最難降的妖魔是什么司蔬? 我笑而不...
    開封第一講書人閱讀 60,097評論 1 300
  • 正文 為了忘掉前任肺缕,我火速辦了婚禮授帕,結(jié)果婚禮上跛十,老公的妹妹穿的比我還像新娘偶器。我一直安慰自己屏轰,他們只是感情好霎苗,可當(dāng)我...
    茶點故事閱讀 69,100評論 6 398
  • 文/花漫 我一把揭開白布唁盏。 她就那樣靜靜地躺著厘擂,像睡著了一般刽严。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上管削,一...
    開封第一講書人閱讀 52,696評論 1 312
  • 那天含思,我揣著相機(jī)與錄音含潘,去河邊找鬼调鬓。 笑死腾窝,一個胖子當(dāng)著我的面吹牛居砖,可吹牛的內(nèi)容都是我干的奏候。 我是一名探鬼主播蔗草,決...
    沈念sama閱讀 41,165評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼镶柱,長吁一口氣:“原來是場噩夢啊……” “哼歇拆!你這毒婦竟也來了故觅?” 一聲冷哼從身側(cè)響起渠啊,我...
    開封第一講書人閱讀 40,108評論 0 277
  • 序言:老撾萬榮一對情侶失蹤替蛉,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后盗迟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體罚缕,經(jīng)...
    沈念sama閱讀 46,646評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡邮弹,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,709評論 3 342
  • 正文 我和宋清朗相戀三年腌乡,在試婚紗的時候發(fā)現(xiàn)自己被綠了与纽。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片急迂。...
    茶點故事閱讀 40,861評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡僚碎,死狀恐怖勺阐,靈堂內(nèi)的尸體忽然破棺而出渊抽,到底是詐尸還是另有隱情议忽,我是刑警寧澤徙瓶,帶...
    沈念sama閱讀 36,527評論 5 351
  • 正文 年R本政府宣布灵疮,位于F島的核電站壳繁,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏蒿赢。R本人自食惡果不足惜羡棵,卻給世界環(huán)境...
    茶點故事閱讀 42,196評論 3 336
  • 文/蒙蒙 一皂冰、第九天 我趴在偏房一處隱蔽的房頂上張望秃流。 院中可真熱鬧舶胀,春花似錦、人聲如沸席怪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽声怔。三九已至醋火,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間芥驳,已是汗流浹背兆旬。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評論 1 274
  • 我被黑心中介騙來泰國打工丽猬, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留脚祟,地道東北人。 一個月前我還...
    沈念sama閱讀 49,287評論 3 379
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親柠座。 傳聞我的和親對象是個殘疾皇子妈经,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,860評論 2 361

推薦閱讀更多精彩內(nèi)容