Go语言中的字符串处理方法示例详解

1 概述

字符串,string,一串固定长度的字符连接起来的字符集合。Go语言的字符串是使用UTF-8编码的。UTF-8是Unicode的实现方式之一。

Go语言原生支持字符串。使用双引号("")或反引号(``)定义。

双引号:"", 用于单行字符串。

反引号:``,用于定义多行字符串,内部会原样解析。

示例:

// 单行
"心有猛虎,细嗅蔷薇"
// 多行
`
大风歌
大风起兮云飞扬。
威加海内兮归故乡。
安得猛士兮守四方!
`

字符串支持转义字符,列表如下:

  • \r 回车符(返回行首)
  • \n 换行符(直接跳到下一行的同列位置)
  • \t 制表符
  • \' 单引号
  • \" 双引号
  • \\ 反斜杠
  • \uXXXX Unicode字符码值转义,例如 "\u5eb7" 就是 "康"

Go语言中字符串的顶层结构是由一个指针和长度构成的。使用 unsafe.Sizeof("") 会得到16长度,其中8个字节是指针,指向字符串的内存地址,8个是存储字符串的长度。

2 常规操作

以下是针对字符串的操作总结,主要来自于Go语言的API的说明和测试。

[]索引访问

可以使用[index]方式,访问到字符串中的字符。可以访问,不可以修改。

s := "Hank"
fmt.Printf("%c", s[2])
// 返回 n

unicode/utf8 包

多字节字符的处理,请参考 unicode/utf8 包的相关说明。

例如:

import "unicode/utf8"
utf8.RuneCountInString("小韩说课")
// 返回 4

len(),字符串占用的字节数

utf-8 是变长字符集,英文标点占用1个字节,中文占用3个字节。

len("Hank康")
// 返回 7

+,字符串连接"

"Hello" + " " + "Hank"

==, >, <

字符串比较,比较机制是字符的对称比较。

"abc" > "bbcd"
// 结果为false

strings.Compare(a, b string) int

字符串比较,比较机制是字符的对称比较。返回值为:

0,表示a == b
-1,表示a < b
1,表示a > b
strings.Compare("abc", "abcd")
// 返回 1

strings.Contains(s, substr string) bool

检测字符串 substr 是否在 s 中。

strings.Contains("foobar", "foo")
// 返回 true
strings.Contains("fobar", "foo")
// 返回 false

strings.ContainsAny(s, chars string) bool

检测字符串 chars 的中任意字符是否出现在 s 中。

fmt.Println(strings.ContainsAny("Hank", "kang"))
// 返回 true
fmt.Println(strings.ContainsAny("Hank", "go"))
// 返回 false

strings.ContainsRune(s string, r rune) bool

检测 rune字符是否出现在 s 中。

strings.ContainsRune("Hank", 'a')
// 返回 true
strings.ContainsRune("Hank", 97)
// 返回 true,a的码值97

strings.Count(s, substr string) int

统计字符串 s 中非重叠substr的数量。若统计空字符串"",会返回 s 的长度加1。

strings.Count("HanZhongKang", "n")
// 返回 3
strings.Count("Hank", "")
// 返回 5,"Hank"每个rune的前后都算

strings.EqualFold(s, t string) bool

检测字符串 s 和 t 在忽略大小写的情况下是否相等。

strings.EqualFold("Hank", "hank")
// 返回 true

strings.Fields(s string) []string

返回使用空格分割的字符串 s,结果为切片。

strings.Fields("Han Zhong Kang")
// 返回 []string, ["Han", "Zhong", "Kang"]

strings.FieldsFunc(s string, f func(rune) bool) []string

使用函数确定分隔符,来分割字符串 s。结果是切片。

// ,|/ 都是分隔符
fn := func(c rune) bool {
 return strings.ContainsRune(",|/", c)
}
strings.FieldsFunc("go,python,c++/c,Js|JavaScript", fn)
// 返回 ["go" "python" "c++" "c" "Js" "JavaScript"]

strings.HasPrefix(s, prefix string) bool

检测字符串 s 是否以字符串 prefix 作为前缀。

strings.HasPrefix("Gopher", "Go")
// 返回 true

strings.HasSuffix(s, suffix string) bool

检测字符串 s 是否以字符串 suffix 作为后缀。

strings.HasSuffix("Gopher", "er")
// 返回 true

