GO指针详解

因为大一的时候c语言没学好,所以看到指针很心烦 ,后来速成了一遍go ,每每写道指针部分就开始遗忘 ,所以专门对指针部分做了此笔记

概念

在 Go 语言中,指针是一种变量类型,它存储的是另一个变量的内存地址。通过指针,你可以访问和修改它指向的内存位置上存储的数据。这和 Java 中的引用类型变量(如对象的引用)有相似之处,因为它们都允许通过一个“引用”或“指针”来操作实际的数据。在go中指针就是通过可以通过操作地址的方式实现引用传递的效果

指针是一个变量 和int int32 float32 uint64一样都是变量 只是该变量存储的是数据是地址

代码观察

/**
指针是变量 地址是变量的内存地址 是数据
指针可以理解为一个变量类型,可以指向其他变量的内存地址,通过指针可以修改其他变量的值。
如果传递一个非引用对象(java中的对象 java的对象名其实就是类似于指针的概念) 如果传递的是数据的话那么方法修改了 也不是会影响到原来的数据的
如果传递的是引用对象(java中的对象 java的对象名其实就是类似于指针的概念) 那么方法修改了 也会影响到原来的数据的
*/

func pointdemo() {

	a, b := 1, 2.34
	fmt.Println("输出变量a和b的值:")
	fmt.Println("a=", a, "b=", b)
	//输出:输出变量a和b的值:a= 1 b= 2.34
	fmt.Println("输出对应的地址:")

	fmt.Printf("a的地址:%p\n", &a)
	fmt.Printf("b的地址:%p\n", &b)
	//输出:a的地址:0xc00000a0d8
//b的地址:0xc00000a0f0
	// 创建指针
	var pA *int = &a
	pB := &b
	//是一样的 所以可以发现 指针的地址是一样的

/* 所以大一我就认为地址和指针是一个东西 导致后面接口 还有方法的时候会出很多问题
a的地址:0xc00000a0d8
b的地址:0xc00000a0f0
*/
	fmt.Printf("a的地址:%p\n", pA)
	fmt.Printf("b的地址:%p\n", pB)
//由于指针也是存储地址的变量 那么作为变量肯定也有自己的地址
	fmt.Printf("指针在取地址a的地址:%p\n", &pA)
	fmt.Printf("指针b的地址:%p\n", &pB)
	/*
	
	指针在取地址a的地址:0xc000060028
	指针b的地址:0xc000060030
	
	*/
	// 通过指针输出数据
	fmt.Println("通过指针输出a的值:", *pA)
	fmt.Println("通过指针输出b的值:", *pB)

	// 修改指针指向的数据
	*pA = 3
	*pB = 4.56
	fmt.Println("修改后的a和b的值:")
	//使用指针获取数值  对指针获取数据
	fmt.Println("指针方式a=", *pA, "b=", *pB)
	//使用地址获取数值 对地址获取数据
	fmt.Println("地址方式取数=", *(&a), "b=", *(&b))
/*
	这里就验证了指针可以通过*指针获取数据并且改变
	地址也可也可以通过这个符号改变 
*/

	*(&a) = 5
	*pB = 6.78

	fmt.Println("指针改变数值后a和b的值=", a, "b=", b)
}

错误笔记

上面讲了和java的类似 那么这里记录我的一个小错误

	fmt.Println("请输入一个字符串:")
	var input string
	scanln, err := fmt.Scanln(&input)
		fmt.Println(input)

既然可以通过传递地址然后输入的数值赋予变量 那么我传递指针也是一样的吧?

	var name *string

	fmt.Println("请输入一个字符串:")
	fmt.Scanln(name)
	fmt.Println("输入的数据:", *name)
	fmt.Println("输入的读取的单位数 这里是行为:", scanln)

运行
在这里插入图片描述
无效的地址指针
这里我就突然忘记了
指针既然是变量 那么是需要初始化的 不初始化只是有一个地址 地址没有指向任何一个空间 所以无法把输入的值传递给他 所以指针线初始化指向一个变量的地址 这里scan赋值过程就变成了 用户输入值->指针地址->找到指针装载的值(地址)->通过地址找到这个变量的内存空间进行修改

demo := "初始化"
   var name *string = &demo

   fmt.Println("请输入一个字符串:")
   fmt.Scanln(name)
   fmt.Println("输入的数据:", *name)

