在“range”语句中生成的数据的值其实是集合元素的拷贝。它们不是原有元素的引用。
这就意味着更新这些值将不会修改原来的数据。我们来直接看段示例:

package main

import "fmt"

func main() {
data := []int{, , }
for _, v := range data {
v *= //原始元素未更改
}
fmt.Println("data:", data) //输出 data: [1 2 3]
}

如果我们需要更新原有集合中的数据,使用索引操作符来获得数据即可:

package main

import "fmt"

func main() {
data := []int{, , }
for i, _ := range data {
data[i] *=
}
fmt.Println("data:", data) //输出 data: [10 20 30]
}

好,重点来了!重点来了!重点来了!,重要的话说三遍,大部分博友们可能会踩坑.

这里我提前总结下:

多个slice可以引用同一个数据。比如,当你从一个已有的slice创建一个新的slice时(比如通过索引截取),这就会发生。

如果你的应用功能需要这种行为,那么你将需要留意下slice的"坑"。

在某些情况下,在一个slice中添加新的数据,在原有数组无法保持更多新的数据时,将导致分配一个新的数组。

而其他的slice还指向老的数组(或者是老的数据)

package main

import "fmt"

func main() {
s1 := []int{, , }
fmt.Println(len(s1), cap(s1), s1) //输出 3 3 [1 2 3]
s2 := s1[:] //索引从第二个元素截取开始
fmt.Println(len(s2), cap(s2), s2) //输出 2 2 [2 3]
for i := range s2 {
s2[i] +=
}
//仍然引用同一数组
fmt.Println(s1) //s1 在s2修改了后面2个元素,所以s1也是更新了。输出 [1 22 23]
fmt.Println(s2) //输出 [22 23]
s2 = append(s2, ) // 注意s2的容量是2,追加新元素后将导致分配一个新的数组 [22 23 4]
for i := range s2 {
s2[i] +=
}
//s1 仍然是更新后的历史老数据
fmt.Println(s1) //输出 [1 22 23]
fmt.Println(s2) //输出 [32 33 14]
}

所以,大家在使用中特别注意。容量不足,追加新元素不影响历史数据。因为重新分配了变量了。

另外,继续聊下高级一点滴技巧:

使用指针接收方法的值

只要值是可取址的,那在这个值上调用指针接收方法是没问题的。

然而并不是所有的变量是可取址的。Map的元素就不是。通过interface引用的变量也不是。我们接着看下面一段代码:

package main

import "fmt"

type user struct {
name string
} func (p *user) print() {
fmt.Println("排名:", p.name)
} type printer interface {
print()
} func main() {
u := user{"乔峰"}
u.print() // 输出 排名: 乔峰
var in printer = user{"鸠摩智"} //error
in.print()
m := map[string]user{"one": user{"风清扬"}}
m["one"].print() //error
}

输出:

cannot use user literal (type user) as type printer in assignment:
user does not implement printer (print method has pointer receiver)
cannot call pointer method on m["one"]
cannot take the address of m["one"]

大致意思是:不能在赋值中使用数据文本(类型数据)作为类型指针,user未执行指针调用(指针方法具有指针接收器),

无法对m[“one”]调用指针方法,不能取m的地址[“one”]。

上面我们看到有一个struct值的map,我们无法更新单个的struct值。比如错误的代码:

package main

type user struct {
name string
} func main() {
m := map[string]user{"one": {"乔峰"}}
m["one"].name = "风清扬" //输出 cannot assign to struct field m["one"].name in map
}

错误意思是:在map中,无法分配给结构字段m["one"].name。这个操作无效是因为map元素是无法取址的。

上面我们提到:slice元素是可以取地址滴:

package main

import "fmt"

type user struct {
name string
} func main() {
one := user{"乔峰"}
u := []user{one}
u[].name = "风清扬" //ok
fmt.Println(u) //输出: [{风清扬}]
}

当然我们还有更好的解决办法:

第一个有效的方法是使用一个临时变量:

package main

import "fmt"

type user struct {
name string
} func main() {
m := map[string]user{"one": {"乔峰"}}
u := m["one"] //使用临时变量
u.name = "风清扬"
m["one"] = u
fmt.Printf("%v\n", m) //输出: map[one:{风清扬}]
}

另一个有效的方法是使用指针的map:

package main

import "fmt"

type user struct {
name string
} func main() {
m := map[string]*user{"one": {"乔峰"}}
m["one"].name = "风清扬" //ok
fmt.Println(m["one"]) //输出: &{风清扬}
}

说到这里,顺便再提一下。继续看下面一段代码:

package main

import "fmt"

type user struct {
name string
} func main() {
m := map[string]*user{"one": {"乔峰"}}
m["two"].name = "鸠摩智" //新增自定义键名值
fmt.Println(m["two"]) //error
}

输出:

panic: runtime error: invalid memory address or nil pointer dereference

无效的内存地址或取消引用空指针?原因在于Go无法动态给结构体添加字段,我们可以间接使用make(map[string]interface{})实现。

好吧,就说这么多了,有不足之处欢迎广大博友留言指正。。。。。。。

