一、map数据类型

1.1 声明和定义

map类型是一个key-value的数据结构,又叫字典。(map也是可以扩容的,内部自动扩容)

声明:

var map1 map[keytype]valuetype

例子:

//var a map[key的类型]value类型
var a map[string]int
var b map[int]string
var c map[float32]string

注意:

1.声明是不会分配内存的,需要make初始化。

2.map必须初始化才能使用,否则panic。

3.map中声明value是什么类型,就存什么类型,混合类型自身不支持(Go是强类型语言),interfice支持存混合类型数据。

map类型的变量默认初始化为nil,所以需要使用make分配map内存然后才能使用,不然会panic(异常)。

实例如下:

实例1-1

package main

import (
"fmt"
) func main() {
var a map[string]int
if a == nil { //map未初始化,其初始默认为nil
fmt.Println("map is nil. Going to make one.")
a = make(map[string]int)
}
}

执行结果如下:

另外也验证一下不初始化跑出panic

实例如下:

实例1-2

package main

import (
"fmt"
) func main() {
var user map[string]int
user["abc"] =
fmt.Println(user)
}

执行结果如下:

已经抛出了panic,所以必须要初始化。

1.2 插入操作

代码示例如下:

实例1-3

package main

import (
"fmt"
) func main() {
a := make(map[string]int)
a["steve"] =
a["jamie"] =
a["mike"] =
fmt.Println("a map contents:", a)
}

执行结果如下:

1.3 初始化

第一种:make

实例1-4

package main

import (
"fmt"
) func main() {
var user map[string]int = make(map[string]int, ) //初始化时可以指定容量也可以不指定,指定的话可以提升性能
//user := make(map[string]int) //也可以写成这样,多种方式
user["abc"] =
fmt.Println(user)
}

执行结果如下:

第二种:声明时进行初始化(借助常量)

实例1-5

package main

import (
"fmt"
) func main() {
a := map[string]int{
"steve": ,
"jamie": ,
}
a["mike"] =
fmt.Println("a map contents:", a)
}

执行结果如下:

1.4 map扩容

map扩容实际上类似切片扩容就是:

map本来的容量是4,现在容量不够了,map内部自动扩容,比如说扩容到8,其在底层的机制就是将旧的内存地址中的4个元素拷贝到新到容量为8的内存地址中,然后再继续接收新元素并使用。

所以说:在初始化时,如果我们知道map大概有多少元素时,可以初始化时指定,这样可以在一定程度上提升性能(频繁扩容影响性能)

1.5如何访问map中元素?

通过key访问map中的元素

实例1-6

package main

import "fmt"

func main() {
a := map[string]int{
"steve": ,
"jamie": ,
}
a["mike"] =
b := "jamie"
fmt.Println("Salary of", b, "is", a[b])
}

执行结果如下:

1.6 如何判断map指定的key是否存在?

相当于做一个白名单,去判断map中指定key是否存在。

格式:value, ok := map[key]

注意:ok仅仅是个变量,可以随便命名

实例1-7    实例1:

package main

import (
"fmt"
) func main() {
a := map[string]int{
"steve": ,
"jamie": ,
}
a["mike"] =
b := "joe"
value, ok := a[b] //ok仅仅是个变量,可以随便命名
if ok == true {
fmt.Println("Salary of", b, "is", value)
} else {
fmt.Println(b, "not found")
}
}

执行结果如下:

实例1-8  实例2  (用户白名单基础版)

package main

import (
"fmt"
) var whiteUser map[int]bool = map[int]bool{
: true,
: true,
: true,
} func isWhiteUser(userId int) bool {
_, ok := whiteUser[userId] //这里不需要返回value,所以直接_忽略
return ok
} func main() {
userId :=
if isWhiteUser(userId) {
fmt.Printf("is white user:%v\n", userId)
} else {
fmt.Printf("is normal user:%v\n", userId)
}
}

执行结果如下:

1.7 map遍历操作

其实就是用for range

range返回key value并赋值给变量,结合数组、切片遍历也是for range,其实数组及切片就是个特殊map。

代码示例如下:

实例1-9

package main

import (
"fmt"
) func main() {
a := map[string]int{
"steve": ,
"jamie": ,
}
a["mike"] =
fmt.Println("All items of a map")
for key, value := range a {
fmt.Printf("personSalary[%s] = %d\n", key, value)
}
}

执行结果如下:

