以太坊智能合约升级(一)

版权声明:1、本BLOG的目的、形式及内容。   此BLOG为个人维护BLOG,内容均来自原创及互连网转载。最终目的为收集整理自己需要的文章技术等内容,不涉及商业用途。 2、有关原创文章的版权   本BLOG上原创文章未经本人许可,不得用于商业用途及传统媒体。网络媒体转载请注明出处,否则属于侵权行为。 3、有关本站侵权   本BLOG所转载的内容,均是本人未发现有对文章版权声明的文章且无来自传统媒体。如 https://blog.csdn.net/xq723310/article/details/84844720

    我们都知道以太坊的智能合约是没办法升级的,然而一个程序出现bug是不可避免的,而通常解决的办法就是升级了。就是因为没有办法升级,导致很多项目出现bug后,就是重新发布合约,这样就给项目方带来了极大的损失,如果我们可以升级智能合约,就可以带来极大的方便。

    今天我们来探索一下,如何升级智能,通常我们把程序分解成逻辑和数据结构。所以我们也把智能合约分解成逻辑合约和数据合约,这样一旦出现问题,我们可以只升级逻辑合约就可以了,而数据合约可以继续时使用。而更进一步,为了扩展和灵活性,我们可以设置一个接口合约,专门负责智能合约的入口,保持一直不变,而逻辑合约和数据合约可以更加需要进行添加、删除和更新。

今天我们只实现一个简单的升级方式,把一个合约分解成逻辑合约和数据合约;暂时不考虑接口合约。例如我们有一个如下合约。

pragma solidity ^0.4.24;

contract Balance {
    mapping (address => uint256) public balance;
    function set(address addr,uint256 n) public {
        balance[addr] = n;
    }
    function add(address addr,uint256 n) public {
        balance[addr] = balance[addr]+n;
    }
}

这是一个存储余额的合约,我们可以把它分成两部分,控制余额操作的逻辑合约和控制数据数据合约;

pragma solidity ^0.4.24;

contract Data {
    mapping (address => uint256) public balance;
    function set(address addr,uint256 n) public {
        balance[addr] = n;
    }
    function get(address addr) public returns(uint){
        return balance[addr];
    }
    function add(address addr,uint256 n) public {
        balance[addr] = balance[addr]+n;
    }
}

contract Logic {
    Data data;
    uint256 balance;
    function Logic(address addr) public {
        data = Data(addr);
    }
    function Add(address addr,uint256 n) public {
        data.add(addr,n);
    }
    function Set(address addr, uint256 n) public {
        data.set(addr,n);
    }
    function Get(address addr) public returns (uint){
        balance = data.get(addr);
        return balance;
    }
}

可以看到我们Logic合约初始化的时候是需要传入数据合约地址的,所以部署的时候我们需要先部署数据合约,在部署逻辑合约。当我们要升级Logic的时候,我们只需要在新Logic初始化的时候将旧的数据合约写入即可。这样就保证了合约升级。

其实这样还是有一些问题,例如升级后合约地址就变化了,查询操作也需要消耗gas,数据的安全性访问等问题;这些我们将会在下一回解决。

猜你喜欢

转载自blog.csdn.net/xq723310/article/details/84844720