深入理解Go中rune类型的使用

 更新时间:2026年02月05日 09:23:17   作者:Penge666  
本文主要介绍了深入理解Go中rune类型的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

在 Go 中处理字符串时,rune 类型常常被当作 “解决中文截取问题” 的万 能钥 匙。但它的意义远不止于此 ——rune 是 Go 对 Unicode 码点(Code Point)的原生支持,是理解 Go 字符串底层逻辑的关键。今天我们就来深挖 rune 的本质、用法和背后的设计哲学。

一、从一个 “反直觉” 的例子说起

先看一段简单的代码:

package main

import "fmt"

func main() {
    str := "Hello, 世界"
    fmt.Println("字符串长度:", len(str))       // 输出:13
    fmt.Println("第7个字符:", str[6])          // 输出:228(一个奇怪的数字)
}

这段代码的输出可能会让初学者困惑:

  • "Hello, 世界" 直观上是 8 个字符("Hello," + 空格 + "世界"),但 len(str) 返回 13。
  • 尝试通过索引 str[6] 获取第 7 个字符(空格),得到的却是 228 这个数字。

问题出在 Go 字符串的底层实现 ——Go 字符串本质是字节(byte)的切片,而不是 “字符” 的集合。

二、byte 与 rune:两种视角看字符串

1.先明确两个核心概念

  1. 字节(Byte) :计算机存储数据的最小单位之一(1 字节 = 8 比特),只能表示 0-255 的整数(2^8 = 256 种可能)。早期计算机用单字节表示英文字符(如 ASCII 编码:A 对应 65,a 对应 97),但无法表示中文、日文等复杂字符(这些字符需要更多位数)。
  2. Unicode 码点(Code Point) :为解决 “全球字符统一表示” 问题,Unicode 标准给每个字符分配了一个唯一的数字编号(例如:A 是 U+0041,中 是 U+4E2D)。这个编号就是 “码点”,范围从 U+0000 到 U+10FFFF,需要 1-4 字节才能存储。

2.byte:对应单字节字符(ASCII 兼容)

  • 定义:byte 是 uint8 的别名(type byte = uint8),本质是 8 位无符号整数,只能表示 0-255 的值。
  • 用途:用于处理单字节字符(如英文字母、数字、标点),对应 Unicode 码点中 U+0000 到 U+00FF 的范围(即 ASCII 及扩展 ASCII 字符)。

示例:

var b byte = 'A' // 'A' 的 ASCII 码是 65,所以 b 的值是 65
fmt.Println(b) // 输出:65
fmt.Printf("%c\n", b) // 输出:A(用 %c 格式化显示字符)

3.rune:对应 Unicode 码点(处理多字节字符)

  • 定义rune 是 int32 的别名(type rune = int32),本质是 32 位整数,能表示 U+0000 到 U+10FFFF 的所有 Unicode 码点。
  • 用途:用于处理多字节字符(如中文、日文、emoji 等),因为这些字符的 Unicode 码点超过 255,需要用多个字节存储。

示例:

var r rune = '中' // '中' 的 Unicode 码点是 U+4E2D(十进制 20013)
fmt.Println(r) // 输出:20013
fmt.Printf("%c\n", r) // 输出:中

三、rune 的核心应用场景

1. 安全截取包含多字节字符的字符串

这是 rune 最常见的用法。直接对字符串做切片(str[i:j])是按字节截取,可能会截断多字节字符导致乱码;而先转换为 []rune 再切片,则能保证字符的完整性:

func safeSubstr(s string, length int) string {
    runes := []rune(s)
    if len(runes) <= length {
        return s
    }
    return string(runes[:length])
}

func main() {
    str := "Go语言是门好语言"
    fmt.Println(safeSubstr(str, 5)) // 输出:Go语言是门
}

2. 遍历字符串中的每个字符

用 for 循环直接遍历字符串时,得到的是字节;用 for range 循环时,Go 会自动按 rune 迭代(即按字符遍历):

str := "Hello, 世界"

// 按字节遍历(可能得到乱码)
for i := 0; i < len(str); i++ {
    fmt.Printf("%c ", str[i]) 
    // 输出:H e l l o ,   ä ¸  ç 世界的后续字节...(乱码)
}

// 按 rune 遍历(正确输出每个字符)
for _, r := range str {
    fmt.Printf("%c ", r) 
    // 输出:H e l l o ,   世 界 
}

