zoukankan      html  css  js  c++  java
  • go中值传递、引用传递、指针传递的区别

    go语言中的值类型:

      int、float、bool、array、sturct等

      值传递是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数

      声明一个值类型变量时,编译器会在栈中分配一个空间,空间里存储的就是该变量的值    

    go语言中的引用类型:

      slice,map,channel,interface,func,string等

      声明一个引用类型的变量,编译器会把实例的内存分配在堆上

      string和其他语言一样,是引用类型,string的底层实现struct String { byte* str; intgo len; }; 但是因为string不允许修改,每次操作string只能生成新的对象,所以在看起来使用时像值类型。

      所谓引用传递是指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

      需要注意的是:引用类型在函数的内部可以对它的值进行修改,但是如果给形参重新赋值,重新赋值后的形参再怎么修改都不会影响外面的实参了

      nil可以赋值给引用类型(除string外)、error类型和指针类型

     

    go语言中的指针类型:

      一个指针变量指向了一个值的内存地址

        当一个指针被定义后没有分配到任何变量时,它的值为 nil。nil 指针也称为空指针

      一个指针变量通常缩写为 ptr

      其实引用类型可以看作对指针的封装

    空指针判断:

    if(ptr != nil)     /* ptr 不是空指针 */
    if(ptr == nil)    /* ptr 是空指针 */

    例子:

    package main
    
    import "fmt"
    
    type pArr []int
    
    func (p *pArr) oopFunc() {
        (*p)[3] = 111
    }
    
    type newFunc func(string)
    
    func strFunc(s string) {
        fmt.Printf("%T ", s[1]) //uint8
        fmt.Println(s)          //hello world
    }
    
    type Person struct {
        Name   string
        Height int
    }
    
    func changeStruct(p Person) {
        p.Name = "李四"
        p.Height = 160
    }
    
    func main() {
    
        //切片指针的使用
        var ptr []*int
        i := 1
        ptr = append(ptr, &i)
        fmt.Println("ptr:", *ptr[0])
    
        //结构体是值类型
        person := Person{
            "张三",
            180,
        }
        changeStruct(person)
        fmt.Println(person) //{张三 180}
    
        //func 可以作为参数进行传递
        var nf newFunc
        nf = strFunc
        str := "hello world"
        nf(str) //uint8 hello world
    
        //类似面向对象的方法
        p := make(pArr, 4)
        p.oopFunc()
        fmt.Println("p:", p) //p: [0 0 0 111]
    
        //值类型无法被改变
        num := 1
        valPass(num)
        fmt.Println("num:", num) //num: 1
    
        //引用类型可以在函数中被改变
        nums := []int{0, 1, 2, 3}
        RefPass(nums)
        fmt.Println("nums:", nums) //nums: [0 100 2 3]
    
        //形参可以改变引用类型的值,但不能够形参重新赋值
        RefPass2(nums)
        fmt.Println("nums:", nums) //nums: [0 100 2 3]
    
        //形参可以改变指针类型的值
        n := new(int)
        *n = 111
        PointPass(n)
        fmt.Println("n:", *n) //n: 12
    
        //形参可以改变指针类型的值,但是对形参重新赋值,不会影响实参
        PointPass2(n)
        fmt.Println("n:", *n) //n: 12
    
    }
    
    //指针传递,对指针重新赋值,指针指向了新的地址,此时对形参做修改将不再影响外面的实参
    func PointPass2(num *int) {
        num = new(int)
        *num = 13
    }
    
    // 指针传递,普通用法
    func PointPass(num *int) {
        *num = 12
    }
    
    //引用传递,普通用法,这个会改变外面的实参
    func RefPass(nums []int) {
        nums[1] = 100
    }
    
    //引用传递,对形参重新赋值,不会改变外面的实参,形参指向了新的地址
    func RefPass2(nums []int) {
        nums = []int{9, 8, 7, 6}
    }
    
    //值传递
    func valPass(num int) {
        num = 5
    }
  • 相关阅读:
    ASP.NET Core: What I learned!
    Entity Framework Core with GraphQL and SQL Server using HotChocolate
    Angular 9 Chart.js with NG2-Charts Demo
    POST调用WCF方法-项目实践
    项目实战-登录速度优化笔记
    MP4视频流base64数据转成Blob对象
    使用Vue+ElementUI实现前端分页
    JS端实现图片、视频时直接下载而不是打开预览
    Dynamic CRM工作流流程实战
    Dynamic CRM插件调试与单元测试
  • 原文地址:https://www.cnblogs.com/xbblogs/p/11102970.html
Copyright © 2011-2022 走看看