聊聊以太坊智能合约ABI
Posted lucasma.eth
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了聊聊以太坊智能合约ABI相关的知识,希望对你有一定的参考价值。
聊聊以太坊智能合约ABI
ABI的全称是 Application Binary Interface
,所以它是一个接口。或者说是一个标准。这个标准描述了在以太坊生态中如何跟合约进行交互,这个交互包含外部客户端调用合约的接口,也包括合约之间的交互。
ABI假设所有的编码都是在编译阶段确定的,也就是静态的。而不是运行时动态生成的。
我们知道智能合约的本质也是程序,程序之间交互要用二进制码(binary code),ABI就是这样一个标准规定合约的代码如何进行编码和解码。
ABI的标准规范的内容包括合约的函数列表、函数名称、参数名称、参数类型、返回类型等。这些信息以JSON格式保存。这些JSON是可读的。
比如一个智能合约
// SPDX-License-Identifier: GPL-3.0
pragma solidity ^0.8.4;
contract Test
constructor() b = hex"12345678901234567890123456789012";
event Event(uint indexed a, bytes32 b);
event Event2(uint indexed a, bytes32 b);
error InsufficientBalance(uint256 available, uint256 required);
function foo(uint a) public emit Event(a, b);
bytes32 b;
它的JSON如下:
[
"type":"error",
"inputs": ["name":"available","type":"uint256","name":"required","type":"uint256"],
"name":"InsufficientBalance"
,
"type":"event",
"inputs": ["name":"a","type":"uint256","indexed":true,"name":"b","type":"bytes32","indexed":false],
"name":"Event"
,
"type":"event",
"inputs": ["name":"a","type":"uint256","indexed":true,"name":"b","type":"bytes32","indexed":false],
"name":"Event2"
,
"type":"function",
"inputs": ["name":"a","type":"uint256"],
"name":"foo",
"outputs": []
]
我们看到的是一个json数组,数组的第一个元素是一个名为InsufficientBalance
,类型是error的接口,同时还指明了它的入参。
其它元素也是类似,就不展开说了。
我们也可以在remix编译的时候看到ABI信息。
接下来看看具体的编码规则。
还是拿一个具体的合约举例子,
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.4.16 <0.9.0;
contract Foo
function bar(bytes3[2] memory) public pure
function baz(uint32 x, bool y) public pure returns (bool r) r = x > 32 || y;
function sam(bytes memory, bool, uint[] memory) public pure
比如我要调用bar
这个方法,传入的参数是69和true,它会被编码为
0xcdcd77c000000000000000000000000000000000000000000000000000000000000000450000000000000000000000000000000000000000000000000000000000000001
首先这个0xcdcd77c0
,是函数的id,它占据了编码的前四个字节,它其实是下面这个函数签名的Keccak hash
的前四个字节。
baz(uint32,bool)
```
![在这里插入图片描述](https://img-blog.csdnimg.cn/2da45e03e19c47bc91ca4f7427e45da3.png#pic_center)
```
0x0000000000000000000000000000000000000000000000000000000000000045
```
是入参69的十六进制数并且补齐到32字节。(ABI的编码会对所有的数据编码强制32字节对齐)
```
0x0000000000000000000000000000000000000000000000000000000000000001
```
是入参true(true就是1)补齐32字节的结果。
----
参考:
* https://docs.soliditylang.org/en/develop/abi-spec.html#abi
以上是关于聊聊以太坊智能合约ABI的主要内容,如果未能解决你的问题,请参考以下文章