1

我正在通过 uniswap 代码试图理解代码,其中大部分都很清楚,但我确实有几个问题。

在这个函数中:

function createPair(address tokenA, address tokenB) external returns (address pair) {
    require(tokenA != tokenB, 'UniswapV2: IDENTICAL_ADDRESSES');
    (address token0, address token1) = tokenA < tokenB ? (tokenA, tokenB) : (tokenB, tokenA);
    require(token0 != address(0), 'UniswapV2: ZERO_ADDRESS');
    require(getPair[token0][token1] == address(0), 'UniswapV2: PAIR_EXISTS'); // single check is sufficient
    bytes memory bytecode = type(UniswapV2Pair).creationCode;
    bytes32 salt = keccak256(abi.encodePacked(token0, token1));
    assembly {
        pair := create2(0, add(bytecode, 32), mload(bytecode), salt)
    }
    IUniswapV2Pair(pair).initialize(token0, token1);
    getPair[token0][token1] = pair;
    getPair[token1][token0] = pair; // populate mapping in the reverse direction
    allPairs.push(pair);
    emit PairCreated(token0, token1, pair, allPairs.length);

有流水线。根据solidity docs,这部署了一个新合同,因为我不明白它是如何工作的,它从哪里获取代码等等。

那么是否有可能以某种方式将其“翻译”为坚固性?非常感谢!

4

1 回答 1

1

它利用create2操作码允许您将合约部署到可由其字节码和盐确定的地址。

Uniswap V2 是用 Solidity 0.5 编写的,无法create2直接从该语言生成操作码。所以你必须使用低级汇编来实际使用这个操作码。

当前版本 0.8 允许传递salt产生create2(而不是常规create)操作码的参数。

pragma solidity ^0.8;

contract UniswapV2Pair {
}

contract MyContract {
    function createPair() external {
        bytes32 salt = 0x1234567890123456789012345678901234567890123456789012345678901234;
        address pair = address(
            new UniswapV2Pair{salt: salt}()
        );
    }
}

Uniswap 使用对令牌地址的组合作为盐,并且字节码始终相同。这有效地允许为每个独特的配对组合部署一个合约。

例子:

  • 代币0x1230x456总是会产生UniswapV2Pair合约地址0xabc
  • 但是一旦你改变了盐,它就会改变部署的合约地址。所以代币总是会产生0x123合约地址。0x789UniswapV2Pair0xdef
于 2022-02-15T09:15:52.433 回答