前期回顾

在上面的文章中,老猫和大家分享了GO语言中比较重要的两种数据结构,一种是数组,另外一种是基于数组的slice。本篇文章想要继续和大家分享剩下的容器以及字符字符串的处理。

MAP

map的定义

在Java语言中其实还有一个键值对的容器,叫做Map,当然在我们GO语言中也有这种数据结构,例如,下面咱们就来看一下GO语言中Map的写法以及用法。看下一Map的语法定义,如下例子:

m :=map[string]string {
"name":"ktdaddy",
"age":"28",
"job":"softEngineer",
}

关于语法的定义其实也是比较简单的,方括号里面的表示key值,后面的那个类型表示value的类型。上面老猫其实定义了一个map,并且给它初始化了一些数据。定义一个空的map其实也是比较简单的,主要有下面两种方式

m1 :=make(map[string]int) //采用make函数直接创建一个map,比较常用
var m2 map[string] int //用var函数申明的方式创建map

说实在的语法这种东西也没法说出什么所以然来,大家既然想要学习这个语言就要去记下它的规则,所以还是得多写多练,长时间不去用可能就忘记了。

总结一下其定义形式map[k]V,当然咱们也可以定义它比较复杂一些的数据结构,例如map里面套map,这种复合型的,那么定义的最终形式就是map[K1]map[K2]V。

对map的操作

查询元素

我们定义出一个map之后,正常的逻辑就要去操作这个map,咱们先来看一下如何遍历一个map,我想大家比较容易想到的就是我们讲array的时候用的range函数的方式。确实,咱们可以写成这样,咱们就以一开始定义的m这个Map为例吧

for k,v :=range m {
fmt.Println(k,v)
}

这样我们就可以获取所有的数据,当然我们也可以根据对应的Key值获取相关的value

name :=m["name"]

但是很多时候,我们是不能知道这个key是否存在的,我们取值的时候就要先判断当前的key是否存在,然后再去获取值,其实我们可以这样写

name,ok :=m["name"]
fmt.Println(name,ok) //结果输出ktdaddy true

很显然,我们可以通过ok的值去判断当前的key是否存在,然后再去获取其中的value值。那么结合前面所讲的if条件语句,咱们就可以写出这样的逻辑

if name,ok :=m["name"];ok {
fmt.Println(name)
}else {
fmt.Println("key does not exist")
}
删除元素

删除元素其实是相当简单的,用内置函数delete函数即可。我们还是承接上面的逻辑,如果发现存在这个Key值,那么我们就从map中进行移除。具体如下

if name,ok :=m["name"];ok {
delete(m,"name")
name,ok = m["name"]
fmt.Println(name,ok)
}else {
fmt.Println("key does not exist")
}

上述删除之后咱们重新赋值map,这样这个代码最终输出的结果就是false。大家可以实际动手练习一下。

字符串的处理

老猫想过好几次,如何写这个字符串,想要将其单独抽出来说感觉也不合适,因为咱们很多时候会遍历容器中的字符,因此索性就将其合并在内建容器里面一起分享好了。这里主要想和大家分享的是rune这个内建类型,从前面的章节中我们都知道这个相当于是Go的char,那么这个又怎么来用呢?咱们先来看下面例子

s:= "hello我爱GO开发"
fmt.Println(len(s))

大家觉得这个应该会输出多少呢?试着运行一下,答案是19,那么为什么呢?我们再接着转换成字节输出一下

s:= "hello我爱GO开发"
fmt.Println(s)
fmt.Println(len(s))
fmt.Printf("%X\n",[]byte(s))

这样呢,我们就获取一段16进制的字节,结果如下

68656C6C6FE68891E788B1474FE5BC80E58F91

很显然,我们还是无法看出来,我们将其格式做一下分割调整来看一下

s:= "hello我爱GO开发"
fmt.Println(s)
fmt.Println(len(s))
fmt.Printf("%X\n",[]byte(s)) for _,b:=range []byte(s) {
fmt.Printf("%X ",b)
}