strings.Index(s, substr string) int

返回字符串 substr 在字符串 s 中第一次出现的索引位置,若没有出现,返回-1。

strings.Index("Gopher", "ph")
// 返回 2

strings.IndexAny(s, chars string) int

返回字符串 chars 中的任意字符在字符串 s 中第一次出现的索引位置,若没有出现,返回-1。

strings.IndexAny("Gopher", "lmno")
// 返回 1

strings.IndexByte(s string, c byte) int

返回byte字符 c 在字符串 s 中第一次出现的索引位置,若没有出现,返回-1。

strings.IndexByte("Gopher", 'h')
// 返回 3

strings.IndexFunc(s string, f func(rune) bool) int

返回字符串 s 中第一次满足函数 f 的rune字符的索引位置,若没有出现,返回-1。

fn := func(c rune) bool {
 return strings.ContainsRune(",|/", c)
}
strings.IndexFunc("go,python,c++/c,Js|JavaScript", fn)
// 返回 2

strings.IndexRune(s string, r rune) int

返回run字符 r 在字符串 s 中第一次出现的索引位置,若没有出现,返回-1。

strings.IndexRune("小韩说课", '说')
// 返回 6

strings.Join(a []string, sep string) string

使用分隔符 sep 连接字符串切片 a。

ss := []string{"Go", "Hank", "Python", "PHP"}
strings.Join(ss, "-")
// 返回 "Go-Hank-Python-PHP"

strings.LastIndex(s, substr string) int

返回字符串 substr 在字符串 s 中最后一次出现的索引位置,若没有出现,返回-1。

strings.LastIndex("Hankang", "an")
// 返回 4

strings.LastIndexAny(s, chars string) int

返回字符串 chars 中的任意字符在字符串 s 中最后一次出现的索引位置,若没有出现,返回-1。

strings.LastIndexAny("Hankang", "lmno")
// 返回 5

strings.LastIndexByte(s string, c byte) int

返回byte字符 c 在字符串 s 中最后一次出现的索引位置,若没有出现,返回-1。

strings.LastIndexByte("Hankang", 'n')
// 返回 5

strings.LastIndexFunc(s string, f func(rune) bool) int

返回字符串 s 中字后一次满足函数 f 的rune字符的索引位置,若没有出现,返回-1。

fn := func(c rune) bool {
 return strings.ContainsRune(",|/", c)
}
strings.LastIndexFunc("go,Js|JavaScript", fn)
// 返回 5

strings.Map(mapping func(rune) rune, s string) string

返回字符串 s 中的每个字符经过映射函数 mapping 处理之后的字符串。

fn := func(c rune) rune {
 if strings.ContainsRune(",|/", c) {
  return '-'
 } else {
  return c
 }
}
strings.Map(fn, "go,Js|JavaScript")
// 返回 "go-Js-JavaScript"

strings.Repeat(s string, count int) string

返回将字符串 s 重复 count 的字符串。

strings.Repeat("la~", 3)
// 返回值 "la~la~la~"

strings.Replace(s, old, new string, n int) string

在字符串 s 中使用字符串 new 替换字符串 old,使用 n 限定替换次数,n设置为负数表示没有限制。返回替换结果。

strings.Replace("han zhong kang", "n", "N", 2)
// 返回 "haN zhoNg kang"

strings.Split(s, sep string) []string

使用分隔符 sep 分割字符串 s,返回字符串切片

strings.Split("go-Js-JavaScript", "-")
// 返回 ["go", "Js", "JavaScript"]

strings.SplitAfter(s, sep string) []string

在分隔符 sep 后分割字符串 s,返回字符串切片

strings.SplitAfter("go-Js-JavaScript", "-")
// 返回 ["go-", "Js-", "JavaScript"]

strings.SplitAfterN(s, sep string, n int) []string

在分隔符 sep 后分割字符串 s,使用 n 限定分割的元素数量,n<0全部子字符串,n>0最后一个子字符串包含余下内容,n==0返回nil。返回子字符串切片。

strings.SplitAfterN("go-Js-JavaScript", "-", 2)
// 返回 ["go-", "Js-JavaScript"]

strings.SplitN(s, sep string, n int) []string

在分隔符 sep 分割字符串 s,使用 n 限定分割的元素数量,n<0全部子字符串,n>0最后一个子字符串包含余下内容,n==0返回nil。返回子字符串切片。

