solidity-6.继承

继承(Inheritance)

继承的本质:
继承的实现方案是代码拷贝,所以合约继承后,部署到网络时,将变成一个合约。代码将从父类拷贝到子类中。

继承的定义

继承使用is,一个合约可以继承多个合约,用逗号分开。
如果继承的合约之间也有父子关系,那么合约要按照先父到子的顺序排序。
比如:

contract X {}
contract A is X {}
contract C is X, A {} // 先X,再A

继承的可见性

子合约不能访问父合约的private私有成员
子合约可以访问父合约所有的非私有成员(包括internal的函数和状态变量)

父合约传参

继承的子合约,必须提供父合约构造方法需要的所有参数,有两种方式实现,如下:

   contract Base {
        uint x;
        function Base(uint _x) { x = _x; }
    }
    contract Derived is Base(7) { //方式一
        //方式二
        function Derived(uint _y) Base(_y * _y) {
        }
    }

继承中的重名

当继承多个合约时,这些父合约中不允许出现相同的函数名,事件名,修改器名,或互相重名。

pragma solidity ^0.4.0;

contract Base1{
  address owner;
  modifier ownd(){
    if(msg.sender == owner) _;
  }

  event dupEvent(address, bytes);

  function dupFunc(){
    dupEvent(msg.sender, msg.data);
  }
}

contract Base2{
  address owner;
  modifier ownd(){
    if(msg.sender == owner) _;
  }

  event dupEvent(address, bytes);

  function dupFunc(){
    dupEvent(msg.sender, msg.data);
  }
}
//失败,将会报错 Identifier already declared
//contract DuplicateNames is Base1, Base2{}
  contract DuplicateNames is Base1{}

还有一种比较隐蔽的情况,默认状态变量的getter函数导致的重名

重写

在子类中允许重写函数,但不允许重写返回参数签名,一起来看看下面的代码:

contract Base{
  function data() returns(uint){
    return 1;
  }
}

contract InheritOverride is Base{
  function data(uint){}
  function data() returns(uint){}
  //Override changes extended function signature
  //function data() returns(string){}
}

上面代码中的function data() returns(string){}将导致Override changes extended function signature报错,因为不能修改返回签名。

抽象(Abstract Contracts)

抽象函数是没有函数体的函数
抽象合约是包含了抽象函数的合约,比如

contract Feline {
    function utterance() returns (bytes32);
}

这样的合约不能通过编译,即使合约内也包含一些正常的函数。但它们可以做为基合约被继承。

接口

接口与抽象合约类似,与之不同的是,接口内没有任何函数是已实现的,同时还有如下限制:

  • 不能继承其它合约,或接口。
  • 不能定义构造器
  • 不能定义变量
  • 不能定义结构体
  • 不能定义枚举类
    其中的一些限制可能在未来放开。

接口基本上限制为合约ABI定义可以表示的内容,ABI和接口定义之间的转换应该是可能的,不会有任何信息丢失。

接口用自己的关键词表示:

interface Token {
    function transfer(address recipient, uint amount);
}

问题

1.a继承b,c继承b,是不是一定要写 constract c is a, b {}
2.下面的案例Final继承的Base1和Base2有重复的kill函数,没有问题吗
3.下面的案例中,super.kill()和mortal.kill()有什么差别

pragma solidity ^0.4.0;

contract owned {
    function owned() public { owner = msg.sender; }
    address owner;
}

contract mortal is owned {
    function kill() public {
        if (msg.sender == owner) selfdestruct(owner);
    }
}

contract Base1 is mortal {
    function kill() public { /* do cleanup 1 */ super.kill(); }
}


contract Base2 is mortal {
    function kill() public { /* do cleanup 2 */ super.kill(); }
}

contract Final is Base1, Base2 {
}
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 原文链接date:20170710 Solidity中合约的概念和其他面向对象语言中的类差不多。他们都有状态变量来...
    gaoer1938阅读 1,012评论 0 0
  • 不明白为什么,给张爱玲的小说作评论就非得提到胡兰成这个过客,仿佛没有胡兰成的张爱玲就不是张爱玲似的。在学界眼里永远...
    山鲁佑德阅读 386评论 0 3
  • 由于项目要求,最近学习了一个被称为 “Android第一神器” 的东西——Xposed,我们看下百度介绍:Xpo...
    leonseven阅读 1,906评论 0 4
  • 就哈哈哈哈 不知道怎么滴,就哈哈就完了,估计得兴奋好几天,嘻嘻, 怎么描述我的心情呢,就幸福来的很突然,猝不及防,...
    翟七七阅读 191评论 2 0
  • 2017年11月2日 星期四 多云 今天有点忙, 下午放学的时候老公去接的儿子 。 由于今天上社团活动五点...
    王韶泽妈妈阅读 203评论 0 0