运行的结果为 :68 65 6C 6C 6F E6 88 91 E7 88 B1 47 4F E5 BC 80 E5 8F 91

数一下成对出现的16进制数,咱们就可以发现19个字节长度原来是这么来的。通过分析我们不难发现一个英文占用了两个字节而中文则占用了3个字节,例如里面的“我”转换之后就是:E6 88 91,那么我们直接输出s又会是什么呢?

for i,ch := range s{ // ch is a rune
fmt.Printf("(%d %X) ",i,ch)
}
fmt.Println()

结果为:

(0 68) (1 65) (2 6C) (3 6C) (4 6F) (5 6211) (8 7231) (11 47) (12 4F) (13 5F00) (16 53D1)

其实通过68 65 6C 6C 6F E6 88 91 E7 88 B1 47 4F E5 BC 80 E5 8F 91 转换成了上面这个是通过什么转换的呢?其实通过utf-8转换成了unicode编码,我们发现转换成unicode之后,相关的中文字符编程了长度为4的16进制数,那么其实就是相当于是rune所占的字节长度了。

所以我们直接用rune对其进行转换。

for i,ch :=range []rune(s){
fmt.Printf("(%d %c)",i,ch)
}

这样我们就得到了如下

(0 h)(1 e)(2 l)(3 l)(4 o)(5 我)(6 爱)(7 G)(8 O)(9 开)(10 发)

如此,其实就遍历获取到了每个字符了。

当然,如果我们要统计utf-8下的字符长度,其实我们要对其进行转换,在此介绍两个函数

utf8.RuneCountInString(s)
utf8.DecodeRune(bytes)

通过这两个函数,咱们可以获取中文字符下的具体的字符个数了。

看起来比较零碎,咱们做一下总结,当然剩下还是得大家自己写一下。

  1. 使用range遍历pos,rune对
  2. 使用utf8.RuneCountInString(s)获取字符数量
  3. 使用len获取字节长度
  4. 使用[]byte获取字节

总结

本次主要和大家分享了内建容器的剩余部分map的定义以及相关的操作使用,后面还和大家穿插了部分字符串的处理方式,当然还是比较散的,后面在写项目的过程中,咱们到时候在具体用起来,所以在此希望大家持续关注老猫,另外的也希望大家多练习,一起进步,加油。

我是老猫,更多内容,欢迎大家搜索关注老猫的公众号“程序员老猫”。

跟着老猫来搞GO-内建容器Map的更多相关文章

  1. 跟着老猫来搞GO-容器(1)

    前期回顾 前面的一章主要和大家分享了GO语言的函数的定义,以及GO语言中的指针的简单用法,那么本章,老猫就和大家一起来学习一下GO语言中的容器. 数组 数组的定义 说到容器,大家有编程经验的肯定第一个 ...

  2. 跟着老猫来搞GO,集跬步而致千里

    上次博客中,老猫已经和大家同步了如何搭建相关的GO语言的开发环境,相信在车上的小伙伴应该都已经搞定了环境了.那么本篇开始,我们就来熟悉GO语言的基础语法.本篇搞定之后,其实期待大家可以和老猫一样,能够 ...

  3. 跟着老猫来搞GO,基础进阶

    回顾一下上一篇博客,主要是和大家分享了GO语言的基础语法,其中包含变量定义,基本类型,条件语句,循环语句.那本篇呢就开始和大家同步一下GO语言基础的进阶. 函数的定义 上次其实在很多的DEMO中已经写 ...

  4. 跟着老猫来搞GO,"面向对象"

    前言 之前和大家分享了容器以及相关的基础语法,以及函数,相信如果大家有接触过C++或者java的朋友都晓得面向对象,其实在GO语言中也存在面向对象,但是还是比较简单的,下面我们来看一下GO语言的&qu ...

  5. go内建容器-Map

    1.基础定义 golang中的map如同它的函数一样"纯粹",map就是用来存储键值对的容器,别管什么哈希不哈希的(底层已实现),用就行 //创建一个map m := map[st ...

  6. 跟着老猫一起来学GO,环境搭建

    老猫的GO学习系列博客已经正式发车了,相信大家以前学习一门编程语言的时候也有经验,咱们一般都是从环境开始,在此呢,大家也跟着老猫从最开始的搭建环境开始. GO语言的安装 首先呢,我们开始需要下载GO语 ...

  7. Python的程序结构[2] -> 类/Class[6] -> 内建类 map

    内建类map / Built-in Type map 关于内建类 map 是一个内建的类,能够返回一个 map 的 obj.map 的第一个参数为一个可执行函数,后续参数均为可迭代对象,map 会分别 ...

  8. go内建容器-切片

    1.基础定义 看到'切片'二字,满脸懵逼.切的啥?用的什么刀法切?得到的切片有什么特点?可以对切片进行什么操作? 先看怎么得到切片,也就是前两个问题.切片的底层是数组,所以切片切的是数组:切的时候采用 ...

  9. go内建容器-字符和字符串操作

    1.基础定义 在基础语法篇提到过golang的rune相当于其他编程语言的char,其本质是一个int32(四字节),用[]rune来转换一个字符串时,得到的是个解码后的结果,存储在新开辟的[]run ...