strings.SplitN("go-Js-JavaScript", "-", 2)
// 返回 ["go", "Js-JavaScript"]

strings.Title(s string) string

返回Title化的字符串。

strings.Title("hello Hank's go")
// 返回 "Hello Hank's Go"

strings.ToLower(s string) string

转换字符串 s 到小写。

strings.ToLower("Hank's Go Guide")
// 返回 "hank's go guide"

strings.ToLowerSpecial(c unicode.SpecialCase, s string) string

使用特定的规则转换字符串 s 到小写。

strings.ToLowerSpecial(unicode.TurkishCase, "Önnek İş")
// 返回 önnek iş

strings.ToTitle(s string) string

返回全部字符都Title化的字符串。

strings.Title("hello Hank's go")
// 返回 "HELLO HANK'S GO"

strings.ToTitleSpecial(c unicode.SpecialCase, s string) string

使用特定的规则将全部字符都Title化。

strings.ToTitleSpecial(unicode.TurkishCase, "dünyanın ilk borsa yapısı Aizonai kabul edilir")
// 返回 "DÜNYANIN İLK BORSA YAPISI AİZONAİ KABUL EDİLİR"

strings.ToUpper(s string) string

将字符串 s 中所有字符转换为大写。

strings.ToUpper("hello Hank's go")
// 返回 "HELLO HANK'S GO"

strings.ToUpperSpecial(c unicode.SpecialCase, s string) string

使用特定的规则将字符串 s 中所有字符转换为大写。

strings.ToUpperSpecial(unicode.TurkishCase, "örnek iş")
// 返回 "ÖRNEK İŞ"

strings.Trim(s string, cutset string) string

截取字符串 s 两端包裹的特定字符集 cutset。

strings.Trim(" user name  ", " ")
// 返回 "user name"

strings.TrimFunc(s string, f func(rune) bool) string

截取字符串 s 两端满足函数 f 的字符。

fn := func(c rune) bool {
 return strings.ContainsRune(",|/", c)
}
strings.TrimFunc("|/user name,/", fn)
// 返回 "user name"

strings.TrimLeft(s string, cutset string) string

截取字符串 s 左边包裹的特定字符集 cutset。

strings.TrimLeft(" user name  ", " ")
// 返回 "user name  "

strings.TrimLeftFunc(s string, f func(rune) bool) string

截取字符串 s 左边满足函数 f 的字符。

fn := func(c rune) bool {
 return strings.ContainsRune(",|/", c)
}
strings.TrimLeftFunc("|/user name,/", fn)
// 返回 "user name,/"

strings.TrimPrefix(s, prefix string) string

截取字符串 s 的前缀 prefix。

strings.TrimPrefix("hank_goGuide", "hank_")
// 返回 "goGuide"

strings.TrimRight(s string, cutset string) string

截取字符串 s 右边包裹的特定字符集 cutset。

strings.TrimRight(" user name  ", " ")
// 返回 " user name"

strings.TrimRightFunc(s string, f func(rune) bool) string

截取字符串 s 右边满足函数 f 的字符。

fn := func(c rune) bool {
 return strings.ContainsRune(",|/", c)
}
strings.TrimRightFunc("|/user name,/", fn)
// 返回 "|/user name"

strings.TrimSpace(s string) string

截取字符串 s 两端的空白字符。

strings.TrimSpace(" \t\n Hello, Gophers \n\t\r\n")
// 返回 "Hello, Gophers"

strings.TrimSuffix(s, suffix string) string

截取字符串 s 的后缀 suffix。

strings.TrimSuffix("goGuide_beta", "_beta")
// 返回 "goGuide"

总结

以上所述是小编给大家介绍的Go语言中的字符串处理方法示例详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

时间: 2018-10-28

Go语言中 Channel 详解

Channel是Go中的一个核心类型,你可以把它看成一个管道,通过它并发核心单元就可以发送或者接收数据进行通讯(communication). 它的操作符是箭头 <- . ch <- v    // 发送值v到Channel ch中 v := <-ch  // 从Channel ch中接收数据,并将数据赋值给v (箭头的指向就是数据的流向) 就像 map 和 slice 数据类型一样, channel必须先创建再使用: ch := make(chan int) Channel类型 Cha

Golang数组的传递详解

