务必注意:solidity0.5.0以上的版本发生了很大的变化,具体的规则以官方最新版为准!!!
transfer、send、call都是可以合约之间相互转账的方法,但是用法有很大的不同!
//如果异常会转账失败,抛出异常(等价于requi(send()))(合约地址转账)
// 有gas限制,最大2300
<address payable>.transfer(uint256 amount)
//如果异常会转账失败,仅会返回false,不会终止执行(合约地址转账)
// 有gas限制,最大2300
<address payable>.send(uint256 amount) returns (bool)
如果异常会转账失败,仅会返回false,不会终止执行(调用合约的方法并转账)
// 没有gas限制
<address>.call(bytes memory) returns (bool, bytes memory)
三个方法的共同点:addr.transfer(1 ether)、addr.send(1 ether)、addr.call.value(1 ether)的接收方都是addr(这一点和我们传统的语言不一样,很容易混淆)
如果使用addr.transfer(1 ether)、addr.send(1 ether),addr合约中必须增加fallback回退函数!
如果使用addr.call.value(1 ether),那么被调用的方法必须添加payable修饰符,否则转账失败!
pragma solidity ^0.5.0;
contract Hello {
address adds;
# 如果使用transfer或send函数必须添加fallback回退函数
function () external payable {
}
function get_balance() public view returns (uint){
return address(this).balance;
}
# 如果使用call.value(1 ether)方法,则被调用的方法必须增加payable修饰符,否则转账会失败!
function set_sender() public payable{
adds = msg.sender;
}
function get_sender() public view returns(address) {
return adds;
}
}
contract Test {
constructor() public payable{
}
function get_balance() public view returns (uint){
return address(this).balance;
}
function send_call(address payable a) public {
# 获得被调用的方法的签名(也就是函数的selector属性)
# 方法签名的原理是:对给定参数进行编码,并以给定的函数选择器作为起始的 4 字节数据一起返回。详情参考官方文档abi编码函数!
# 如果有参数:bytes memory payload = abi.encodeWithSignature("set_sender(string)","参数");后面"参数"就是带的参数
bytes memory payload = abi.encodeWithSignature("set_sender()");
(bool success, bytes memory returnData) = a.call.value(0.1 ether)(payload);
require(success);
}
function send_transfer(address payable a) public {
//这里是被调用的方法签名(0.5.0以上的版本和之前的不一样,查看官方最新文档)
a.transfer(1 ether);
}
}