for range 循环本质上是在迭代 []rune(str),因此能正确处理所有 Unicode 字符。

3. 处理 Emoji 和特殊符号

Emoji 通常占 4 个字节(如 😊 的 UTF-8 编码是 0xF0 0x9F 0x98 0x8A),但在 []rune 中依然是一个元素:

str := "Hello 😊"
runes := []rune(str)
fmt.Println(len(runes)) // 输出:7(H e l l o  空格 😊)
fmt.Println(string(runes[6])) // 输出:😊

四、rune 背后的 Unicode 与 UTF-8

理解 rune 必须先明确两个概念:

  • Unicode:一种字符集(Character Set),为每个字符分配唯一的码点(如 'A' 是 65,' 中 ' 是 20013)。
  • UTF-8:一种编码方式(Encoding),将 Unicode 码点转换为字节序列(如 ' 中 ' 的 UTF-8 编码是 0xE4 0xB8 0xAD,3 个字节)。

Go 字符串的底层存储是 UTF-8 编码的字节序列,而 rune 是 Unicode 码点的内存表示。因此:

  • []byte(str) 得到的是 UTF-8 编码的字节切片。
  • []rune(str) 得到的是 Unicode 码点的切片(每个码点对应一个字符)。

二者的转换是 UTF-8 编码与解码的过程,这个过程由 Go runtime 自动完成,无需开发者手动处理。

到此这篇关于深入理解Go中rune类型的使用的文章就介绍到这了,更多相关Go rune类型内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:

相关文章

  • Go语言占位符的使用

    Go语言占位符的使用

    本文主要介绍了Go语言占位符的使用,字符串占位符在fmt包的各种打印函数中使用,下面就一起来介绍一下,感兴趣的可以了解一下
    2024-08-08
  • 详解Go语言的计时器

    详解Go语言的计时器

    Go语言的标准库里提供两种类型的计时器Timer和Ticker。这篇文章主要介绍了Go语言的计时器的相关知识,需要的朋友可以参考下
    2020-05-05
  • go语言中字符串嵌套的几种实现方式

    go语言中字符串嵌套的几种实现方式

    在Go语言中,字符串嵌套可以通过多种方式实现,包括使用双引号和转义字符、反引号、字符串拼接和格式化字符串,下面就来介绍一下,感兴趣的可以了解一下
    2025-03-03
  • Go语言中的访问权限控制

    Go语言中的访问权限控制

    在 go 中进行权限管理时,推荐使用 grouper、casbin 和 sentry 框架,grouper 适合基于角色的访问控制,casbin 提供高级 rbac 功能,而 sentry 提供云托管权限服务和丰富的功能集,包括多因素认证和活动审核,这些框架有助于在电子商务网站等实际场景中实施细粒度的访问控制
    2024-08-08
  • Go语言编写一个简易聊天室服务端的实现

    Go语言编写一个简易聊天室服务端的实现

    本文介绍用Go语言构建TCP聊天室服务端,通过Goroutine实现多客户端并发通信,提供基础聊天功能,具有一定的参考价值,感兴趣的可以了解一下
    2025-08-08
  • Go+Redis缓存设计与优化实现

    Go+Redis缓存设计与优化实现

    本文主要介绍了Go+Redis缓存设计与优化实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2025-02-02
  • Golang中如何使用lua进行扩展详解

    Golang中如何使用lua进行扩展详解

    这篇文章主要给大家介绍了关于Golang中如何使用lua进行扩展的相关资料,这是最近在工作中遇到的一个问题,觉着有必要分享出来给大家学习,文中给出了详细的示例,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-10-10
  • 一文带你搞懂go中的请求超时控制

    一文带你搞懂go中的请求超时控制

    在日常开发中,对于RPC、HTTP调用设置超时时间是非常重要的,这就需要我们设置超时控制,本文将通过相关示例为大家深入介绍一下go中的请求超时控制,希望对大家有所帮助
    2023-11-11
  • golang中defer延迟机制的实现示例

    golang中defer延迟机制的实现示例

    本文讲解了golang特殊的defer延迟机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2025-09-09
  • go语言中的template使用示例详解

    go语言中的template使用示例详解

    在Go语言中,可以通过text/template和html/template包来处理模板,本文提供了一个使用Go模板的基本示例,包括导入包、创建数据结构、定义模板、执行模板及运行程序,通过这些步骤,可以输出一个格式化的YAML配置
    2024-10-10

最新评论