# WTF Solidity极简入门: 47. 可升级合约 **Published by:** [0xAA](https://paragraph.com/@wtfacademy/) **Published on:** 2022-09-04 **URL:** https://paragraph.com/@wtfacademy/wtf-solidity-47 ## Content 我最近在重新学solidity,巩固一下细节,也写一个“WTF Solidity极简入门”,供小白们使用(编程大佬可以另找教程),每周更新1-3讲。 推特:所有代所有代码和教程开源在github: github.com/AmazingAng/WTFSolidity 社区:Discord|微信群|官网 wtf.academy 所有代码和教程开源在github: github.com/AmazingAng/WTFSolidity这一讲,我们将介绍可升级合约(Upgradeable Contract)。教学用的合约由openzepplin中的合约简化而来,可能有安全问题,不应用于生产环境。可升级合约如果你理解了代理合约,就很容易理解可升级合约。它是一个可以更改逻辑合约的代理合约。简单实现下面我们实现一个简单的可升级合约,它包含3个合约:代理合约,旧的逻辑合约,和新的逻辑合约。代理合约这个代理合约比第46讲中的简单。我们没有在它的fallback()函数中使用内联汇编,而仅仅用了implementation.delegatecall(msg.data);。因此,回调函数没有返回值,但足够教学使用了。 它包含3个变量:implementation:逻辑合约地址。admin:admin地址。words:字符串,可以通过逻辑合约的函数改变。它包含3个函数构造函数:初始化admin和逻辑合约地址。fallback():回调函数,将调用委托给逻辑合约upgrade():升级函数,改变逻辑合约地址,只能由admin调用// SPDX-License-Identifier: MIT // wtf.academy pragma solidity ^0.8.4; // 简单的可升级合约,管理员可以通过升级函数更改逻辑合约地址,从而改变合约的逻辑。 // 教学演示用,不要用在生产环境 contract SimpleUpgrade { address public implementation; // 逻辑合约地址 address public admin; // admin地址 string public words; // 字符串,可以通过逻辑合约的函数改变 // 构造函数,初始化admin和逻辑合约地址 constructor(address _implementation){ admin = msg.sender; implementation = _implementation; } // fallback函数,将调用委托给逻辑合约 fallback() external payable { (bool success, bytes memory data) = implementation.delegatecall(msg.data); } // 升级函数,改变逻辑合约地址,只能由admin调用 function upgrade(address newImplementation) external { require(msg.sender == admin); implementation = newImplementation; } } 旧逻辑合约这个逻辑合约包含3个状态变量,与保持代理合约一致,防止插槽冲突。它只有一个函数foo(),将代理合约中的wrods的值改为"old"。// 逻辑合约1 contract Logic1 { // 状态变量和proxy合约一致,防止插槽冲突 address public implementation; address public admin; string public words; // 字符串,可以通过逻辑合约的函数改变 // 改变proxy中状态变量,选择器: 0xc2985578 function foo() public{ words = "old"; } } 新逻辑合约这个逻辑合约包含3个状态变量,与保持代理合约一致,防止插槽冲突。它只有一个函数foo(),将代理合约中的wrods的值改为"new"。// 逻辑合约2 contract Logic2 { // 状态变量和proxy合约一致,防止插槽冲突 address public implementation; address public admin; string public words; // 字符串,可以通过逻辑合约的函数改变 // 改变proxy中状态变量,选择器:0xc2985578 function foo() public{ words = "new"; } } Remix实现部署新旧逻辑合约Logic1和Logic2。部署可升级合约SimpleUpgrade,将implementation地址指向把旧逻辑合约。利用选择器0xc2985578,在代理合约中调用旧逻辑合约Logic1的foo()函数,将wrods的值改为"old"。调用upgrade(),将implementation地址指向新逻辑合约Logic2。利用选择器0xc2985578,在代理合约中调用新逻辑合约Logic2的foo()函数,将wrods的值改为"new"。总结这一讲,我们介绍了一个简单的可升级合约。它是一个可以改变逻辑合约的代理合约,给不可更改的智能合约增加了升级功能。但是,这个合约有选择器冲突的问题,存在安全隐患。之后我们会介绍解决这一隐患的可升级合约标准:透明代理和UUPS。 ## Publication Information - [0xAA](https://paragraph.com/@wtfacademy/): Publication homepage - [All Posts](https://paragraph.com/@wtfacademy/): More posts from this publication - [RSS Feed](https://api.paragraph.com/blogs/rss/@wtfacademy): Subscribe to updates - [Twitter](https://twitter.com/0xAA_Science): Follow on Twitter