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

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

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

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

转自 https://www.jianshu.com/p/fd5075ff0ab9

你可能感兴趣的:(solidity,solidity,智能合约)