
本文深入探讨go语言中通过指针修改字符串值的两种常见操作:`*dest = src` 和 `dest = &src`。我们将详细解析这两种赋值方式的底层机制和作用域影响,阐明为何前者能成功修改原始字符串,而后者仅在函数局部生效,旨在帮助开发者避免常见的指针误用,掌握go语言中指针的正确使用姿态。
在Go语言中,理解指针的工作原理对于编写高效且无误的代码至关重要,尤其是在需要通过函数修改外部变量值时。字符串作为一种不可变类型,其值的修改通常涉及创建新字符串。然而,当我们通过指针传递字符串时,如何正确地修改指针所指向的字符串变量,常常会引起混淆。本文将通过一个具体的例子,深入剖析 *dest = src 和 dest = &src 这两种看似相似却效果迥异的指针操作。
在Go语言中,指针是一个存储另一个变量内存地址的变量。
Go语言函数参数传递总是采用值传递(pass-by-value)的方式。这意味着当一个变量作为参数传递给函数时,函数会接收到该变量的一个副本。对于指针类型,传递的是指针本身的副本,而非指针所指向的值的副本。
考虑以下函数 changeStringValueNotOK:
立即学习“go语言免费学习笔记(深入)”;
func changeStringValueNotOK(dest *string, src string) {
dest = &src
}当调用 changeStringValueNotOK(&a, b) 时:
这种方式的本质是修改了函数内部的指针副本,使其指向了另一个内存地址,而原始指针在函数外部仍然指向其最初的内存地址。
现在,我们来看正确的做法 changeStringValueOK:
func changeStringValueOK(dest *string, src string) {
*dest = src
}当调用 changeStringValueOK(&a, b) 时:
这种方式的本质是利用了指针的解引用,直接修改了指针所指向的内存地址中的内容,从而实现了对外部变量的修改。
为了更好地理解上述概念,请看以下完整的Go程序:
package main
import (
"fmt"
)
// changeStringValueNotOK 示例错误的指针赋值
// 它会修改函数内部的指针变量dest,使其指向局部变量src的地址
// 但不会影响函数外部的原始变量a
func changeStringValueNotOK(dest *string, src string) {
fmt.Printf(" [NotOK] dest (inside func, before assignment): %p, points to: %q\n", dest, *dest)
fmt.Printf(" [NotOK] src (inside func): %p, value: %q\n", &src, src)
dest = &src // 错误:修改的是局部指针副本,使其指向局部变量src的地址
fmt.Printf(" [NotOK] dest (inside func, after assignment): %p, points to: %q\n", dest, *dest)
}
// changeStringValueOK 示例正确的指针赋值
// 它会解引用指针dest,并修改其所指向的内存地址中的值
// 从而影响函数外部的原始变量a
func changeStringValueOK(dest *string, src string) {
fmt.Printf(" [OK] dest (inside func, before assignment): %p, points to: %q\n", dest, *dest)
fmt.Printf(" [OK] src (inside func): %p, value: %q\n", &src, src)
*dest = src // 正确:解引用dest,修改其指向的值
fmt.Printf(" [OK] dest (inside func, after assignment): %p, points to: %q\n", dest, *dest)
}
func main() {
a := "Hello"
b := "World"
fmt.Printf("main: Initial a: %q (address: %p)\n", a, &a)
fmt.Printf("main: Initial b: %q (address: %p)\n", b, &b)
fmt.Println("--- Calling changeStringValueNotOK ---")
changeStringValueNotOK(&a, b)
fmt.Printf("main: After changeStringValueNotOK, a: %q (address: %p)\n", a, &a) // 仍然是 "Hello"
fmt.Println("--- Calling changeStringValueOK ---")
changeStringValueOK(&a, b)
fmt.Printf("main: After changeStringValueOK, a: %q (address: %p)\n", a, &a) // 现在是 "World"
}运行上述代码,你将看到清晰的输出,展示 a 的值在 changeStringValueNotOK 调用后未变,而在 changeStringValueOK 调用后成功改变。额外添加的 Printf 语句可以帮助你观察函数内部指针变量 dest 和 src 的地址变化。
理解 *dest = src 和 dest = &src 之间的微妙差异是掌握Go语言指针操作的关键。前者用于修改指针所指向的值,而后者用于修改指针变量自身的值。在大多数需要通过函数修改外部变量的场景中,我们都应该使用解引用赋值 *dest = value。正确地运用这些概念,将使你的Go程序更加健壮和可预测。
以上就是Go语言指针操作:如何正确修改通过指针传递的字符串值的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号