为了账号安全,请及时绑定邮箱和手机立即绑定

按数字拆分 big.Int

按数字拆分 big.Int

Go
慕后森 2023-03-15 14:44:55
我正在尝试将 a 拆分big.Int为多个int64s,以便每个都是较大数字的一部分,标准偏移量为 18 位。例如,给定以下输入值1234512351234088800000999,我希望得到以下输出:[351234088800000999, 1234512]。对于负数,我希望所有部分都是负数(即-1234512351234088800000999produces [-351234088800000999, -1234512])。我已经知道我可以这样做以获得我想要的结果:func Split(input *big.Int) []int64 {    const width = 18    asStr := in.Coefficient().Text(10)    strLen := len(asStr)    offset := 0    if in.IsNegative() {        offset = 1    }    length := int(math.Ceil(float64(strLen-offset) / width))    ints := make([]int64, length)    for i := 1; i <= length; i++ {        start := strLen - (i * width)        end := start + width        if start < 0 || (start == 1 && asStr[0] == '-') {            start = 0        }        ints[i-1], _ = strconv.ParseInt(asStr[start:end], 10, 64)        if offset == 1 && ints[i-1] > 0 {            ints[i-1] = 0 - ints[i-1]        }    }    return ints}但是,我不喜欢使用字符串解析的想法,也不喜欢使用strconv. 有什么方法可以直接使用吗big.Int?
查看完整描述

1 回答

?
沧海一幻觉

TA贡献1824条经验 获得超5个赞

您可以使用该DivMod函数在这里执行您需要的操作,并特别注意处理负数:


var offset = big.NewInt(1e18)


func Split(input *big.Int) []int64 {

    rest := new(big.Int)

    rest.Abs(input)


    var ints []int64

    r := new(big.Int)

    for {

        rest.DivMod(rest, offset, r)

        ints = append(ints, r.Int64() * int64(input.Sign()))

        if rest.BitLen() == 0 {

            break

        }

    }

    return ints

}

将每个输出乘以input.Sign()确保如果输入为负,则每个输出将为负。输出值的总和乘以 1e18 乘以它们在输出中的位置应该等于输入。


查看完整回答
反对 回复 2023-03-15
  • 1 回答
  • 0 关注
  • 73 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信