概念介绍 数组与切片 数组是具有相同唯一类型的一组已编号且长度固定的数据项序列.数组长度最大为2Gb,它是值类型.切片是对数组一个连续片段的引用,所以切片是一个引用类型. 按值传递和按引用传递 Go语言中函数的参数有两种传递方式,按值传递和按引用传递.Go默认使用按值传递来传递参数,也就是传递参数的副本.在函数中对副本的值进行更改操作时,不会影响到原来的变量. 按引用传递其实也可以称作"按值传递",只不过该副本是一个地址的拷贝,通过它可以修改这个值所指向的地址上的值. Go语言中,在函

Go语言的文件操作代码汇总

# 建立与打开文件 // 新建文件可以通过如下两个方法: func Create(name string) (file *File, err Error) 根据提供的文件名创建新的文件,返回一个文件对象,默认权限是0666的文件,返回的文件对象是可读写的. func NewFile(fd uintptr, name string) *File 根据文件描述符创建相应的文件,返回一个文件对象 // 通过如下两个方法来打开文件: func Open(name string) (file *File,

Go语言中http和mysql的实现代码

http 编程 Go 原生支持http: import "net/http" Go 的http服务性能和nginx比较接近: 就是说用Go写的Web程序上线,程序前面不需要再部署nginx的Web服务器,这里省掉的是Web服务器.如果服务器上部署了多个Web应用,还是需要反向代理的,一般这也是nginx或apache. 几行代码就可以实现一个web服务: package main import ( "fmt" "net/http" ) func

Go语言中多字节字符的处理方法详解

1 概述 Go语言的字符串是使用 UTF-8 编码的.UTF-8 是 Unicode 的实现方式之一.本文内容包括:UTF-8 和 Unicode 的关系,Go语言提供的 unicode 包和 unicode/utf8 包的使用. 下面话不多说了,来一起看看详细的介绍吧 2 UTF-8 和 Unicode 的关系 Unicode一种字符集,是国际标谁化组织(ISO)设计的一个包括了地球上所有文化.所有字母和符号 的编码.他们叫它 Universal Multiple-Octet Coded Ch

go语言中strings包的用法汇总

strings 包中的函数和方法 // strings.go ------------------------------------------------------------ // Count 计算字符串 sep 在 s 中的非重叠个数 // 如果 sep 为空字符串,则返回 s 中的字符(非字节)个数 + 1 // 使用 Rabin-Karp 算法实现 func Count(s, sep string) int func main() { s := "Hello,世界!!!!!&quo

Go语言的JSON处理详解

Go语言内建对JSON的支持.使用Go语言内置的encoding/json标准库,开发者可以轻松使用Go程序生成和解析JSON格式的数据.在Go语言实现JSON的编码和解码时,遵循RFC4627协议标准. 1.编码为JSON格式 使用json.Marshal()函数可以对一组数据进行JSON格式的编码.json.Marshal()函数的声明如下: 假如有如下一个Book类型的结构体: 并且有如下一个Book类型的实例对象: 然后,我们可以使用json.Marshal()函数将gobook实例生成

Go语言实现互斥锁、随机数、time、List

