最稳方案是先用 strconv.parseint(s, 2, 64) 解析二进制字符串为 int64,再用 strconv.formatint(n, 16) 转十六进制;超长串需用 big.int;务必预处理空格、校验非法字符并处理前导零需求。

用 strconv.FormatInt 转二进制字符串到十六进制最稳
别自己写循环除16,Go 标准库早有现成方案。核心是先把二进制字符串转成整数,再格式化为十六进制——两步都得走对,否则会丢前导零或溢出。
-
strconv.ParseInt("1010", 2, 64)是唯一靠谱的解析方式;传"0b1010"会直接报strconv.ParseInt: parsing "0b1010": invalid syntax - 结果必须用
strconv.FormatInt(n, 16),不是fmt.Sprintf("%x", n)——后者在处理负数或大数时行为不一致,且无法控制大小写 - 如果原始二进制串带空格或换行(比如从文件读入),得先
strings.TrimSpace,否则ParseInt直接失败
处理带前导零的二进制输入:别依赖 len() 算位宽
比如输入 "0001",你希望输出 "1" 还是 "01"?标准转换默认去前导零,但有些协议要求保留特定长度(如 8 位一组)。
- 如果要保留固定宽度,不能靠
ParseInt+FormatInt拼凑;得先算原字符串有效位数:strings.TrimLeft("0001", "0")长度为 0 时说明全是 0,要特殊返回"0" - 需要补前导零时,用
fmt.Sprintf("%04x", n)更直观,其中4是目标十六进制字符数(对应 16 位二进制) - 注意:
ParseInt对全 0 字符串返回 0,没问题;但若输入为空或只有 0(如""或"0000"),需提前校验,否则下游 panic
遇到超长二进制串(>64 位)怎么办
标准 ParseInt 最多支持 64 位,但现实里常碰到 128 位 UUID 的二进制表示或哈希值。这时必须切片处理或换库。
- 简单切法:从右往左每 4 位一组(因为 4 位二进制 = 1 位十六进制),用
strconv.ParseUint解析每组,再拼接——但要注意高位补零逻辑 - 更可靠的是用
math/big.Int:new(big.Int).SetString("11010101...", 2),然后调.Text(16);它没长度限制,但性能略低,日常工具够用 - 别用
unsafe或字节操作硬转——二进制字符串不是内存布局,位序和字节序容易搞反
常见错误:把十六进制当成大小写敏感的标识符
十六进制字母 a–f 在协议里通常不区分大小写,但 Go 的 FormatInt 默认小写,ParseInt 也接受大小写混用。问题出在后续比对或存储时。
立即学习“go语言免费学习笔记(深入)”;
- 如果输出要给其他系统消费(比如 Web API),统一用小写更安全;用
strings.ToLower(strconv.FormatInt(n, 16))多此一举,直接传16就是小写 - 若需大写(如某些嵌入式协议),用
strings.ToUpper包一层,或者用fmt.Sprintf("%X", n)——但注意它不兼容负数,所以优先选FormatInt+ToUpper - 测试时别只喂
"1010",一定要试"0"、"1"、"11111111"和含"a"的非法输入,ParseInt错误返回的err必须显式检查
真正麻烦的是二进制串里混了非 0/1 字符,比如不小心粘了 BOM 或不可见空格——这种错误不会报错,但结果完全不对,建议所有输入都先跑一遍 strings.Trim(" \t\n\r", s) == "" || strings.ContainsAny(s, "01") == false 做兜底校验。










