• Golang


    先看指针类型的receiver:

    package main
    import (
        "fmt"
    )
    type BasicEvent struct {
        EventId int
    }
    func (ev *BasicEvent) updateEventID(id int) {
        ev.EventId = id
    }
    func main() {
        ev1 := &BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev1.EventId)
        ev1.updateEventID(2)
        fmt.Printf("After update id = %d
    ", ev1.EventId)
        ev2 := BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev2.EventId)
        ev2.updateEventID(2)
        fmt.Printf("After update id = %d
    ", ev2.EventId)
    }
    

    输出的结果是:

    before update id = 1
    After update id = 2
    before update id = 1
    After update id = 2
    

    由此可见,不管调用的时候是个值还是指针,只要是方法的receiver是指针类型,都能够修改调用者的内部状态。我估计这个时候是Go自动做了指针转换。

    值类型的receiver:

    package main
    import (
        "fmt"
    )
    type BasicEvent struct {
        EventId int
    }
    func (ev BasicEvent) updateEventID(id int) {
        ev.EventId = id
    }
    func main() {
        ev1 := &BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev1.EventId)
        ev1.updateEventID(2)
        fmt.Printf("After update id = %d
    ", ev1.EventId)
        ev2 := BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev2.EventId)
        ev2.updateEventID(2)
        fmt.Printf("After update id = %d
    ", ev2.EventId)
    }
    

    输出的结果是:

    before update id = 1
    After update id = 1
    before update id = 1
    After update id = 1
    

    由此可见,不管调用的时候是个值还是指针,只要是方法的receiver是值类型,都无法改变调用者的内部状态。
    总是可见,主要关注的点是方法定义的时候到底receiver是指针还是值类型,如果receiver是指针,内部状态才可以改。
    另外,关于关于传入参数是否可改的问题

    package main
    import (
        "fmt"
    )
    type BasicEvent struct {
        EventId int
    }
    func updateEventIDByPointer(ev *BasicEvent, newID int) {
        ev.EventId = newID
    }
    func updateEventIDByValue(ev BasicEvent, newID int) {
        ev.EventId = newID
    }
    func main() {
        ev1 := &BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev1.EventId)
        updateEventIDByPointer(ev1, 2)
        fmt.Printf("After update id = %d
    ", ev1.EventId)
        ev2 := BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev2.EventId)
        updateEventIDByValue(ev2, 2)
        fmt.Printf("After update id = %d
    ", ev2.EventId)
        ev3 := &BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev3.EventId)
        updateEventIDByValue(*ev3, 2)
        fmt.Printf("After update id = %d
    ", ev3.EventId)
        ev4 := BasicEvent{EventId: 1}
        fmt.Printf("before update id = %d
    ", ev4.EventId)
        updateEventIDByPointer(&ev4, 2)
        fmt.Printf("After update id = %d
    ", ev4.EventId)
    }
    

    结果如下:

    before update id = 1
    After update id = 2
    before update id = 1
    After update id = 1
    before update id = 1
    After update id = 1
    before update id = 1
    After update id = 2
    

    前两个比较好理解。第三个值得注意的是,对指针解引用会创建另一个值对象,所以无法改变内部状态。第四个也好说,取地址后得到了原始值对象的引用,所以可以修改内部状态。

  • 相关阅读:
    【Java集合】JDK1.7和1.8 HashMap有什么区别
    【Java集合】为什么HashMap的长度是2的N次幂?
    【VritualEnv】虚拟环境的介绍和基本使用
    【分布式事务】分布式事务解决方案
    【JVM】JVM中的垃圾收集器
    jQuery事件触发前后进行其他的操作
    在jQuery中使用自定义属性
    使用其他服务器引入JS文件
    引入其他服务的JS、和 本地的JS文件,script的属性
    trigger 和 triggerHandler(),自定义事件
  • 原文地址:https://www.cnblogs.com/pzblog/p/9088298.html
Copyright © 2020-2023  润新知