gpt4 book ai didi

java - Java中的C#BigInteger等效项

转载 作者:行者123 更新时间:2023-12-01 08:56:11 26 4
gpt4 key购买 nike

在将旧项目从C#迁移到Java时,我遇到了以下函数:

private static byte[] solveChallenge(byte[] data, int offset, int level){
var x = new BigInteger(1, data, 00 + offset, 64);
var n = new BigInteger(1, data, 64 + offset, 64);
Console.Out.WriteLine("x " + x);
Console.Out.WriteLine("n " + n);
return x.ModPow(BigInteger.Two.Pow(level), n).ToByteArrayUnsigned();
}


第一个问题是Java需要使用一个int数组,因为它不支持4个字节。

我试图迁移此功能,结果是:

private byte[] solveChallenge(int[] dummyData, int offset, int level) {
int[] x = Arrays.copyOfRange(dummyData, 00 + offset, 64 + offset);
int[] n = Arrays.copyOfRange(dummyData, 64 + offset, 64 + offset + 64);
BigInteger bigInteger = this.toBigInteger(x, offset, level);
BigInteger bigInteger2 = this.toBigInteger(n, offset, level);
System.out.println("bigInteger : " + bigInteger);
System.out.println("bigInteger2 : " + bigInteger2);
return null; //only for debugging purposes since it doesn't calculate the right value
}

private BigInteger toBigInteger(int[] data, int offset, int length) {
byte[] array = new byte[data.length * 4];
ByteBuffer bbuf = ByteBuffer.wrap(array);
bbuf.order(ByteOrder.BIG_ENDIAN);
IntBuffer ibuf = bbuf.asIntBuffer();
ibuf.put(data);
int pos = bbuf.position();
return new BigInteger(1, array);
}


带有伪数据的字节数组如下:

byte[] dummyData = { 3, 184, 233, 18, 14, 29, 29, 96, 191, 154, 42, 188, 39, 129, 28, 43, 179, 155, 17, 253, 104, 150, 211, 242, 218, 37, 111, 45, 234, 231, 159, 226, 229, 241, 118, 6, 93, 154, 216, 39, 147, 240, 235, 37, 235, 4, 162, 199, 3, 146, 219, 250, 55, 145, 187, 109, 129, 154, 21, 99, 195, 213, 24, 171, 72, 186, 108, 207, 23, 245, 227, 206, 155, 28, 83, 117, 208, 52, 146, 247, 107, 86, 250, 91, 120, 132, 103, 96, 151, 135, 239, 236, 171, 105, 148, 132, 210, 164, 24, 121, 46, 63, 26, 6, 147, 66, 0, 113, 226, 228, 96, 54, 247, 58, 216, 123, 239, 127, 35, 235, 117, 225, 156, 139, 231, 191, 23, 16, 199, 0, 0, 39, 16, 24, 148, 19, 244, 58, 27, 168, 77, 144, 41, 52, 231, 216, 63, 52, 240, 101, 221, 196, 61, 210, 132, 247, 242, 148, 127, 73, 20, 136, 213, 82, 154, 2, 118, 247, 27, 193, 213, 31, 20, 107, 184, 115, 129, 102, 95, 130, 42, 105, 70, 228, 111, 211, 108, 149, 43, 200, 239, 220, 58, 61, 35, 169, 51, 37, 140, 12, 150, 126, 170, 183, 95, 104, 102, 218, 171, 207, 57, 172, 57, 179, 147, 134, 251, 49, 168, 124, 154, 198, 102, 204, 88, 99, 110, 128, 73, 0, 0, 0, 0, }
int[] dummyData = { 3, 184, 233, 18, 14, 29, 29, 96, 191, 154, 42, 188, 39, 129, 28, 43, 179, 155, 17, 253, 104, 150, 211, 242, 218, 37, 111, 45, 234, 231, 159, 226, 229, 241, 118, 6, 93, 154, 216, 39, 147, 240, 235, 37, 235, 4, 162, 199, 3, 146, 219, 250, 55, 145, 187, 109, 129, 154, 21, 99, 195, 213, 24, 171, 72, 186, 108, 207, 23, 245, 227, 206, 155, 28, 83, 117, 208, 52, 146, 247, 107, 86, 250, 91, 120, 132, 103, 96, 151, 135, 239, 236, 171, 105, 148, 132, 210, 164, 24, 121, 46, 63, 26, 6, 147, 66, 0, 113, 226, 228, 96, 54, 247, 58, 216, 123, 239, 127, 35, 235, 117, 225, 156, 139, 231, 191, 23, 16, 199, 0, 0, 39, 16, 24, 148, 19, 244, 58, 27, 168, 77, 144, 41, 52, 231, 216, 63, 52, 240, 101, 221, 196, 61, 210, 132, 247, 242, 148, 127, 73, 20, 136, 213, 82, 154, 2, 118, 247, 27, 193, 213, 31, 20, 107, 184, 115, 129, 102, 95, 130, 42, 105, 70, 228, 111, 211, 108, 149, 43, 200, 239, 220, 58, 61, 35, 169, 51, 37, 140, 12, 150, 126, 170, 183, 95, 104, 102, 218, 171, 207, 57, 172, 57, 179, 147, 134, 251, 49, 168, 124, 154, 198, 102, 204, 88, 99, 110, 128, 73, 0, 0, 0, 0, }


