3分钟Solidity: 9.3 通过 Create2 预计算合约地址

欢迎订阅专栏3分钟Solidity--智能合约--Web3区块链技术必学

如需获取本内容的最新版本,请参见 Cyfrin.io 上的 Precompute Contract 地址与 Create2(代码示例)

合约地址可以在合约部署前通过create2预先计算出来

scss 复制代码
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.26;

contract Factory {
    // 返回新部署合约的地址
    function deploy(address _owner, uint256 _foo, bytes32 _salt)
        public
        payable
        returns (address)
    {
        // 这种语法是一种无需汇编即可调用 create2 的新方法,你只需要传递 salt 参数。
        // https://docs.soliditylang.org/en/latest/control-structures.html#salted-contract-creations-create2
        return address(new TestContract{salt: _salt}(_owner, _foo));
    }
}

// 这是使用汇编语言的旧方法
contract FactoryAssembly {
    event Deployed(address addr, uint256 salt);

    // 1.  获取待部署合约的字节码
    // 注意:_owner 和 _foo 是 TestContract 构造函数的参数
    function getBytecode(address _owner, uint256 _foo)
        public
        pure
        returns (bytes memory)
    {
        bytes memory bytecode = type(TestContract).creationCode;

        return abi.encodePacked(bytecode, abi.encode(_owner, _foo));
    }

    // 2.  计算要部署的合约地址
    //注意:_salt是一个随机数,用于生成地址。
    function getAddress(bytes memory bytecode, uint256 _salt)
        public
        view
        returns (address)
    {
        bytes32 hash = keccak256(
            abi.encodePacked(
                bytes1(0xff), address(this), _salt, keccak256(bytecode)
            )
        );

        // 注意:将哈希的最后20个字节转换为地址
        return address(uint160(uint256(hash)));
    }

    // 3. 部署合约
    // 注意:
    // 检查事件日志 Deployed,其中包含已部署的 TestContract 的地址。
    // 日志中的地址应与上述计算的地址一致。
    function deploy(bytes memory bytecode, uint256 _salt) public payable {
        address addr;

        /*
        注意:如何调用create2
        create2(v, p, n, s)
        在内存位置p到p + n处创建新合约代码
        并发送v wei
        返回新地址
        其中新地址 = keccak256(0xff + address(this) + s + keccak256(mem[p...(p+n)))的前20字节
        s = 大端序256位值
        */
        assembly {
            addr :=
                create2(
                    callvalue(), // 当前调用中发送的wei
                    // 跳过前32字节后开始实际代码
                    add(bytecode, 0x20),
                    mload(bytecode), // 加载前32字节中包含的代码大小
                    _salt // 来自函数参数的盐
                )

            if iszero(extcodesize(addr)) { revert(0, 0) }
        }

        emit Deployed(addr, _salt);
    }
}

contract TestContract {
    address public owner;
    uint256 public foo;

    constructor(address _owner, uint256 _foo) payable {
        owner = _owner;
        foo = _foo;
    }

    function getBalance() public view returns (uint256) {
        return address(this).balance;
    }
}

Remix Lite 尝试一下

相关推荐
DICOM医学影像21 小时前
15. Go-Ethereum测试Solidity ERC20合约 - Go-Ethereum调用合约方法
开发语言·后端·golang·区块链·智能合约·以太坊·web3.0
雷焰财经1 天前
智能合约引领数字人民币新征程 宇信科技参与共建科技金融新图景
科技·金融·智能合约
雷焰财经1 天前
智能合约破解“砖头逻辑”:宇信科技解读数字人民币如何重塑科技金融
科技·金融·智能合约
Rockbean1 天前
3分钟Solidity: 8.6 ERC721(NFT)
web3·智能合约·solidity
Rockbean1 天前
3分钟Solidity: 9.1 gasless代币转账
web3·智能合约·solidity
Rockbean1 天前
3分钟Solidity: 8.7 ERC1155
web3·智能合约·solidity
Sui_Network1 天前
Sui 2025 年终回顾:支付、BTC 与机构采用篇
大数据·人工智能·物联网·web3·去中心化·区块链
DICOM医学影像1 天前
14. Go-Ethereum测试Solidity ERC20合约 - Go-Ethereum调用区块链方法
开发语言·golang·go·区块链·solidity·以太坊·go-ethereum
Zoey的笔记本2 天前
构建去中心化协作引擎:基于开源框架的Web3团队项目管理实践
开源·web3·区块链