值拷贝导致使用container/list出现的诡异问题分析

栏目: 数据库 · 发布时间: 6年前

内容简介:输出输出为什么

golang值拷贝导致使用container/list出现的诡异问题分析

先看正确使用list的两种方式

  • 使用list.New()

    package main
    
    import (
    	"container/list"
    	"fmt"
    )
    
    func main() {
    	lPtr := list.New()
    	lPtr.PushBack(1)
    	for front := lPtr.Front(); front != nil; front = front.Next() {
    		fmt.Println(front.Value)
    	}
    }

    输出

  • 使用list.List{}

package main

import (
   "container/list"
   "fmt"
)

func main() {
   l := list.List{}
   l.PushBack(1)
   for front := l.Front(); front != nil; front = front.Next() {
      fmt.Println(front.Value)
   }
}

输出

list的错误使用方式

package main

import (
   "container/list"
   "fmt"
)

func main() {
   l := *list.New()
   l.PushBack(1)
   for front := l.Front(); front != nil; front = front.Next() {
      fmt.Println(front.Value)
   }
}

输出

<nil>

错误原因分析

为什么 *list.New() 方式给 l 后,放进 l 中的元素打印不出来了呢?

让我们来看看l.Front()中的逻辑, 如果 len 不为 0 ,返回的是 l.root.next

func (l *List) Front() *Element {
	if l.len == 0 {
		return nil
	}
	return l.root.next
}

上面打印list的循环条件是 front != nil ,打印不出元素,也就说明 l.root.next==nil

那我们把 l.root.next 的地址打印出来看看, 当 len!=0 时才会返回 l.root.next ,我们要先添加一个元素

package main

import (
   "container/list"
   "fmt"
)

func main() {
   lPtr := list.New()
   lPtr.PushBack("ptr1")
   fmt.Printf("lPtr=%p  lPtr.root.next=%p\n", lPtr, lPtr.Front())
   l := *lPtr
   l.PushBack(1)
   fmt.Printf("&l=%p  l.root.next=%p\n", &l, l.Front())
   for front := l.Front(); front != nil; front = front.Next() {
      fmt.Println(front.Value)
   }
}

输出

lPtr=0xc000084150  lPtr.root.next=0xc000084180
&l=0xc0000841b0  l.root.next=0xc000084180
ptr1
1
<nil>

我们看到 l.root.next 的值 lPtr.root.next 的值都是 0xc000084180 , 问题就出在这里

lPtr 添加元素 "ptr1" 后, lPtr 内部链表如下

lPtr.root.prev->Element{"ptr1"}->&lPtr.root  //Element{"ptr1"}地址:0xc000084180
lPtr.root.next->Element{"ptr1"}->&lPtr.root

l := *lPtr 时,内部属性的值拷贝如下:

l.root.prev->Element{"ptr1"}->&lPtr.root  //Element{"ptr1"}地址:0xc000084180
l.root.next->Element{"ptr1"}->&lPtr.root

l 添加元素 "1" 后, l 内部链表如下

l.root.prev->Element{"1"}->Element{"ptr1"}->&lPtr.root
l.root.next->Element{"ptr1"}->Element{"1"}->&lPtr.root

至此就是弄清楚了上面发生怪异输出结果的原因了。

注意:在 go 中尽量避免对复杂对象做简单的值传递赋值。


以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

数据化管理

数据化管理

黄成明 (@数据化管理) / 电子工业出版社 / 2014-7 / 59.90元

《数据化管理:洞悉零售及电子商务运营》讲述了两个年轻人在大公司销售、商品、电商、数据等部门工作的故事,通过大量案例深入浅出地讲解了数据意识和零售思维。作者将各种数据分析方法融入到具体的业务场景中,最终形成数据化管理模型,从而帮助企业提高运营管理能力。 《数据化管理:洞悉零售及电子商务运营》全部案例均基于Excel,每个人都能快速上手应用并落地。一起来看看 《数据化管理》 这本书的介绍吧!

RGB转16进制工具
RGB转16进制工具

RGB HEX 互转工具

HTML 编码/解码
HTML 编码/解码

HTML 编码/解码

HSV CMYK 转换工具
HSV CMYK 转换工具

HSV CMYK互换工具