1.8 map删除元素

借助一个delete内置函数,delete指定map的key即可删除

代码示例如下:

实例1-10

package main

import (
"fmt"
) func main() {
a := map[string]int{
"steve": ,
"jamie": ,
}
a["mike"] =
fmt.Println("map before deletion", a)
delete(a, "steve")
fmt.Println("map after deletion", a)
}

执行结果如下:

1.9 map的长度

借助len函数

实例1-11

package main

import (
"fmt"
) func main() {
a := map[string]int{
"steve": ,
"jamie": ,
}
a["mike"] =
fmt.Println("length is", len(a))
}

执行结果如下:

1.10 map是引用类型

通过下面这个例子验证map是引用类型

实例1-12

package main

import (
"fmt"
) func main() {
a := map[string]int{
"steve": ,
"jamie": ,
}
a["mike"] =
fmt.Println("origin map", a)
b := a
b["mike"] =
fmt.Println("a map changed", a)
}

执行结果如下:

解释:

可以发现a为map,将a赋值给b,b也是map,map b做了修改,此时打印a,map a也发生了变化,这证明map是引用类型

1.11 map排序

默认情况下, map并不是按照key有序进行遍历的。

千万不要依赖map中的顺序,map是无序的(go1.6之前好像是有序的)

可见如下实例:

实例1-13

package main

import (
"fmt"
) func main() {
var a map[string]int = make(map[string]int, )
for i := ; i < ; i++ {
key := fmt.Sprintf("key%d", i)
a[key] = i
}
for key, value := range a {
fmt.Printf("key:%s = %d\n", key, value)
}
}

执行结果如下:

但是如何map如何按照key进行排序

思路:先把key排序,然后再按照排序后的key去遍历map

代码示例如下:

实例1-14

package main

import (
"fmt"
"sort"
) func main() {
var a map[string]int = make(map[string]int, )
for i := ; i < ; i++ {
key := fmt.Sprintf("key%d", i)
a[key] = i
}
var keys []string //将无序的map的key放入到切片中
for key, _ := range a {
keys = append(keys, key)
}
sort.Strings(keys) //借助sort进行排序
for _, key := range keys { //将切片进行遍历
fmt.Printf("key:%s=%d\n", key, a[key])
}
}

执行结果如下:

1.12 map类型切片

首先需要先将切片初始化,切片中的每一个元素又是一个map,所以要使用map时,又需要对每一个元素进行一次初始化(引用类型)

实例1-15    实例1

package main

import (
"fmt"
) func mapSlince() { } func main() {
var s []map[string]int //s是一个map类型的切片
s = make([]map[string]int, ) //切片初始化 for k, v := range s {
fmt.Printf("index:%d val:%v\n", k, v)
} s[] = make(map[string]int, ) //map初始化
s[]["abc"] = for key, val := range s[] {
fmt.Printf("key:%s value:%v\n", key, val)
}
}

执行结果如下:

实例1-16  实例2

package main

import (
"fmt"
) func main() {
var mapSlice []map[string]int
mapSlice = make([]map[string]int, ) //此时是一个未初始化的map类型切片,所以需要先将切片初始化
fmt.Println("before map init")
for index, value := range mapSlice {
fmt.Printf("index:%d value:%v\n", index, value)
}
fmt.Println()
mapSlice[] = make(map[string]int, ) //要使用map,需要将map进行初始化。
mapSlice[]["a"] =
mapSlice[]["b"] =
mapSlice[]["c"] =
mapSlice[]["d"] =
mapSlice[]["e"] =
fmt.Println("after map init")
for index, value := range mapSlice {
fmt.Printf("index:%d value:%v\n", index, value)
}
}

执行结果如下:

Go语言基础之6--map(字典)数据类型的更多相关文章

  1. 语言基础:C#输入输出与数据类型及其转换

    今天学习了C#的定义及特点,Visual Studio.Net的集成开发环境和C#语言基础. C#语言基础资料——输入输出与数据类型及其转换 函数的四要素:名称,输入,输出,加工 输出 Console ...

  2. 01 C语言程序设计--01 C语言基础--第3章 基本数据类型01

    01.1.3.1序言 00:02:17 01.1.3.2 C语言中的基本元素和常量的概念 00:08:54 01.1.3.3示例--常量 00:12:08 01.1.3.4变量的概念和命名规则 00: ...

  3. 第二章(java程序设计)第三章(语言基础)

    第二章 2.1 对象 对象的概念是由现实世界引入问题模型: 对象包含有:状态和行为.具体地来说是: 数据封装:对象的方法的作用就是:将内部变量封装起来,提供给外界交互的窗口.(实现对数据的隐藏) 继承 ...

  4. NDK以及C语言基础语法(一)

    一.什么是NDK? Native Development Kit (本地开发工具包): NDK中提供了一系列的工具,帮助我们快速开发C/C++的动态库,并能自动将so文件和java文件一起打包成apk ...

  5. Go语言基础之数据类型

    Go语言基础之数据类型 Go语言中有丰富的数据类型,除了基本的整型.浮点型.布尔型.字符串外,还有数组.切片.结构体.函数.map.通道(channel)等.Go 语言的基本类型和其他语言大同小异. ...

  6. JavaScript 引入方式 语言规范 语言基础 数据类型 常用方法 数组 if_else 比较运算符 for while 函数 函数的全局变量和局部变量 {Javascript学习}

    Javascript学习 JavaScript概述 ECMAScript和JavaScript的关系 1996年11月,JavaScript的创造者--Netscape公司,决定将JavaScript ...

  7. Python语言基础-语法特点、保留字与标识符、变量、基本数据类型、运算符、基本输入输出、Python2.X与Python3.X区别

    Python语言基础 1.Python语法特点 注释: 单行注释:# #注释单行注释分为两种情况,例:第一种#用于计算bim数值bim=weight/(height*height)第二种:bim=we ...

  8. Go语言基础之map

    Go语言基础之map Go语言中提供的映射关系容器为map,其内部使用散列表(hash)实现. map map是一种无序的基于key-value的数据结构,Go语言中的map是引用类型,必须初始化才能 ...

  9. C#-语言基础+数据类型+运算符

    一.C#语言基础 新建项目:文件→新建→项目→Visual C#(默认.NET Framework 4.5)→控制台应用程序 1.项目结构 (1)项目后缀 .config ——配置文件(存放配置参数文 ...

随机推荐

  1. C++面向对象类的实例题目六

    问题描述: 编写一个程序计算两个给定长方形的面积,其中在设计类成员函数addarea()(用于计算两个长方形的总面积)时使用对象作为参数. 程序代码: #include<iostream> ...

  2. ubuntu下使用PIL中的show函数,无法显示图片的问题

    问题描述:ubuntu14.04系统,python2.7(version),正在学习python中, from PIL import Image im = Image.open('1.jpg') im ...

  3. 算法Sedgewick第四版-第1章基础-023-MultiwordSearch.java

    Multi-word search. Program MultiwordSearch.java reads a sequence of query words q[1], ..., q[k] from ...

  4. 100211D Police Cities

    传送门 分析 看到这个题我们的第一反应自然是Tarjan缩点,在这之后我们可以发现实际只要在缩点之后所有出度或入度为0的点布置警察局就可以达到要求,我们用dpij表示考虑前i个出度或入度为0的点共布置 ...

  5. String的字符串相加是怎么实现的?

    http://bbs.csdn.net/topics/60485130 首先看 String str = "a" + "b" + "c"; ...

  6. 缓存淘汰算法之LRU

    1. LRU1.1. 原理 LRU(Least recently used,最近最少使用)算法根据数据的历史访问记录来进行淘汰数据,其核心思想是“如果数据最近被访问过,那么将来被访问的几率也更高”. ...

  7. matlab任务:FCM分类

    一个朋友让帮忙做图像分类,用FCM聚类算法,网上查了一下,FCM基本都是对一幅图像进行像素的分类,跟他说的任务不太一样,所要做的是将一个文件夹里的一千多幅图像进行分类.图像大概是这个样子的(是25*2 ...

  8. jquery('tr','div')和jquery('tr,div')

     jQuery('tr', 'div') 等价于 $('tr', 'div')    表示div里面寻找tr    jQuery('tr, div') <=> $('tr, div') 表 ...

  9. Redhat 6 git服务器配置 (git-daemon)

    git-daemon是按照git的自己的git协议进行访问git服务   1.git-daemon软件安装 软件仓库见 redhat 6 git 服务器 配置 (http)   2.配置git dae ...

  10. 线程池ThreadPool实现异步多线程

    ThreadPool线程池的主要方法: 1. public static Boolean QueueUserWorkItem(WaitCallback wc, Object state); WaitC ...