从零开始了解 Uniswap

本文通过大概100行代码演示了Uniswap 原理

你一定听说过Uniswap,甚至非常了解它的运作方式。

Uniswap是一个神奇的产品,没有订单簿?没有服务器?没有预言机?WTF!!!!

X* Y = K 是其✨魔法所在。✨

这是一个永远执行、去中心化的AMM的执行引擎

让我们从头开始实现它,了解它到底是如何工作的。

本文的目的不是要进行一个完美和安全的Uniswap实现。而是为了以最简化的方式展示其机制,你可以在文末查看到本文的完整代码。

我们假定你对以太坊有基本了解,例如了解 Solidity、智能合约、交易和代币。

如果还不了解,订阅全面掌握Solidity智能合约开发专栏全面学习。

Pool(流动性资金池)

Uniswap的故事从Pool(以下简称:流动池)开始。流动池是一个智能合约,它储备了两个代币的token0token1

contract LooneySwapPool is ERC20 {
  address public token0;
  address public token1;

  // Reserve of token 0
  uint public reserve0;

  // Reserve of token 1
  uint public reserve1;
  ...
}

创建一个流动池很简单:只需要指定这个流动池可以存储的2个代币:

constructor(address _token0, address _token1) ERC20("LiquidityProvider", "LP") {
  token0 = _token0;
  token1 = _token1;
}

合约创建后,它将在状态变量reserve0reserve1中分别记录token0token1的数量余额。

请注意,该合约也扩展了ERC20。这是因为,除了计算每个代币的储备量之外,它本身也是一个代币,正如你从名字中可以看出,该代币代表了流动性提供者的余额。

增加流动性资金

为了将流动性添加到我们的资金池中,必须调用add函数,指定我们要存入的每个代币的数量:

function add(uint amount0, uint amount1)

首先我们把代币转移到流动池(本身)里:

assert(IERC20(token0).transferFrom(msg.sender, address(this), amount0));
assert(IERC20(token1).transferFrom(msg.sender, address(this), amount1));

由于用户不再拥有这些代币的控制权,我们需要一种方法来给他们一个对象,代表他们在流动池中的所占的份额。这就是流动池代币的作用!

我们按照用户在资金池中的份额,铸造新的流动池代币。

如果这是第一个提供流动资金的用户,我们会铸造一个初始金额,有效地给他们一个100%的份额。

_mint(msg.sender, INITIAL_SUPPLY);

否则,我们将计算出按比例的份额,并铸造出等值的LP代币:


uint reserve0After = reserve0 + amount0;
uint reserve1After = reserve1 + amount1;

...

uint currentSupply = totalSupply(); // Current supply of LP tokens
uint newSupplyGivenReserve0Ratio = reserve0After * currentSupply / reserve0;
uint newSupplyGivenReserv...

剩余50%的内容订阅专栏后可查看

0 条评论

请先 登录 后评论
翻译小组
翻译小组

首席翻译官

92 篇文章, 14105 学分