首页 > 后端开发 > Golang > Golang 在结构体中使用 Exp for big.Int 奇怪地改变了值

Golang 在结构体中使用 Exp for big.Int 奇怪地改变了值

王林
发布: 2024-02-12 14:00:07
转载
924 人浏览过

Golang 在结构体中使用 Exp for big.Int 奇怪地改变了值

问题内容

在下面,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 保持不变。

编辑:要添加的是,我还打印了 abd.cd.q 的指针地址,并且在 Exp 之前和之后都不同: fmt.Printf("%p %p %p %p n", &a, &b, &d.c, &d.q)bd.cd.q 的指针地址,并且在 Exp 之前和之后都不同: fmt.Printf("%p %p %p %p n", &a, &b, &d.c, &d.q)

解决方法

这是一个更简单的示例,显示了相同的内容:

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
}
登录后复制

因此,对其进行浅复制将导致两个实例的切片共享相同的后备数组< /a> (当切片中的元素发生更改时,这可能会导致不可预测的结果)。

在您的示例中,当 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中文网其他相关文章!

来源:stackoverflow.com
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板