
在下面,a 发生了意外的变化。 d 使用 a (*a) 的值构造,然后 d.c 被正确更改。但是为什么 a 更改为 Exp 的第一个参数?
type Decimal struct {
c big.Int // coefficient
q big.Int // exponent
}
a := big.NewInt(1)
b := big.NewInt(2)
d := Decimal{*a, *b}
d.c.Exp(big.NewInt(11), big.NewInt(2), big.NewInt(0))
fmt.Println(a, b, d) // 11 2 {{false [121]} {false [2]}}我希望 a 保持不变。
编辑:要添加的是,我还打印了 a、b、d.c、d.q 的指针地址,并且在 Exp 之前和之后都不同: fmt.Printf("%p %p %p %p \n", &a, &b, &d.c, &d.q)
这是一个更简单的示例,显示了相同的内容:
立即学习“go语言免费学习笔记(深入)”;
x := big.NewInt(1) y := *x y.Exp(big.NewInt(11), big.NewInt(2), big.NewInt(0)) fmt.Println(x) // 11 fmt.Println(&y) // 121
首先要考虑的是 y.Exp “设置 z = x**y mod |m| (即 m 的符号被忽略),并返回 z。";因此 y 的值发生变化(如上所示)。
要了解为什么会改变 x 的值,您可以从文档开始:
“浅复制”正是上面的 y := *x (或代码中的 d := Decimal{*a, *b} )所做的。所以解决方案是遵循上面的建议:
x := big.NewInt(1) y := new(big.Int).Set(x) // Avoid shallow copy y.Exp(big.NewInt(11), big.NewInt(2), big.NewInt(0)) fmt.Println(x) // 1 fmt.Println(y) // 121
(您可以在示例中执行类似的操作)。
要解释为什么会发生这种情况,您需要查看big.Int 已定义。这需要检查一些文件,但归结为(简化!):
type Int struct {
neg bool // sign
abs []uint // absolute value of the integer
}因此,对其进行浅复制将导致两个实例的切片共享相同的后备数组 (当切片中的元素发生更改时,这可能会导致不可预测的结果)。
在您的示例中,当 set 已运行。展示这一点的更简单方法是:
x := big.NewInt(1) y := *x y.Set(big.NewInt(11)) fmt.Println(x) // 11 fmt.Println(&y) // 11
以上就是Golang 在结构体中使用 Exp for big.Int 奇怪地改变了值的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号