在以太坊智能合约开发中,call、staticcall 和 delegatecall 是三种低级调用方法(Low-level Call),它们允许合约与其他合约进行交互。这三者的主要区别体现在调用上下文和执行权限等方面。
1. call
• 作用:执行目标合约的函数,同时改变目标合约的状态。
• 上下文:目标合约的代码在自己的上下文中执行,msg.sender 和 msg.value 不变。
• 修改状态:可以修改目标合约的状态。
• Gas 消耗:调用者需要为目标合约的执行分配 Gas。
• 返回值:返回 true 或 false,表示调用是否成功。
• 适用场景:通用的跨合约调用。
用法示例
(bool success, bytes memory data) = targetContractAddress.call{value: msgValue}(
abi.encodeWithSignature("functionName(uint256)", 123)
);
require(success, "Call failed");
** 特点**
1. 目标合约的代码执行在目标合约的存储(Storage)中。
2. 可以携带 ETH(通过 {value: msgValue} 传递)。
3. 适合用于调用其他合约的函数,可以调用 public 和 external 的函数。
2. staticcall
• 作用:执行目标合约的代码,但不能修改目标合约或 EVM 的状态。
• 上下文:与 call 类似,代码在目标合约的上下文中执行。
• 修改状态:不能修改状态(包括目标合约和 EVM 状态),否则会导致交易回滚。
• Gas 消耗:与普通调用相比,Gas 消耗更低,因为没有状态修改。
• 返回值:返回 true 或 false,表示调用是否成功。
• 适用场景:只读操作,例如查询目标合约的存储或执行纯计算逻辑。
用法示例
(bool success, bytes memory data) = targetContractAddress.staticcall(
abi.encodeWithSignature("viewFunction()")
);
require(success, "StaticCall failed");
特点
1. 适用于 view 和 pure 函数,因为它们不会修改状态。
2. 只能进行只读操作。
3. 如果尝试修改状态(如修改变量或发送 ETH),交易会失败并回滚。
3. delegatecall
• 作用:执行目标合约的代码,但在调用者的存储(Storage)上下文中运行。
• 上下文:
• msg.sender 和 msg.value 保持不变(继承调用者的上下文)。
• 目标合约的代码会在调用者的存储中执行。
• 修改状态:可以修改调用者的状态,而不是目标合约的状态。
• 返回值:返回 true 或 false,表示调用是否成功。
** 用法示例**
(bool success, bytes memory data) = targetContractAddress.delegatecall(
abi.encodeWithSignature("functionName()")
);
require(success, "DelegateCall failed");
特点
1. 目标合约的代码操作的是调用者的存储(Storage)。
2. 适合代理合约的实现,比如 OpenZeppelin 中的代理模式(Proxy Pattern)。
3. delegatecall 会继承调用者的 msg.sender 和 msg.value。
4. 要小心目标合约的代码对调用者存储的修改,因为存储位置是共享的。
示例场景说明
1. call 的应用场景
• 用于简单的跨合约调用或向外部合约发送 ETH。
• 例如:
• 调用外部合约的函数。
• 将 ETH 发送给目标合约。
2. staticcall 的应用场景
• 用于只读操作(调用 view 或 pure 函数),确保不会修改链上的状态。
• 常用于查询外部合约的状态。
3. delegatecall 的应用场景
• 用于 代理模式(Proxy Pattern)。
• 在代理合约中调用目标合约的逻辑,但状态存储在代理合约中。
注意事项
1. 安全风险:delegatecall 非常强大,但也非常危险,因为它执行目标合约的代码但修改的是调用者的存储,可能导致存储数据被意外覆盖。
2. Gas 消耗:低级调用(call、staticcall、delegatecall)的 Gas 消耗通常比直接函数调用更高,因为需要手动处理编码和返回值。
3. 错误处理:低级调用返回布尔值表示成功与否,因此需要检查 success 是否为 true,否则调用可能会默默失败。
通过这些低级调用,你可以实现灵活的合约交互,但需要特别小心存储和安全相关的问题,尤其是在使用 delegatecall 时。
完整例子
1. 使用 call
场景:调用外部合约函数并传递 ETH
• call 适用于调用外部合约的函数,可以携带 ETH,并且执行的是目标合约的逻辑和存储。
示例代码
被调用的合约 (TargetContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract TargetContract {
uint256 public value;
function setValue(uint256 _value) public payable {
value = _value;
}
}
调用合约 (CallerContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract CallerContract {
event Response(bool success, bytes data);
function callSetValue(address target, uint256 _value) public payable {
(bool success, bytes memory data) = target.call{value: msg.value}(
abi.encodeWithSignature("setValue(uint256)", _value)
);
emit Response(success, data);
}
}
解释
(1). call 使用 abi.encodeWithSignature 生成调用数据。
(2). 参数:{value: msg.value} 传递 ETH。
(3). 如果调用成功,返回 success = true,否则为 false。
2. 使用 staticcall
** 场景:只读操作(调用 view 或 pure 函数)**
• staticcall 适用于调用 不会修改状态 的函数,例如 view 和 pure 函数。
• 如果目标函数尝试修改状态,调用会失败并回滚。
示例代码
被调用的合约 (TargetContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract TargetContract {
uint256 public value;
function getValue() public view returns (uint256) {
return value;
}
function setValue(uint256 _value) public payable {
value = _value;
}
}
调用合约 (CallerContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract CallerContract {
event Response(bool success, bytes data);
function staticCallGetValue(address target) public {
(bool success, bytes memory data) = target.staticcall(
abi.encodeWithSignature("getValue()")
);
require(success, "Staticcall failed");
uint256 result = abi.decode(data, (uint256));
emit Response(success, abi.encode(result));
}
}
解释
(1). staticcall 只能调用 view 或 pure 函数。
(2). 如果目标函数尝试修改状态(如写入变量),调用会失败。
(3). 返回的数据需要通过 abi.decode 进行解码。
3. 使用 delegatecall
场景:代理合约调用其他合约的逻辑,但在自己的存储上下文中执行
• delegatecall 会在调用者合约的存储上下文中运行目标合约的代码。
• 常用于代理模式(Proxy Pattern)。
示例代码
逻辑合约 (LogicContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract LogicContract {
uint256 public value; // 注意:这里的存储位置与调用者合约要一致
function setValue(uint256 _value) public {
value = _value;
}
}
代理合约 (ProxyContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract ProxyContract {
uint256 public value; // 与逻辑合约存储位置相同
address public logicContract; // 逻辑合约的地址
constructor(address _logicContract) {
logicContract = _logicContract;
}
event Response(bool success, bytes data);
function setValueThroughDelegateCall(uint256 _value) public {
(bool success, bytes memory data) = logicContract.delegatecall(
abi.encodeWithSignature("setValue(uint256)", _value)
);
require(success, "Delegatecall failed");
emit Response(success, data);
}
}
解释
(1). ProxyContract 调用 LogicContract 的 setValue 方法。
(2). 存储上下文:delegatecall 执行 LogicContract 的代码,但修改的是 ProxyContract 中的 value 变量。
(3). 逻辑合约的存储结构必须与调用者合约一致,否则会导致存储冲突。
总结

疑问
1、delegatecall 并不拷贝目标合约的上下文或存储到调用者合约,而是动态映射目标合约的逻辑到调用者的上下文中执行。那么这种映射在EVM或者在EVM运行过程中是如何体现的呢?
delegatecall 的执行机制是 EVM 的核心功能之一,它通过 上下文动态切换 和 逻辑重用 实现目标合约的代码在调用者合约的上下文中运行。这种行为不是通过“拷贝”目标合约的代码或存储来完成的,而是通过 EVM 的堆栈帧、程序计数器(Program Counter, PC)、存储空间映射等方式动态实现的。以下从 EVM 的运行过程来详细解析 delegatecall 的映射机制。
delegatecall 在 EVM 的运行过程中如何体现:
1. delegatecall 的调用指令
在 Solidity 编译后的字节码中,delegatecall 是一个原生的 EVM 操作码(opcode),对应的操作码值是 0xf4。它的作用是:
•执行目标合约的代码逻辑。
•使用调用者合约的存储和上下文。
2. delegatecall 的调用过程
当调用者合约执行 delegatecall 指令时,EVM 的运行步骤如下:
(1) 初始化调用
•调用者合约将通过堆栈将 delegatecall 的参数(包括目标合约地址、输入数据、Gas 限额等)压入 EVM 的主栈中。
•EVM 根据参数,定位到目标合约的代码存储区。
(2) 上下文切换
•存储上下文:EVM 将调用者合约的存储地址(storage root)映射为当前执行上下文,而不是目标合约的存储。
•msg.sender 和 msg.value:这些调用元数据保持调用者合约的上下文,目标合约无法直接感知其存储和状态。
(3) 加载目标合约代码
•EVM 从目标合约的字节码中加载代码到程序计数器(Program Counter, PC),开始逐行解释执行。
•代码执行时,所有的存储访问指令(如 SLOAD、SSTORE)会操作调用者合约的存储,而非目标合约的存储。
delegatecall 的具体实现细节
1. 堆栈帧的切换
每次合约调用(包括 call、delegatecall 等)都会在 EVM 中创建一个新的 堆栈帧(stack frame):
•堆栈帧存储了当前执行的上下文信息,包括:
•当前的程序计数器(PC)。
•当前的存储地址映射(Storage Root)。
•当前的内存分配范围(Memory Context)。
•当前的调用者信息(msg.sender、msg.value 等)。
•对于 delegatecall:
•程序计数器(PC) 切换到目标合约的代码执行起点。
•存储地址映射(Storage Root) 保持调用者合约的存储地址(调用者的存储空间被目标合约使用)。
•调用者信息(msg.sender 和 msg.value) 保持调用者合约的上下文信息。
EVM 的堆栈帧切换逻辑确保目标合约的逻辑在调用者合约的上下文中运行。
2. 存储访问的重定向
•目标合约中的存储访问指令(SLOAD、SSTORE)在 EVM 中会使用一个全局状态树(State Trie)。
•对于 delegatecall:
•目标合约的存储访问指令,会被重定向到调用者合约的存储区域。
•这通过 EVM 的存储根哈希(Storage Root Hash)完成,delegatecall 保持调用者合约的存储根作为执行上下文的存储根。
3. 代码执行的动态加载
•delegatecall 不会将目标合约的代码“拷贝”到调用者合约。
•EVM 通过字节码指令直接加载目标合约的代码到程序计数器(PC),并在调用者上下文中逐行执行。
在以太坊智能合约开发中,call、staticcall 和 delegatecall 是三种低级调用方法(Low-level Call),它们允许合约与其他合约进行交互。这三者的主要区别体现在调用上下文和执行权限等方面。
1. call
• 作用:执行目标合约的函数,同时改变目标合约的状态。
• 上下文:目标合约的代码在自己的上下文中执行,msg.sender 和 msg.value 不变。
• 修改状态:可以修改目标合约的状态。
• Gas 消耗:调用者需要为目标合约的执行分配 Gas。
• 返回值:返回 true 或 false,表示调用是否成功。
• 适用场景:通用的跨合约调用。
用法示例
(bool success, bytes memory data) = targetContractAddress.call{value: msgValue}(
abi.encodeWithSignature("functionName(uint256)", 123)
);
require(success, "Call failed");
** 特点**
1. 目标合约的代码执行在目标合约的存储(Storage)中。
2. 可以携带 ETH(通过 {value: msgValue} 传递)。
3. 适合用于调用其他合约的函数,可以调用 public 和 external 的函数。
2. staticcall
• 作用:执行目标合约的代码,但不能修改目标合约或 EVM 的状态。
• 上下文:与 call 类似,代码在目标合约的上下文中执行。
• 修改状态:不能修改状态(包括目标合约和 EVM 状态),否则会导致交易回滚。
• Gas 消耗:与普通调用相比,Gas 消耗更低,因为没有状态修改。
• 返回值:返回 true 或 false,表示调用是否成功。
• 适用场景:只读操作,例如查询目标合约的存储或执行纯计算逻辑。
用法示例
(bool success, bytes memory data) = targetContractAddress.staticcall(
abi.encodeWithSignature("viewFunction()")
);
require(success, "StaticCall failed");
特点
1. 适用于 view 和 pure 函数,因为它们不会修改状态。
2. 只能进行只读操作。
3. 如果尝试修改状态(如修改变量或发送 ETH),交易会失败并回滚。
3. delegatecall
• 作用:执行目标合约的代码,但在调用者的存储(Storage)上下文中运行。
• 上下文:
• msg.sender 和 msg.value 保持不变(继承调用者的上下文)。
• 目标合约的代码会在调用者的存储中执行。
• 修改状态:可以修改调用者的状态,而不是目标合约的状态。
• 返回值:返回 true 或 false,表示调用是否成功。
** 用法示例**
(bool success, bytes memory data) = targetContractAddress.delegatecall(
abi.encodeWithSignature("functionName()")
);
require(success, "DelegateCall failed");
特点
1. 目标合约的代码操作的是调用者的存储(Storage)。
2. 适合代理合约的实现,比如 OpenZeppelin 中的代理模式(Proxy Pattern)。
3. delegatecall 会继承调用者的 msg.sender 和 msg.value。
4. 要小心目标合约的代码对调用者存储的修改,因为存储位置是共享的。
示例场景说明
1. call 的应用场景
• 用于简单的跨合约调用或向外部合约发送 ETH。
• 例如:
• 调用外部合约的函数。
• 将 ETH 发送给目标合约。
2. staticcall 的应用场景
• 用于只读操作(调用 view 或 pure 函数),确保不会修改链上的状态。
• 常用于查询外部合约的状态。
3. delegatecall 的应用场景
• 用于 代理模式(Proxy Pattern)。
• 在代理合约中调用目标合约的逻辑,但状态存储在代理合约中。
注意事项
1. 安全风险:delegatecall 非常强大,但也非常危险,因为它执行目标合约的代码但修改的是调用者的存储,可能导致存储数据被意外覆盖。
2. Gas 消耗:低级调用(call、staticcall、delegatecall)的 Gas 消耗通常比直接函数调用更高,因为需要手动处理编码和返回值。
3. 错误处理:低级调用返回布尔值表示成功与否,因此需要检查 success 是否为 true,否则调用可能会默默失败。
通过这些低级调用,你可以实现灵活的合约交互,但需要特别小心存储和安全相关的问题,尤其是在使用 delegatecall 时。
完整例子
1. 使用 call
场景:调用外部合约函数并传递 ETH
• call 适用于调用外部合约的函数,可以携带 ETH,并且执行的是目标合约的逻辑和存储。
示例代码
被调用的合约 (TargetContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract TargetContract {
uint256 public value;
function setValue(uint256 _value) public payable {
value = _value;
}
}
调用合约 (CallerContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract CallerContract {
event Response(bool success, bytes data);
function callSetValue(address target, uint256 _value) public payable {
(bool success, bytes memory data) = target.call{value: msg.value}(
abi.encodeWithSignature("setValue(uint256)", _value)
);
emit Response(success, data);
}
}
解释
(1). call 使用 abi.encodeWithSignature 生成调用数据。
(2). 参数:{value: msg.value} 传递 ETH。
(3). 如果调用成功,返回 success = true,否则为 false。
2. 使用 staticcall
** 场景:只读操作(调用 view 或 pure 函数)**
• staticcall 适用于调用 不会修改状态 的函数,例如 view 和 pure 函数。
• 如果目标函数尝试修改状态,调用会失败并回滚。
示例代码
被调用的合约 (TargetContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract TargetContract {
uint256 public value;
function getValue() public view returns (uint256) {
return value;
}
function setValue(uint256 _value) public payable {
value = _value;
}
}
调用合约 (CallerContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract CallerContract {
event Response(bool success, bytes data);
function staticCallGetValue(address target) public {
(bool success, bytes memory data) = target.staticcall(
abi.encodeWithSignature("getValue()")
);
require(success, "Staticcall failed");
uint256 result = abi.decode(data, (uint256));
emit Response(success, abi.encode(result));
}
}
解释
(1). staticcall 只能调用 view 或 pure 函数。
(2). 如果目标函数尝试修改状态(如写入变量),调用会失败。
(3). 返回的数据需要通过 abi.decode 进行解码。
3. 使用 delegatecall
场景:代理合约调用其他合约的逻辑,但在自己的存储上下文中执行
• delegatecall 会在调用者合约的存储上下文中运行目标合约的代码。
• 常用于代理模式(Proxy Pattern)。
示例代码
逻辑合约 (LogicContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract LogicContract {
uint256 public value; // 注意:这里的存储位置与调用者合约要一致
function setValue(uint256 _value) public {
value = _value;
}
}
代理合约 (ProxyContract.sol)
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract ProxyContract {
uint256 public value; // 与逻辑合约存储位置相同
address public logicContract; // 逻辑合约的地址
constructor(address _logicContract) {
logicContract = _logicContract;
}
event Response(bool success, bytes data);
function setValueThroughDelegateCall(uint256 _value) public {
(bool success, bytes memory data) = logicContract.delegatecall(
abi.encodeWithSignature("setValue(uint256)", _value)
);
require(success, "Delegatecall failed");
emit Response(success, data);
}
}
解释
(1). ProxyContract 调用 LogicContract 的 setValue 方法。
(2). 存储上下文:delegatecall 执行 LogicContract 的代码,但修改的是 ProxyContract 中的 value 变量。
(3). 逻辑合约的存储结构必须与调用者合约一致,否则会导致存储冲突。
总结

疑问
1、delegatecall 并不拷贝目标合约的上下文或存储到调用者合约,而是动态映射目标合约的逻辑到调用者的上下文中执行。那么这种映射在EVM或者在EVM运行过程中是如何体现的呢?
delegatecall 的执行机制是 EVM 的核心功能之一,它通过 上下文动态切换 和 逻辑重用 实现目标合约的代码在调用者合约的上下文中运行。这种行为不是通过“拷贝”目标合约的代码或存储来完成的,而是通过 EVM 的堆栈帧、程序计数器(Program Counter, PC)、存储空间映射等方式动态实现的。以下从 EVM 的运行过程来详细解析 delegatecall 的映射机制。
delegatecall 在 EVM 的运行过程中如何体现:
1. delegatecall 的调用指令
在 Solidity 编译后的字节码中,delegatecall 是一个原生的 EVM 操作码(opcode),对应的操作码值是 0xf4。它的作用是:
•执行目标合约的代码逻辑。
•使用调用者合约的存储和上下文。
2. delegatecall 的调用过程
当调用者合约执行 delegatecall 指令时,EVM 的运行步骤如下:
(1) 初始化调用
•调用者合约将通过堆栈将 delegatecall 的参数(包括目标合约地址、输入数据、Gas 限额等)压入 EVM 的主栈中。
•EVM 根据参数,定位到目标合约的代码存储区。
(2) 上下文切换
•存储上下文:EVM 将调用者合约的存储地址(storage root)映射为当前执行上下文,而不是目标合约的存储。
•msg.sender 和 msg.value:这些调用元数据保持调用者合约的上下文,目标合约无法直接感知其存储和状态。
(3) 加载目标合约代码
•EVM 从目标合约的字节码中加载代码到程序计数器(Program Counter, PC),开始逐行解释执行。
•代码执行时,所有的存储访问指令(如 SLOAD、SSTORE)会操作调用者合约的存储,而非目标合约的存储。
delegatecall 的具体实现细节
1. 堆栈帧的切换
每次合约调用(包括 call、delegatecall 等)都会在 EVM 中创建一个新的 堆栈帧(stack frame):
•堆栈帧存储了当前执行的上下文信息,包括:
•当前的程序计数器(PC)。
•当前的存储地址映射(Storage Root)。
•当前的内存分配范围(Memory Context)。
•当前的调用者信息(msg.sender、msg.value 等)。
•对于 delegatecall:
•程序计数器(PC) 切换到目标合约的代码执行起点。
•存储地址映射(Storage Root) 保持调用者合约的存储地址(调用者的存储空间被目标合约使用)。
•调用者信息(msg.sender 和 msg.value) 保持调用者合约的上下文信息。
EVM 的堆栈帧切换逻辑确保目标合约的逻辑在调用者合约的上下文中运行。
2. 存储访问的重定向
•目标合约中的存储访问指令(SLOAD、SSTORE)在 EVM 中会使用一个全局状态树(State Trie)。
•对于 delegatecall:
•目标合约的存储访问指令,会被重定向到调用者合约的存储区域。
•这通过 EVM 的存储根哈希(Storage Root Hash)完成,delegatecall 保持调用者合约的存储根作为执行上下文的存储根。
3. 代码执行的动态加载
•delegatecall 不会将目标合约的代码“拷贝”到调用者合约。
•EVM 通过字节码指令直接加载目标合约的代码到程序计数器(PC),并在调用者上下文中逐行执行。
Share Dialog
Share Dialog
Subscribe to Untitled
Subscribe to Untitled
<100 subscribers
<100 subscribers
No activity yet