深度解析Solidity合约调用call,callcode,及delegatecall函数

三个方法都是用来进行合约交互的方法。由于没有进行更进一步的封装,不是最好的选择,一般不会直接使用到它们;另外一个显著的问题由于可以使用任意参数类型,在语言层面不能保证类型安全,所以不推荐使用。

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);
    }
    
}

下图实际操作演示。


test

可以看到,由于没有找到对应的函数调用,最终调用的是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可以看看操作演示:


test1

函数的结果

call()的返回结果是一个bool,表示是否成功的调用,或者是失败引起了EVM异常。该方法无法直接访问函数返回结果(因为需要事前知道编码和返回结果大小)。

call()的返回结果即使成功,并不能说操作成功了,只是没有出现异常,比如我们第一个例子中,实际是调用到了fallback()函数。

delegatecall()

calldelegatecall的功能类似,区别仅在于后者仅使用给定地址的代码,其它信息则使用当前合约(如存储,余额等等)。

函数的设计目的是为了使用存储在另一个合约的库代码。

所以开发者在提供这样的库时,就要如何安排存储来达到这样的目的。

参考资料

关于作者

专注基于以太坊的相关区块链技术,了解以太坊,Solidity,Truffle。
博客:http://me.tryblockchain.org


  1. 关于这个的详细说明,可以参考这里。http://ethereum.stackexchange.com/questions/8168/understanding-namereg-callregister-myname-style-call-between-contracts

  2. 类似构造函数的定义方式。

  3. 参数编码格式与ABI的编码格式一致,直接参考ABI。

  4. 详细了解ABI格式,可以参考:【文档翻译系列】ABI详解

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

推荐阅读更多精彩内容

  • 原文地址:C语言函数调用栈(一)C语言函数调用栈(二) 0 引言 程序的执行过程可看作连续的函数调用。当一个函数执...
    小猪啊呜阅读 4,725评论 1 19
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,075评论 19 139
  • 本文翻译自:https://github.com/ConsenSys/smart-contract-best-pr...
    tolak阅读 5,035评论 4 21
  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,272评论 0 4
  • 今天被一个群里一个发R语言的生信小哥哥炸了回来。 自从考研,我就退出简书一大段时间了,期间还有些原因是因为首页那些...
    邱天阅读 149评论 0 0