Go 修改map slice array元素值的更多相关文章

  1. Numpy修改数组中的元素值

    import numpy as np x = np.arange(8) # [0 1 2 3 4 5 6 7] # 在数组尾部追加一个元素 np.append(x,10) # array([ 0, 1 ...

  2. javascript 常见数组操作( 1、数组整体元素修改 2、 数组筛选 3、jquery 元素转数组 4、获取两个数组中相同部分或者不同部分 5、数组去重并倒序排序 6、数组排序 7、数组截取slice 8、数组插入、删除splice(需明确位置) 9、数组遍历 10、jQuery根据元素值删除数组元素的方)

    主要内容: 1.数组整体元素修改 2. 数组筛选 3.jquery 元素转数组 4.获取两个数组中相同部分或者不同部分 5.数组去重并倒序排序 6.数组排序 7.数组截取slice 8.数组插入.删除 ...

  3. Map java中的map 如何修改Map中的对应元素

    Map java中的map 如何修改Map中的对应元素 Map以按键/数值对的形式存储数据,和数组非常相似,在数组中存在的索引,它们本身也是对象.         Map的接口         Map ...

  4. Map四种获取key和value值的方法,以及对map中的元素排序(转)

    获取map的值主要有四种方法,这四种方法又分为两类,一类是调用map.keySet()方法来获取key和value的值,另一类则是通过map.entrySet()方法来取值,两者的区别在于,前者主要是 ...

  5. C++:map用法及元素的默认值

    C++:map用法 一.map基本用法 键值对 第一个参数为键的类型,第二个参数为值的类型. 源代码 #include <iostream> #include <string> ...

  6. slice,Array.prototype.slice,Array.protyotype.slice.call

    slice 特点:基于当前数组中的一或多个项创建一个新数组.[原数组不会被修改] 返回结果:返回一个从开始到结束(不包括结束)选择的数组的一部分浅拷贝到一个新数组对象. 语法: arr.slice() ...

  7. 【转】 golang slice array

    1. array   同一类型数据的集合     var arr [n]type    //声明type类型一维数组     var arr [m][n]type //声明type类型二维数组     ...

  8. list、set、map以及array的区别

    对于刚刚学习集合框架来说,如何选择list.set.map以及array是比较模糊的 在此我将对这四种情况做总结: array:数组,可以存储对象和基本数据类型,长度固定. Collection:集合 ...

  9. 关于C++ STL标准库中map 的多元素应用

    map的特性是,所有的元素会根据键值自动排序.map的所有元素都是pair,同时拥有实值(value)和键值(key).pair的第一个元素被视为键值,第二个被视为实质piar 的定义 templat ...

随机推荐

  1. 力扣(LeetCode)删除排序链表中的重复元素II 个人题解

    给定一个排序链表,删除所有含有重复数字的节点,只保留原始链表中 没有重复出现 的数字. 思路和上一题类似(参考 力扣(LeetCode)删除排序链表中的重复元素 个人题解)) 只不过这里需要用到一个前 ...

  2. (三十五)golang--面向对象之多态

    多态:变量具有多种形态,可以用统一的接口来调用不同的实现. 接口体现多态特征: (1)多态参数:之前所讲的Usb接口案例,既可以接受手机变量,也可以接受相机变量,就体现了usb接口的多态: (2)多台 ...

  3. [Windows篇] 在windows 10上源码编译gtest 并编写CMakeLists.txt

    本文首发于个人博客https://kezunlin.me/post/aca50ff8/,欢迎阅读! compile gtest on windows 10 Guide compile gtest on ...

  4. Linux虚拟机扩容根分区CentOs6.9 VMware14

    1.首先关闭虚拟机点击编辑虚拟机设置 2.点击想要扩容的硬盘点击扩容 3.增加容量 输入想增加的容量,因为我本身是30G写到35G是加了5G不是增加30G.(此处为了演示只增加5G) 4.开启虚拟机 ...

  5. C#学习笔记03--循环和一维数组

    一.循环(重点) 什么时候用循环? 想让一段代码执行多次, 这段代码可能不一样但是一定有一个规律. 1.while 循环 格式:  while(循环条件) { 循环执行的代码; } 循环的机制:  当 ...

  6. [ASP.NET Core 3框架揭秘] 异步线程无法使用IServiceProvider?

    标题反映的是上周五一个同事咨询我的问题,我觉得这是一个很好的问题.这个问题有助于我们深入理解依赖注入框架在ASP.NET Core中的应用,以及服务实例的生命周期. 一.问题重现 我们通过一个简单的实 ...

  7. linux磁盘分区三步走

    为了便于理解硬盘的物理结构 ,可将硬盘看作一个圆,它是坚硬金属材料制成的涂以磁性介质的盘片,不同容量硬盘的盘片数不等.每个盘有两面,都可记录信息.要了解硬盘的物理结构,需要弄懂磁道.扇区.柱面.簇等几 ...

  8. EntityFramework Core 3多次Include导致查询性能低之解决方案

    前言 上述我们简单讲解了几个小问题,这节我们再来看看如标题EF Core中多次Include导致出现性能的问题,废话少说,直接开门见山. EntityFramework Core 3多次Include ...

  9. 【JavaEE】之MyBatis开发DAO

    在SSM框架中的DAO层就是MyBatis中的Mapper,Mapper分为两部分:Mapper接口(JAVA文件)和Mapper映射文件(XML文件).DAO开发(Mapper开发)有两种方式:原始 ...

  10. CSS 了解一下

    CSS 认识一下 1.CSS 的那些事 CSS(Cascading Style Sheets)译「层叠样式表」,我的理解是:各种样式叠加的表. 一个网页,如果没有 CSS,就是穿着"国王的新 ...