美文网首页Blockchain区块链研究区块链实践专题
深度解析Solidity合约调用call,callcode,及d

深度解析Solidity合约调用call,callcode,及d

作者: TryBlockchain | 来源:发表于2017-03-19 21:48 被阅读2841次

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

    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详解

    相关文章

      网友评论

        本文标题:深度解析Solidity合约调用call,callcode,及d

        本文链接:https://www.haomeiwen.com/subject/txmvnttx.html