web3j 和 web3js 中的签名不同?

Signatures different in web3j and web3js?

我正在使用 web3J 版本“org.web3j:core:4.6.0”并遇到以下问题。当我签署相同的原始消息(字符串)时,我使用 web3js 和 web3j 获得不同的签名。来自 web3js 的签名是有效的(而 web3j 不是),我使用 solidity ecrecover 函数对其进行了测试。值得一提的是,您在代码片段中看到的哈希是相同的。

Java 代码 (web3j).

String private_key = "25218ba6de76757feba214961b588345e1415b267383af9fda13dd032ae75fff";
Credentials credentials1 = Credentials.create(privateKey1)


String message = "0x12cf1496120ada41033631fd6fa12613416c18696b70e4b072b3d1157ee165c7";

Sign.SignatureData signature = Sign.signMessage(message.getBytes(), credentials.getEcKeyPair());

String signature_V = Numeric.toHexString(signature.getV());
String signature_R = Numeric.toHexString(signature.getR());
String signature_S = Numeric.toHexString(signature.getS());

java 代码结果 -

signature_V=: 0x1b
signature_R=: 0x9f09f364e24577eb9dde9f1e3e2c0db0473fdd03e38945de0a5d97a92ee9b5c1
signature_S=: 0x3d7860741f64144ac8317880dc110ffadf020f712322f7c11bcf9cf3e446c212

使用 web3js 代码。

let privateKey = "25218ba6de76757feba214961b588345e1415b267383af9fda13dd032ae75fff"
let message = "0x12cf1496120ada41033631fd6fa12613416c18696b70e4b072b3d1157ee165c7";

let sign = web3.eth.accounts.sign(message, privateKey); 

web3js 结果 -

signature_V=: 0x1b
signature_R=: 0x3db7bc52699c3b34d9b8b617c5e7646ce5b6899d278c061cf83dcd216316f0ef
signature_S=: 0x0dea9a606165cb7fc3e4b7959ba43f33025b44886c6363cd4d7788297a1cbf39

首先,使用您的 js 示例和最新的 web3 库 (1.3.0) 我得到了这些结果:

  1. Web3js 示例


然后,在您的 java 代码中,您需要使用 signPrefixedMessage,因为 web3js 版本包装输入消息,对其进行哈希处理,然后才进行签名。此外,如果消息为十六进制,web3js 会将其正确转换为字节数组,您只需使用字节的字符串。结果我收到了两个相同的签名。

  1. Web3j 示例 (kotlin)

ECDSA 具有随机成分,因此相同数据的签名应该不同