三个方法都是用来进行合约交互的方法。由于没有进行更进一步的封装,不是最好的选择,一般不会直接使用到它们;另外一个显著的问题由于可以使用任意参数类型,在语言层面不能保证类型安全,所以不推荐使用。
call() 方法
call()
是一个底层的接口,用来向一个合约发送消息[1],也就是说如果你想实现自己的消息传递,可以使用这个函数。函数支持传入任意类型的任意参数,并将参数打包成32字节,相互拼接后向合约发送这段数据。
函数的传输的数据
由于向另一个合约发送数据时,找不到对应的方法签名,会默认调用fallback()
函数[2],所以我们可以通过这个来看看call()
传的具体数据。
pragma solidity ^0.4.0;
contract Person{
bytes fail;
function(){
fail = msg.data;
}
function getFail() returns (bytes){
return fail;
}
}
contract CallTest{
function callData(address addr) returns (bool){
return addr.call("abc", 256);
}
}
下图实际操作演示。
可以看到,由于没有找到对应的函数调用,最终调用的是fallback()
函数,通过fail
字段,我们看到了收到msg.data
:
0x61626300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000100
前32字节为abc
对应的acii编码值,后32位为256
的对应编码值[3]。
call指定函数
如果第一个参数刚好是四个字节,会认为这四个字节指定的是函数签名的序号值,生成方式参见ABI协议的函数选择器[4]。由如果你只是想传个参数值,而不是想指定一个函数序号,应避免第一个参数刚好是四个字节。
pragma solidity ^0.4.0;
contract Person{
uint age = 10;
function increaseAge(string name, uint num) returns (uint){
return ++age;
}
function getAge() returns (uint){
return age;
}
}
contract CallTest{
function callByFun(address addr)returns (bool){
bytes4 methodId = bytes4(keccak256("increaseAge(string,uint256)"));
return addr.call(methodId,"jack", 1);
}
}
通过下图的gif可以看看操作演示:
函数的结果
call()
的返回结果是一个bool
,表示是否成功的调用,或者是失败引起了EVM异常。该方法无法直接访问函数返回结果(因为需要事前知道编码和返回结果大小)。
call()
的返回结果即使成功,并不能说操作成功了,只是没有出现异常,比如我们第一个例子中,实际是调用到了fallback()
函数。
delegatecall()
call
与delegatecall
的功能类似,区别仅在于后者仅使用给定地址的代码,其它信息则使用当前合约(如存储,余额等等)。
函数的设计目的是为了使用存储在另一个合约的库代码。
所以开发者在提供这样的库时,就要如何安排存储来达到这样的目的。
参考资料
关于作者
专注基于以太坊的相关区块链技术,了解以太坊,Solidity,Truffle。
博客:http://me.tryblockchain.org
-
关于这个的详细说明,可以参考这里。http://ethereum.stackexchange.com/questions/8168/understanding-namereg-callregister-myname-style-call-between-contracts ↩
-
类似构造函数的定义方式。 ↩
-
参数编码格式与ABI的编码格式一致,直接参考ABI。 ↩
-
详细了解ABI格式,可以参考:【文档翻译系列】ABI详解 ↩