Uniswap合约开发7大技巧解析 提升DeFi项目效率

robot
摘要生成中

从Uniswap代码中学到的智能合约开发技巧

近期在编写去中心化交易所开发教程时,参考了Uniswap V3的代码实现,学到了很多有趣的智能合约开发技巧。作为第一次尝试开发DeFi合约的开发者,这些技巧对想要学习合约开发的新手会很有帮助。

Web3 新手系列:我从 Uniswap 代码中学到的合约开发小技巧

可预测的合约地址

通常部署合约得到的地址看似随机,因为与nonce有关。但在某些场景下,我们需要通过交易对等信息推断出合约地址,比如判断交易权限或获取池子地址。

Uniswap采用CREATE2方式创建合约,添加salt参数使地址可预测。新地址生成逻辑为:hash("0xFF",创建者地址, salt, initcode)。这种方法使合约地址可预先计算。

Web3 新手系列:我从 Uniswap 代码中学到的合约开发小技巧

巧用回调函数

Solidity中合约可以互相调用。某些场景下,A方法调用B,B在被调用方法中回调A,这种模式很实用。

例如Uniswap的swap方法交易时,会回调swapCallback,传入实际需要的Token数量。调用方需在回调中转入Token,而不是将swap拆分为两步调用。这确保了swap方法的完整执行,无需繁琐的变量记录来保证安全性。

用异常传递信息,用try-catch预估交易

Uniswap的Quoter合约中,用try-catch包裹执行了UniswapV3Pool的swap方法。这是为了模拟swap来预估所需Token数量,但预估时并不实际交换Token所以会报错。

Uniswap在交易回调中抛出特殊错误,然后捕获该错误并从中解析所需信息。这种方法避免了为预估需求而改造swap方法,使逻辑更简洁。

Web3 新手系列:我从 Uniswap 代码中学到的合约开发小技巧

大数运算解决精度问题

Uniswap中涉及大量计算,如根据价格和流动性计算交换Token数量。为避免除法运算丢失精度,计算过程常用<< FixedPoint96.RESOLUTION操作,即左移96位,相当于乘以2^96。

左移后再做除法,可在正常交易不溢出的前提下保证精度。虽然理论上仍有微小精度损失,但已可接受。

Share方式计算收益

Uniswap需记录LP(流动性提供者)的手续费收益。显然不能每次交易都为每个LP记录手续费,这会消耗大量Gas。

解决方案是记录总手续费和每单位流动性应分配的手续费。LP提取时根据持有的流动性计算可提取手续费,类似股票分红。

Web3 新手系列:我从 Uniswap 代码中学到的合约开发小技巧

区分链上链下信息

链上存储相对昂贵,不是所有信息都需要上链或从链上获取。如Uniswap前端调用的许多接口是传统Web2接口。

交易池列表、信息等可存储在普通数据库,定期从链上同步。无需实时调用RPC接口获取相关数据。当然关键交易仍在链上进行。

合约拆分与标准合约复用

一个项目可能包含多个实际部署的合约。即使只部署一个合约,代码也可通过继承拆分为多个合约维护。

如Uniswap的NonfungiblePositionManager合约继承了多个合约。其中ERC721Permit直接使用了OpenZeppelin的ERC721合约,既便于通过NFT管理头寸,又提高了开发效率。

总结

亲自动手开发比单纯阅读文章更有效。尝试实现简易版去中心化交易所,能更深入理解Uniswap的代码实现,学习到实际项目中的知识点。

Web3 新手系列:我从 Uniswap 代码中学到的合约开发小技巧

UNI-3.59%
DEFI-9.7%
此页面可能包含第三方内容,仅供参考(非陈述/保证),不应被视为 Gate 认可其观点表述,也不得被视为财务或专业建议。详见声明
  • 赞赏
  • 7
  • 分享
评论
0/400
PoolJumpervip
· 07-28 02:11
零基建搞几把
回复0
老韭の自白vip
· 07-27 04:18
你能让我看懂就算厉害了...
回复0
链上资深吃瓜群众vip
· 07-27 03:51
建议实操练练手~
回复0
钱包被套到退休vip
· 07-25 17:02
终于找到解决我被套现状的出路了
回复0
PanicSeller69vip
· 07-25 17:00
V3还是有点东西的 干!
回复0
degenonymousvip
· 07-25 16:58
代码党的福音了
回复0
空投碎梦师vip
· 07-25 16:56
开发不如直接偷代码
回复0
交易,随时随地
qrCode
扫码下载 Gate APP
社群列表
简体中文
  • 简体中文
  • English
  • Tiếng Việt
  • 繁體中文
  • Español
  • Русский
  • Français (Afrique)
  • Português (Portugal)
  • Bahasa Indonesia
  • 日本語
  • بالعربية
  • Українська
  • Português (Brasil)