想知道更多关于区块链技术知识,请百度【链客区块链技术问答社区】 链客,有问必答!
下面来继续介绍做为一个分布式网络语言所特有的internal和external这两种不一样的函数调用方式,以及Solidity提供的对函数调用时的可见性控制语法。
1、 调用方式
Solidity封装了两种函数的调用方式internal和external。
internal
internal调用,实现时转为简单的EVM跳转,因此它能直接使用上下文环境中的数据,对于引用传递时将会变得很是高效(不用拷贝数据)。
在当前的代码单元内,如对合约内函数,引入的库函数,以及父类合约中的函数直接使用便是以internal方式的调用。咱们来看个简单的例子:
pragma solidity ^0.4.0;数组
contract Test {网络
function f(){} //以`internal`的方式调用 function callInternally(){ f(); }
}
在上述代码中,callInternally()以internal的方式对f()函数进行了调用。
external
external调用,实现为合约的外部消息调用。因此在合约初始化时不能external的方式调用自身函数,由于合约还未初始化完成。下面来看一个以external方式调用的例子:
pragma solidity ^0.4.0;分布式
contract A{ide
function f(){}
}函数
contract B{性能
//以`external`的方式调用另外一合约中的函数 function callExternal(A a){ a.f(); }
}
虽然当前合约A和B的代码放在一块儿,但部署到网络上后,它们是两个彻底独立的合约,它们之间的方法调用是经过消息调用。上述代码中,在合约B中的callExternal()以external的方式调用了合约A的f()。
external调用时,实际是向目标合约发送一个消息调用。消息中的函数定义部分是一个24字节大小的消息体,20字节为地址,4字节为函数签名。
this
咱们能够在合约的调用函数前加this.来强制以external方式的调用。须要注意的是这里的this的用法与大多数语言的都不一致。
pragma solidity ^0.4.0;区块链
contract A{this
function f() internal{} function callInternally(){ f(); } //以`external`的方式调用 //f()只能以`internal`的方式调用 //Untitled3:7:9: Error: Member "f" not found or not visible after argument-dependent lookup in contract A function callExternally(){ //this.f(); }
}
调用方式说明
上面所提到的internal和external指的函数调用方式,请不要与后面的函数可见性声明的external,public,internal,private弄混。声明只是意味着这个函数须要使用相对应的调用方式去调用。后续说明中会用以某某方式调用,来强调是对调用方式的阐述以加以区分。
2、函数的可见性
Solidity为函数提供了四种可见性,external,public,internal,private。
external
声明为external的能够从其它合约或经过Transaction进行调用,因此声明为external的函数是合约对外接口的一部分。
不能以internal的方式进行调用。
有时在接收大的数据数组时性能更好。
pragma solidity ^0.4.5;spa
contract FuntionTest{code
function externalFunc() external{} function callFunc(){ //以`internal`的方式调用函数报错 //Error: Undeclared identifier. //externalFunc(); //以`external`的方式调用函数 this.externalFunc(); }
}
声明为external的externalFunc()只能以external的方式进行调用,以internal的方式调用会报Error: Undeclared identifier.。
public
函数默认声明为public。
public的函数既容许以internal的方式调用,也容许以external的方式调用。
public的函数因为被外部合约访问,是合约对外接口的一部分。
pragma solidity ^0.4.5;
contract FuntionTest{
//默认是public函数 function publicFunc(){} function callFunc(){ //以`internal`的方式调用函数 publicFunc(); //以`external`的方式调用函数 this.publicFunc(); }
}
咱们能够看到声明为public的publicFunc()容许两种调用方式。
internal
在当前的合约或继承的合约中,只容许以internal的方式调用。
pragma solidity ^0.4.5;
contract A{
//默认是public函数 function internalFunc() internal{} function callFunc(){ //以`internal`的方式调用函数 internalFunc(); }
}
contract B is A{
//子合约中调用 function callFunc(){ internalFunc(); }
}
上述例子中声明为internal的internalFunc()在定义合约,和子合约中均只能以internal的方式能够进行调用。
private
只能在当前合约中被访问(不可在被继承的合约中访问)。
即便声明为private,仍能被全部人查看到里面的数据。访问权限只是阻止了其它合约访问函数或修改数据。
pragma solidity ^0.4.5;
contract A{
//默认是public函数 function privateFunc() private{} function callFunc(){ //以`internal`的方式调用函数 privateFunc(); }
}
contract B is A{
//不可调用`private` function callFunc(){ //privateFunc(); }
}上述例子中,声明为private的privateFunc()只能在定义的合约中以internal的方式进行调用。