随机推荐

  1. 高效动画实现原理-Jetpack Compose 初探索

    一.简介 Jetpack Compose是Google推出的用于构建原生界面的新Android 工具包,它可简化并加快 Android上的界面开发.Jetpack Compose是一个声明式的UI框架 ...

  2. 【UE4】Windows 的几种打包方式

    简述 自动化工具(Unreal Automation Tool,简称 UAT) 自动化工具使用特定的命令 BuildCookRun 封装流程包含 构建(Build):该阶段将为所选择的平台编译可执行文 ...

  3. vue3.x非兼容的变更

    走马观花似的看看从vue2.x开始到vue3.x的一些非兼容性的变更,这样在将来升级过程中遇到那些奇奇怪怪的不能用的时候,就会很容易解决啦. 全局 API 全局 Vue API 已更改为使用应用程序实 ...

  4. vue3.x全局$toast、$message、$loading等js插件

    有时候我们需要使用一些类似toast,messge.loading这些跟js交互很频繁的插件,vue3.x这类插件的定义跟vue2.x插件稍大,而且相对变得复杂了一点点. 第一种.需要时创建,用完移除 ...

  5. SDIO总线协议

    SDIO采用HOST-DEVICE模式,所有通信都由HOST端发命令,DEVICE设备只要解析HOST命令就可与HOST进行通信. SDIO总线的几根线: 1.  CLK信号:HOST给DEVICE的 ...

  6. 必备的60个常用的Linux命令

    Linux必学的60个命令Linux提供了大量的命令,利用它可以有效地完成大量的工 作,如磁盘操作.文件存取.目录操作.进程管理.文件权限设定等.所以,在Linux系统上工作离不开使用系统提供的命令. ...

  7. 设计模式(1-2)-动态代理(newProxyInstance)

    上节设计模式(1-1)-代理模式,讲了代理模式的静态代理与动态代理的写法.本节,会从Proxy.newProxyInstance() 这个方法开始讲,上一节文末的那个class文件怎么一步步的来的. ...

  8. 超级好用的轻量级JSON处理命令jq

    1 简介 jq是一个轻量级的命令行工具,让你可以非常方便地处理JSON数据,如切分.过滤.映射.转化等,就像sed.awk.grep文本处理三剑客一样.jq是用C写的,没有运行时依赖,你可以直接下载可 ...

  9. FPGA基础之锁存器与触发器的设计

    转载:https://blog.csdn.net/lg2lh/article/details/39081061 一.锁存器 首先设计锁存器的时候应该清楚什么是锁存器,锁存器其实是对电平信号敏感的,一定 ...

  10. linux命令中find, which、whereis、locate,有什么区别?

    whatis 用于查询一个命令执行什么功能,并将查询结果打印到终端上 which 查看可执行文件的位置 whereis 查看文件的位置 man Linux提供了丰富的帮助手册,当你需要查看某个命令的参 ...