在一个智能合约中调用另外一个外部智能合约的函数,我们可以通过接口 interface
的方式进行调用。另外,还有一种比较底层的调用方法,就是使用call、staticcall和delegatecall函数。它们是一种低级、底层的调用方式,具有更大的灵活性。我们将分别进行讲解。
一、底层调用 call
1、函数语法
(bool success, bytes memory result) = address(contractAddress).call{value: valueToSend}(data);
其中的返回值的含义如下:
success
:指示调用外部函数是否成功。
result
:调用的外部函数的返回值。
其中的参数的含义如下:
contractAddress
:要调用的外部合约的地址。
valueToSend
:发送到外部合约的 ETH
数量,它的单位是 wei
。这是一个可选参数,如果无需发送 ETH
,就可以选择忽略这个参数。
data
:发送到外部合约的数据。它是对外部函数签名和参数进行编码,而生成的字节数组。
比如,我们要调用一个外部合约的函数 functionName(uint256)
,那么就需要使用 abi
对函数签名和参数进行编码。
编码方法如下:
abi.encodeWithSignature("functionName(uint256)", parameter);
2、函数示例
我们先准备一个被调用的合约Receive.sol
,合约中定义了一个函数 foo(),且该函数能够接受ETH
。另外,这个合约还定义了 receive()
和constructor()
函数,使之具有接收 ETH
的能力。call在合约Caller.sol的使用场景如下:
- 只调用外部函数
- 只向外部合约发送ETH
- 调用外部函数并发送ETH
调用者合约代码:
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
contract Caller{
constructor() payable {}
// 1只调用外部合约的函数
// 参数 contractAddress 是被调用合约的地址
function callExternalFunc(address contractAddress) external returns(bool, bytes memory) {
// 对函数签名和参数进行编码
bytes memory data = abi.encodeWithSignature("foo(uint256)", 8);
// 调用外部合约函数
return contractAddress.call(data);
}
// 2只向外部合约发送ETH
// 参数 contractAddress 是被调用合约的地址
function callExternal(address contractAddress) external returns(bool, bytes memory) {
// 调用外部合约函数
return contractAddress.call{value: 1 ether}("");
}
// 3调用外部合约的函数及发送ETH
// 参数 contractAddress 是被调用合约的地址
function callExternalFuncAndETH(address contractAddress) external returns(bool, bytes memory) {
// 对函数签名和参数进行编码
bytes memory data = abi.encodeWithSignature("foo(uint256)",