指针分类

二重

在go中,数组算是基本数据类型 ,即参数传递的过程中,传递的是引用,函数修改参数,原来的数据发生改变,而go中并不会这样,所以需要数组指针来完成这个操作

func main() {
	var a int = 10
	var b *int = &a
	*b = 20
	fmt.Println("a = ", a) // Output: a = 20

	fmt.Printf("value of a  %p\n ", b)    // Output: a = int
	fmt.Printf("type of b  %T\n ", b)     // Output: b = 0xc000014080
	fmt.Printf("address of b  %p\n ", &b) // Output: b = 0xc000014080
	c := 12
	d := 20
	e := &c
	f := &d

	add(&c, &d)
	fmt.Printf("c =%d\n d=%d\n", c, d) // Output: c = 32	d=20  传递数值地址 改变数值地址也会改变
	swap(e, f)                   
	//并没有交换
	fmt.Printf("c =%d\n d=%d\n", *e, *f) // Output: c = 32	d=20  传递数值地址 改变数值地址也会改变

}

// 交换地址
func swap(i *int, i2 *int) {
	i, i2 = i2, i
}

/*
*
指针作为参数
*/
func add(a, b *int) {
	*a = *a + *b
}

swap 这里测试通过函数修改参数,影响原数据,一般数据可以通过传递地址修改来影响原来数据,指针也是一样的
这里和函数哪里交换函数体是一样的,传递数据地址可以改变原数据的内容,那么这里想要哦通过改变指针存放的地址,来影响原来的数据,那么就要传递的数据,那么就传递这个指针的地址既即可完成交换(二重指针)

func main() {
	var a int = 10
	var b *int = &a
	*b = 20
	fmt.Println("a = ", a) // Output: a = 20

	fmt.Printf("value of a  %p\n ", b)    // Output: a = int
	fmt.Printf("type of b  %T\n ", b)     // Output: b = 0xc000014080
	fmt.Printf("address of b  %p\n ", &b) // Output: b = 0xc000014080
	c := 12
	d := 20
	e := &c
	f := &d

	add(&c, &d)
	fmt.Printf("c =%d\n d=%d\n", c, d) // Output: c = 32	d=20  传递数值地址 改变数值地址也会改变
	swap(&e, &f)                       //传递指针的1地址 完成交换
	//完成交换
	fmt.Printf("c =%d\n d=%d\n", *e, *f) // Output: c = 20	d=32  传递数值地址 改变数值地址也会改变

}

//接收指针的地址  
func swap(i **int, i2 **int) {
//改变指针地址存放的内容
	*i, *i2 = *i2, *i//地址取值 就是一种指针(一重地址的空间) 数值是书数据地址 
}

/*
*
指针作为参数
*/
func add(a, b *int) {
	*a = *a + *b
}

数组指针

由于go中的数组是基本类型 所以智能通过指针实现修改参数 改变原来的数值

func main() {

	//数组指针
	var arr [5]int
	var ptr *[5]int
	ptr = &arr
	(*ptr)[0] = 10
	(*ptr)[1] = 20
	(*ptr)[2] = 30
	(*ptr)[3] = 40
	(*ptr)[4] = 50
	//for i := 0; i < len(arr); i++ {
	//	fmt.Print(arr[i], " ")
	//}
	fmt.Println(ptr)
	//fmt.Println("之前", arr)
	//array(arr)
	//fmt.Println("之后", arr)
	fmt.Println("之前", arr)
	array2(ptr)
	fmt.Println("之后", arr)
	/**
	指针数组
	*/

	var arr2 [5]*int
	arr2 = [...]*int{
		&arr[0],
		&arr[1],
		&arr[2],
		&arr[3],
		&arr[4],
	}
	//每一个元素都是指针
	fmt.Println(*arr2[0])
}
func array(arr [5]int) {
	arr[0] = 200
}
func array2(arr *[5]int) {
	arr[0] = 200
}

函数指针

可以详情查看go中的数据类型 —函数

func main() {
	f := func(a int, b int) {
		fmt.Println(a + b)
	}
	fmt.Println("已知函数本身就可以看作存储函数体地址的指针", f)
	fmt.Println("函数取地址就像多重指针", &f)
	//a2 := []int{12, 34, 56}
	//c := &a2

}

  • 22
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蓝胖子不是胖子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值