当调用其它合约的函数时,可以通过选项.value()
,和.gas()
来分别指定,要发送的ether
量(以wei
为单位),和gas
值。
pragma solidity ^0.4.0; contract InfoFeed { function info() payable returns (uint ret) { return msg.value; } } contract Consumer { function deposit() payable returns (uint){ return msg.value; } function left() constant returns (uint){ return this.balance; } function callFeed(address addr) returns (uint) { return InfoFeed(addr).info.value(1).gas(8000)(); } }
上面的代码中,我们首先调用deposit()
为Consumer
合约存入一定量的ether
。然后调用callFeed()
通过value(1)
的方式,向InfoFeed
合约的info()
函数发送1ether
。需要注意的是,如果不先充值,由于合约余额为0,余额不足会报错Invalid opcode
1。
InfoFeed.info()
函数,必须使用payable
关键字,否则不能通过value()
选项来接收ether
。
代码InfoFeed(addr)
进行了一个显示的类型转换,声明了我们确定知道给定的地址是InfoFeed
类型。所以这里并不会执行构造器的初始化。显示的类型强制转换,需要极度小心,不要尝试调用一个你不知道类型的合约。
我们也可以使用function setFeed(InfoFeed _feed) { feed = _feed; }
来直接进行赋值。.info.value(1).gas(8000)
只是本地设置发送的数额和gas值,真正执行调用的是其后的括号.info.value(1).gas(8000)()
。
如果被调用的合约不存在,或者是不包代码的帐户,或调用的合约产生了异常,或者gas不足,均会造成函数调用发生异常。
如果被调用的合约源码并不事前知道,和它们交互会有潜在的风险。当前合约会将自己的控制权交给被调用的合约,而对方几乎可以做任何事。即使被调用的合约是继承自一个已知的父合约,但继承的子合约仅仅被要求正确实现了接口。合约的实现,可以是任意的内容,由此会有风险。另外,准备好处理调用你自己系统中的其它合约,可能在第一调用结果未返回之前就返回了调用的合约。某种程度上意味着,被调用的合约可以改变调用合约的状态变量(state variable)
来标记当前的状态。如,写一个函数,只有当状态变量(state variables)
的值有对应的改变时,才调用外部函数,这样你的合约就不会有可重入性漏洞。