对于这两个程序(Java的int [])。
现在棘手的部分是:C#版本中的两个console.writeLines打印出这些值:


x 9684545129450563760811299662317393444224628107338840479787476089424784627212472709829491894762094799169291328402088637379520119916743215796610009247820616

n 9763871338200074467010224713304495633710029487465175174521229296454337320893449010179832428198865193070649591972415477477112413797655820164788963534901447


但是,java版本会打印出来:


BIGINTEGER:1384487636125275878231756214272583599746580615359367670970277020749291139661692747315333543253640854017991792553366599843364101930280429076504504568047635739899488976412197760572976468791377536211931839579896955320180378597836121754579135558686349090024111181198802897769845015231252620523393484949244408744682782755518452496013632526359151451572349278849384043601027009481486942547342423606635954241049110830041192225899912691920909177042864305251984053702303477007620462086885017315368482937756361611079609304592527067945041572917438130926972982331458069293145225576550553455068231595004541872039273047261256

bigInteger2:1399536414555386036016732078389615490583790294241057959880006399246816751078900125109635269290836392362378727229942541859385680156220979323032655120786012653956497057790441490280149802553283148934560467004214861358910591562936643819189479913049351409549959873164398598055666211815485920029665855456844605176135634635342254062123114318883965379446208014067811273077439277906607738683244293540374300284127713297527831803091844750310847726168860685624174631286757167096065831296412508483930854847141757821223969111690911979811180162621071129937054751632338429538035407779745520240238094955254575062803261704634567


两个程序的方法参数分别为1和10000(例如Java):

testclass.solveChallenge(dummyData, 1, 10000);


理想情况下,bigInteger应该具有与x和bigInteger2相同的值,例如n。
我的错误在哪里?我已经搜寻了5个小时,但找不到任何错误:/

最佳答案

我终于找到了问题:来自Java的IntBuffer以4字节格式导入其数字(从逻辑上讲,因为int包含4字节)。这就是为什么它用int数组中的给定值的有符号值填充三个字节(零),最后一个填充。
该命令将光带入了黑暗:

BigInteger x = new BigInteger("9684545129450563760811299662317393444224628107338840479787476089424784627212472709829491894762094799169291328402088637379520119916743215796610009247820616")


由于我能够获得应该获得的实际BigInteger。应用 x.toByteArray()之后,我看到这些字节以不同的方式填充(没有三个零字节)。
通过仔细检查C#生成的值,我发现它奇怪地总是使第二个变量的第一个字节为空(在我的情况下为n)。
应用这些规则后,我能够获得所需的值。

这是我的最终解决方法:

private byte[] solveChallenge(int[] dummyData, int offset, int level) {
int[] x = Arrays.copyOfRange(dummyData, 00 + offset, 65 + offset);
int[] n = Arrays.copyOfRange(dummyData, offset, 65 + offset + 64);
n[0] = 0; //null first byte in n, don't copy this line, this only works for my problem!
BigInteger xInt = this.toBigInteger(x, offset);
BigInteger nInt = this.toBigInteger(n, offset);
BigInteger inner = pow(BigInteger.valueOf(2), BigInteger.valueOf(level));
return xInt.modPow(inner, nInt).toByteArray();
}
private BigInteger pow(BigInteger base, BigInteger exponent) {
BigInteger result = BigInteger.ONE;
while (exponent.signum() > 0) {
if (exponent.testBit(0)) result = result.multiply(base);
base = base.multiply(base);
exponent = exponent.shiftRight(1);
}
return result;
}
private BigInteger toBigInteger(int[] data, int offset, int length) {
byte[] array = new byte[data.length * 4];
ByteBuffer bbuf = ByteBuffer.wrap(array);
IntBuffer ibuf = bbuf.asIntBuffer();
ibuf.put(data);
bbuf.order(ByteOrder.BIG_ENDIAN);
int i = 0;
byte[] newArr = new byte[(array.length / 4)-1];
//convert to Little-Endian like systems and only keep every fourth byte
for(int j = 0; j < array.length-1; ++j) {
if(j % 4 == 0 && j > 0) {
int pos = j - 1;
newArr[i] = array[pos];
++i;
}
}
return new BigInteger(1, newArr);
}

关于java - Java中的C#BigInteger等效项,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42027489/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com