Go语言实现互斥锁.随机数.time.List import ( "container/list" "fmt" "math/rand" //备注2:随机数的包 "sync" //备注1:异步任务的包 "time" ) type INFO struct { lock sync.Mutex //备注1:异步锁 Name string Time int64 } var List *list.List = list

go语言中int和byte转换方式

主机字节序 主机字节序模式有两种,大端数据模式和小端数据模式,在网络编程中应注意这两者的区别,以保证数据处理的正确性:例如网络的数据是以大端数据模式进行交互,而我们的主机大多数以小端模式处理,如果不转换,数据会混乱 参考 :一般来说,两个主机在网络通信需要经过如下转换过程:主机字节序 -> 网络字节序 -> 主机字节序 大端小端区别 大端模式:Big-Endian就是高位字节排放在内存的低地址端,低位字节排放在内存的高地址端 低地址 --------------------> 高地址 高

Go语言range关键字循环时的坑

关键字range可用于循环,类似迭代器操作,它可以遍历slice,array,string,map和channel,然后返回索引或值.可以使用"_"来忽略不想要的返回值.可以方便的读取上面类型中的内容,例如: package main import "fmt" func main() { str1 := []string{"1", "2", "3", "4"} for key, valu

浅析Go语言中的Range关键字

前言 相信用过Range的朋友们都知道,Go语言中的range关键字使用起来非常的方便,它允许你遍历某个slice或者map,并通过两个参数(index和value),分别获取到slice或者map中某个元素所在的index以及其值. 比如像这样的用法: for index, value := range mySlice { fmt.Println("index: " + index) fmt.Println("value: " + value) } 上面的例子足够

浅谈c语言中类型隐性转换的坑

谨记:在C语言中,当两种不同类型之间运算时,低字节长度类型会向高自己长度类型转换,有符号会向无符号类型转换. 举例子如下: #include <stdio.h> void func(void) { int i = 1; unsigned char c1 = 1; signed char c2 = -1; if (c2 > i){ printf("\r\n -1 > 1"); } else{ printf("\r\n -1 <= 1");

弱类型语言javascript开发中的一些坑实例小结【变量、函数、数组、对象、作用域等】

本文实例讲述了弱类型语言javascript开发中的一些坑.分享给大家供大家参考,具体如下: 测试1: (未声明变量自动提升为全局变量) test1(); function test1() { function setName() { name = '张三'; // 此处没有var声明,提升至全局 } setName(); console.log(name);// '张三' } 测试2: (函数内部局部变量的变量提升) test2(); function test2() { var a = 1;

C语言实现的循环单链表功能示例

本文实例讲述了C语言实现的循环单链表功能.分享给大家供大家参考,具体如下: SClist.h #ifndef __SCLIST_H__ #define __SCLIST_H__ #include<cstdio> #include<malloc.h> #include<assert.h> typedef int ElemType; typedef struct Node { ElemType data; struct Node *next; }Node,*PNode; t

快速了解C语言静态关键字static的作用

静态关键字static C语言中,static关键字修饰变量和函数 1.局部变量 2.全局变量 3.函数 修饰局部变量 1.用静态关键字static修饰的局部变量,在编译的过程中,会在数据区为该变量开辟空间,并对其进行初始化,如果代码中未对其进行初始化,则系统默认初始化为0 2.用static修饰的局部变量,会延长局部变量的寿命,超出函数的生存期 3.对静态关键字修饰的局部变量的初始化 以下面两个变量a和变量为b进行说明,在编译过程中,发现变量a和b是静态变量,会标识变量a与b,等到程序运行的时

解决PHP里大量数据循环时内存耗尽的方法

最近在开发一个PHP程序的时候遇到如下一问题: PHP Fatal error: Allowed memory size of 268 435 456 bytes exhausted 错误信息显示允许的最大内存已经耗尽.遇到这样的错误起初让我很诧异,但转眼一想,也不奇怪,因为我正在开发的这个程序是要用一个foreach循环语句在一个有4万条记录的表里全表搜索具有特定特征的数据,也就是说,一次要把4万条数据取出,然后逐条检查每天数据.可想而知,4万条数据全部加载到内存中,内存不爆才怪. 毕竟编程这

go语言在请求http时加入自定义http header的方法

本文实例讲述了go语言在请求http时加入自定义http header的方法.分享给大家供大家参考.具体实现方法如下: 复制代码 代码如下: client := &http.Client{] req, err := http.NewRequest("POST", "http://example.com", bytes.NewReader(postData)) req.Header.Add("User-Agent", "myCli

C语言按关键字搜索文件夹中文件的方法

本文实例讲述了C语言按关键字搜索文件夹中文件的方法.分享给大家供大家参考.具体实现方法如下: 方法1: #include<iostream> #include<string> #include<io.h> using namespace std; void filesearch(string path,string mode) { struct _finddata_t filefind; if(path[path.size()-1]=='\\') path.resize

Spring整合Mybatis使用&lt;context:property-placeholder&gt;时的坑

背景 最近项目要上线,需要开发一个数据迁移程序.程序的主要功能就是将一个数据库里的数据,查询出来经过一系列处理后导入另一个数据库.考虑到开发的方便快捷.自然想到用spring和mybatis整合一下.甚至用mybatis的自动代码生成,可以省下大量dao层的开发. 整合的坑 之前的项目:以前也有过这种类似的程序,就把spring和mybatis整合的配置直接拿来修改下用.之前的整合配置是这样子的: 1.考虑到数据库url.用户名密码的可配置性,将这些信息放入properties文件.在sprin