Go 的 bytes 包提供高效字节切片操作函数,支持查找、替换、分割、截取等,避免字符串转换开销,适用于网络、文件和协议解析等场景。
在 Go 中,bytes 包是处理 []byte(字节切片)最常用、最高效的工具。它提供了大量与 strings 包行为一致但专为字节设计的函数,避免了不必要的字符串转换开销,特别适合网络、文件、协议解析等场景。
bytes 提供多个查找函数,全部直接操作 []byte,不涉及 UTF-8 解码或内存拷贝:
bytes.Index(b, sep):返回子切片 sep 在 b 中首次出现的起始索引,未找到返回 -1bytes.LastIndex(b, sep):返回最后一次出现的位置bytes.Contains(b, subslice):返回布尔值,判断是否包含bytes.HasPrefix(b, prefix) 和 bytes.HasSuffix(b, suffix):检查前缀/后缀例如:查找 HTTP 响应头中的换行分隔位置
data := []byte("HTTP/1.1 200 OK\r\nContent-Length: 123\r\n\r\nbody")
pos := bytes.Index(data, []byte("\r\n\r\n"))
if pos != -1 {
headers := data[:pos]
body := data[pos+4:] // 跳过 \r\n\r\n 的 4 字节
}
bytes.Replace 和 bytes.ReplaceAll 支持原地字节级替换,无需转成 string 再转回:
bytes.Replace(b, old, new, n):最多替换 n 次;n 表示全部替换(等价于 ReplaceAll)
[]byte,支持任意字节序列,包括二进制数据或控制字符例如:将日志中的敏感路径 /tmp/xxx 替换为 [REDACTED]
logLine := []byte("error at /tmp/cache.dat: invalid checksum")
clean := bytes.Replace(logLine, []byte("/tmp/"), []byte("[REDACTED]/"), 1)
注意:返回的是新分配的切片,原切片不变(Go 切片本身不可变,函数内部会做 copy)。
避免手动计算索引,用标准方法安全提取片段:
bytes.Split(b, sep):按 sep 分割,返回 [][]byte;空分隔符([]byte{})会 panicbytes.Fields(b):按任意 ASCII 空白字符(\t、\n、\v、\f、\r、' ')分割,自动跳过前后及中间连续空白bytes.Trim(b, cutset)、bytes.TrimSpace(b)、bytes.TrimPre
fix(b, prefix)、bytes.TrimSuffix(b, suffix):去除首尾指定字节例如:解析以 NUL (\x00) 分隔的二进制协议字段
packet := []byte("cmd\x00open\x00file.txt\x00")
parts := bytes.Split(packet, []byte("\x00"))
// parts[0] = "cmd", parts[1] = "open", parts[2] = "file.txt"
高频处理时注意性能:
bytes.Buffer 累积写入,比反复 append 更高效(底层自动扩容)b[i:j],而非 bytes.Copy 或 copy(),零分配bytes.IndexFunc 或使用 bytes.IndexByte(单字节更快)例如:快速提取 HTTP 状态码(空格后第一个单词)
line := []byte("HTTP/1.1 404 Not Found")
start := bytes.IndexByte(line, ' ') + 1
end := bytes.IndexByte(line[start:], ' ')
if end != -1 {
statusCode := line[start : start+end] // "404"
}
不复杂但容易忽略:始终确认输入切片非 nil,bytes 函数对 nil 输入行为不一(如 Index 返回 -1,Split 返回 [][]byte{nil}),建议前置校验或用 len(b) == 0 统一处理空情况。