分享web3j使用教程,包括转账,查询以及合约调用
简介
Web3j 是一个轻量级的 Java 库,用于与以太坊区块链交互。支持以下功能:
- 创建和管理以太坊钱包。
- 发送交易和查询区块链数据。
- 部署和调用智能合约。
- 监听事件和过滤器。
环境准备
在 pom.xml
中添加 Maven 依赖:
1 2 3 4 5 6
| <dependency> <groupId>org.web3j</groupId> <artifactId>core</artifactId> <version>4.8.7</version> </dependency>
|
基础使用
连接到以太坊节点
1 2 3 4 5
| Web3j web3j = Web3j.build(new HttpService("https://mainnet.infura.io/v3/YOUR_API_KEY"));
Web3j web3j = Web3j.build(new HttpService("http://localhost:8545"));
|
区块相关
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| public BigInteger getBlockNumber() throws Exception { return this.w3.ethBlockNumber().send().getBlockNumber(); }
public Block getBlockByNumber(BigInteger blockNumber) throws Exception { return this.w3.ethGetBlockByNumber(DefaultBlockParameter.valueOf(blockNumber), true).send().getBlock(); }
public BigInteger getBlockTimestamp(BigInteger blockNumber) throws Exception { return this.getBlockByNumber(blockNumber).getTimestamp(); }
public Block getBlockByHash(String blockHash) throws Exception { return this.w3.ethGetBlockByHash(blockHash, true).send().getBlock(); }
|
账户相关
1 2 3 4 5 6 7 8 9 10 11
|
public BigInteger getBalance(String address) throws Exception { return this.w3.ethGetBalance(address, DefaultBlockParameterName.LATEST).send().getBalance(); }
public BigInteger getAddressNextNonce(String address) throws Exception { return this.w3.ethGetTransactionCount(address, DefaultBlockParameterName.PENDING).send().getTransactionCount(); }
|
交易相关
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
|
public List<Transaction> getTransactionsByBlockNumber(BigInteger blockNumber) throws Exception { List<Transaction> txs = new ArrayList<>(); Block block = this.w3.ethGetBlockByNumber(DefaultBlockParameter.valueOf(blockNumber), true).send().getBlock(); if (block == null) { return null; } for (TransactionResult<Transaction> tx : block.getTransactions()) { Transaction t = tx.get(); txs.add(t); } return txs; }
public Transaction getTransactionByHash(String transactionHash) throws Exception { return this.w3.ethGetTransactionByHash(transactionHash).send().getTransaction().get(); }
public Optional<TransactionReceipt> getTransactionReceipt(String transactionHash) throws Exception { return this.w3.ethGetTransactionReceipt(transactionHash).send().getTransactionReceipt(); }
|
代币转账
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
|
public String ethSendTransaction(String privateKey, String to, BigInteger value) throws Exception { ECKeyPair ecKeyPair = ECKeyPair.create(new BigInteger(privateKey, 16)); String address = Keys.toChecksumAddress(Keys.getAddress(ecKeyPair));
BigInteger nonce = this.getAddressNextNonce(address); BigInteger gasPrice = this.getGasPrice(); BigInteger gasLimit = BigInteger.valueOf(21000); RawTransaction rawTransaction = RawTransaction.createEtherTransaction(nonce, gasPrice, gasLimit, to, value);
Credentials credentials = Credentials.create(ecKeyPair);
byte[] signedMessage = TransactionEncoder.signMessage(rawTransaction, credentials); String signedTx = Numeric.toHexString(signedMessage);
EthSendTransaction transaction = this.w3.ethSendRawTransaction(signedTx).send(); return transaction.getTransactionHash(); }
|
合约交互
需要转准备已经部署好的合约,并知道合约相关函数以及事件信息
需要了解solidity相关知识
合约代码示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
| pragma solidity ^0.5.0; //指定编译器版本 contract DemoStudy { //合约名称 uint256 public myNumber;
// 构造函数,init data constructor() public { myNumber = 10; }
function setNumber0(uint256 number) public { myNumber = number; }
event NumberSet1(uint256 number); // 事件 function setNumber1(uint256 number) public { myNumber = number; emit NumberSet1(number); //触发事件 }
// 抛出的事件中有index下标 event NumberSet2(uint256 indexed number); function setNumber2(uint256 number) public { myNumber = number; emit NumberSet1(number); emit NumberSet2(number); }
// 抛出多个参数 event NumberSet3(uint256 number,address user); function setNumber3(uint256 number) public returns (bool) { myNumber = number; emit NumberSet1(number); emit NumberSet2(number); emit NumberSet3(number,msg.sender); return true; }
function getNumber() public view returns (uint256) { return myNumber; } }
|
call合约方法
零地址调用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
|
public String ethCall(String functionName, String contractAddress, Type... functionArguments) throws Exception {
List<Type> argumentList = new ArrayList<>(Arrays.asList(functionArguments)); Function functionCall = new Function(functionName, argumentList, Collections.<TypeReference<?>>emptyList()); String functionEncoder = FunctionEncoder.encode(functionCall);
org.web3j.protocol.core.methods.request.Transaction requestTransaction = new org.web3j.protocol.core.methods.request.Transaction(EthConstant.BLACK_HOLE_ADDRESS, BigInteger.ZERO, BigInteger.ZERO, EthConstant.DEFAULT_GAS_LIMIT, contractAddress, BigInteger.ZERO, functionEncoder);
EthCall ethCall = this.w3.ethCall(requestTransaction, DefaultBlockParameterName.LATEST).send();
if (ethCall.isReverted()) { throw new Exception("Call contract revert! Reason: " + ethCall.getRevertReason()); }
return ethCall.getValue();
er.ethCall("getNumber","ContractAddress");
Type[] setNumberParam = { new Uint256(BigInteger.valueOf(123)) }; er.ethCall("setNumber0","ContractAddress",setNumberParam); }
|
指定账户调用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
|
public String ethCall(String privateKey, String functionName, String contractAddress, Type... functionArguments) throws Exception {
ECKeyPair ecKeyPair = ECKeyPair.create(new BigInteger(privateKey, 16)); String address = Keys.toChecksumAddress(Keys.getAddress(ecKeyPair));
BigInteger nonce = this.getAddressNextNonce(address); BigInteger gasPrice = this.getGasPrice(); BigInteger gasLimit = BigInteger.valueOf(21000);
List<Type> argumentList = new ArrayList<>(Arrays.asList(functionArguments)); Function functionCall = new Function(functionName, argumentList, Collections.<TypeReference<?>>emptyList()); String functionEncoder = FunctionEncoder.encode(functionCall);
org.web3j.protocol.core.methods.request.Transaction requestTransaction = new org.web3j.protocol.core.methods.request.Transaction(address, nonce, gasPrice, gasLimit, contractAddress, BigInteger.ZERO, functionEncoder);
EthCall ethCall = this.w3.ethCall(requestTransaction, DefaultBlockParameterName.LATEST).send();
if (ethCall.isReverted()) { throw new Exception("Call contract revert! Reason: " + ethCall.getRevertReason()); }
return ethCall.getValue(); }
|
调用合约方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
|
public String sendCallContractRawTransaction(String privateKey, String contractAddress, String functionName, Type... functionArguments) throws Exception { ECKeyPair ecKeyPair = ECKeyPair.create(new BigInteger(privateKey, 16)); String address = Keys.toChecksumAddress(Keys.getAddress(ecKeyPair));
BigInteger nonce = this.getAddressNextNonce(address); BigInteger gasPrice = this.getGasPrice(); BigInteger gasLimit = BigInteger.valueOf(21000); List<Type> argumentList = new ArrayList<>(Arrays.asList(functionArguments)); Function functionCall = new Function(functionName, argumentList, Collections.<TypeReference<?>>emptyList()); String functionEncoder = FunctionEncoder.encode(functionCall); RawTransaction rawTransaction = RawTransaction.createTransaction(nonce, gasPrice, gasLimit, contractAddress, BigInteger.ZERO, functionEncoder);
Credentials credentials = Credentials.create(ecKeyPair);
byte[] signedMessage = TransactionEncoder.signMessage(rawTransaction, credentials); String signedTx = Numeric.toHexString(signedMessage);
Transaction requestTransaction = new Transaction(address, nonce, gasPrice, gasLimit, contractAddress, BigInteger.ZERO, functionEncoder); EthCall ethCall = this.w3.ethCall(requestTransaction, DefaultBlockParameterName.LATEST).send(); if (ethCall.isReverted()) { throw new Exception("Call contract revert! Reason: " + ethCall.getRevertReason()); } EthSendTransaction transaction = this.w3.ethSendRawTransaction(signedTx).send(); return transaction.getTransactionHash(); }
|
强烈建议在发送交易之前先call一下合约方法,防止因参数有误或者合约执行失败而造成的gas浪费
监听功能
创建日志过滤器
创建一个过滤器,用于监听符合特定条件的合约事件日志, 返回一个过滤器ID。
topic说明:对合约方法调用后抛出事件的keccak256的hash。
keccak256(NumberSet1(uint256 number)) = 0x9283ff3607e3da47fe0733402e95abd90472527fbbac5a26edffcbd508134773
1 2 3 4 5 6 7 8 9 10 11 12 13 14
|
public BigInteger createFilter(BigInteger blockStart, BigInteger blockEnd, String contractAddress, String... topics) throws Exception { EthFilter requestEthFilter = new EthFilter(DefaultBlockParameter.valueOf(blockStart), DefaultBlockParameter.valueOf(blockEnd), contractAddress); requestEthFilter = requestEthFilter.addOptionalTopics(topics); return this.w3.ethNewFilter(requestEthFilter).send().getFilterId(); }
|
获取过滤器日志
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
|
public List<Log> getFilterLogs(BigInteger filterId) throws Exception { List<Log> logs = new ArrayList<>(); EthLog ethLog = this.w3.ethGetFilterLogs(filterId).send(); for (LogResult<Log> log : ethLog.getLogs()) { Log l = log.get(); logs.add(l); } return logs; }
public List<Log> getFilterChanges(BigInteger filterId) throws Exception { List<Log> logs = new ArrayList<>(); EthLog ethLog = this.w3.ethGetFilterChanges(filterId).send(); for (LogResult<Log> log : ethLog.getLogs()) { Log l = log.get(); logs.add(l); } return logs; }
|
直接查询日志
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
|
public List<Log> getLogs(BigInteger blockStart, BigInteger blockEnd, String contractAddress,String... topics) throws Exception {
EthFilter filter = new EthFilter( DefaultBlockParameter.valueOf(blockStart), DefaultBlockParameter.valueOf(blockEnd), contractAddress ); filter.addOptionalTopics(topics);
List<Log> logs = new ArrayList<>(); EthLog ethLog = this.w3.ethGetLogs(filter).send(); for (LogResult<Log> log : ethLog.getLogs()) { Log l = log.get(); logs.add(l); } return logs; }
|
参考资料