web3.js中estimateGas如何计算智能合约消耗的gas量

简介: 我们可使用web3.js框架的estimateGas函数获得一个以太坊智能合约的Gas估计值 ,通过执行一个消息调用或交易,该消息调用或交易直接在节点的VM中执行,并未在区块链中确认,函数会返回估算使用的gas量。

我们可使用web3.js框架的estimateGas函数获得一个以太坊智能合约的Gas估计值 ,通过执行一个消息调用或交易,该消息调用或交易直接在节点的VM中执行,并未在区块链中确认,函数会返回估算使用的gas量。

函数调用:

web3.eth.estimateGas(callObject [, callback])

参数:

在 web3.eth.sendTransaction 中, 参数大都是可选的。

1. Object - 要发送的交易对象:

  • fromString - 用来传送的账户地址. 默认使用web3.eth.defaultAccount属性。
  • toString - (可选) 目标地址,对于创建合同的交易没有定义。
  • valueNumber|String|BigNumber - (可选) 为交易转移的价值以Wei为单位,如果是合同创建交易,也是基金。
  • gasNumber|String|BigNumber - (可选, 默认: 待定) 用于交易的gas量(未使用的gas已退还)。
  • gasPriceNumber|String|BigNumber - (可选, 默认: 待定) 此交易的gas价格以wei为单位,默认为平均网络gas价格。
  • dataString - (可选) Either 包含消息关联数据的字节字符串,或者创建合同事务的初始化代码。
  • nonceNumber - (可选)一个随机数的整数。 这允许覆盖使用相同随机数的您自己的未决事务。

2.Function - (optional)如果传递回调,则HTTP请求将变为异步。详细说明在这里 this note .

返回值: 

数字:模拟呼叫/交易的需要使用的gas值。

一个简单示例:
var result = web3.eth.estimateGas({
    to: "0xc4abd0339eb8d57087278718986382264244252f", 
    data: "0xc6888fa10000000000000000000000000000000000000000000000000000000000000003"
});
console.log(result); // "0x0000000000000000000000000000000000000000000000000000000000000015"
 

用web3js库中可能遇到estimateGas方法出错的问题。大多数情况下得到的错误是这个:“所需的gas超过允许值或总是交易失败”。

首先要检查的下交易是否有效。例如,如果正在估计将一定数量的通证发送到另一个地址的gasAmount,那么最主要的检查两件事:

1. 发送地址中是否有足够的以太。
2. 发送地址中是否有足够的通证/代币。
这些似乎是显而易见要检查的,但是还是可能会犯这种低级错误,认为方法估计Gas只是用来计算估计值,其实不是。如果参数设置的实际条件不对,它在运行这个方法时不会真正执行任何代码就直接抛出错误。

评估发送通证的所需gas量的代码片段:

tokenContract.methods.transfer(recipientAddress,numtokens)
 .estimateGas({from:tokenHolderAddress},function(gasAmount){
 console.log(gasAmount);
 });

官网在这里https://github.com/ethereum/wiki/wiki/JavaScript-API#web3ethestimategas.

也可以在你的浏览器地址栏输入https://ethereum.github.io/browser-solidity,然后直接copy你的合约就可以获得估计值。

这个代码中默认的一个示例是提案投票的代码如下:

pragma solidity ^0.4.0;
contract Ballot {

    struct Voter {
        uint weight;
        bool voted;
        uint8 vote;
        address delegate;
    }
    struct Proposal {
        uint voteCount;
    }

    address chairperson;
    mapping(address => Voter) voters;
    Proposal[] proposals;

    /// Create a new ballot with $(_numProposals) different proposals. //为不同的提案创建一个新的投票合约
    function Ballot(uint8 _numProposals) public {
        chairperson = msg.sender;
        voters[chairperson].weight = 1;
        proposals.length = _numProposals;
    }

    /// Give $(toVoter) the right to vote on this ballot.//授予投票权
    /// May only be called by $(chairperson). //只能被主席调用
    function giveRightToVote(address toVoter) public {
        if (msg.sender != chairperson || voters[toVoter].voted) return;
        voters[toVoter].weight = 1;
    }

    /// Delegate your vote to the voter $(to).//委托你的投票权
    function delegate(address to) public {
        Voter storage sender = voters[msg.sender]; // assigns reference  指定参数
        if (sender.voted) return;
        while (voters[to].delegate != address(0) && voters[to].delegate != msg.sender)
            to = voters[to].delegate;
        if (to == msg.sender) return;
        sender.voted = true;
        sender.delegate = to;
        Voter storage delegateTo = voters[to];
        if (delegateTo.voted)
            proposals[delegateTo.vote].voteCount += sender.weight;
        else
            delegateTo.weight += sender.weight;
    }

    /// Give a single vote to proposal $(toProposal). //对某个提案投一票
    function vote(uint8 toProposal) public {
        Voter storage sender = voters[msg.sender];
        if (sender.voted || toProposal >= proposals.length) return;
        sender.voted = true;
        sender.vote = toProposal;
        proposals[toProposal].voteCount += sender.weight;
    }

    function winningProposal() public constant returns (uint8 _winningProposal) {
        uint256 winningVoteCount = 0;
        for (uint8 prop = 0; prop < proposals.length; prop++)
            if (proposals[prop].voteCount > winningVoteCount) {
                winningVoteCount = proposals[prop].voteCount;
                _winningProposal = prop;
            }
    }
}

可以run下试试。

如果对以太坊开发有兴趣,推荐两个教程:

1.适合区块链新手的以太坊DApp开发:

http://xc.hubwiz.com/course/5a952991adb3847553d205d1

2.用区块链、星际文件系统(IPFS)、Node.js和MongoDB来构建以太坊DApp电商平台:

http://xc.hubwiz.com/course/5abbb7acc02e6b6a59171dd6

如果想加入以太坊技术开发群可以加微信拉你入群。



相关文章
|
8月前
|
JavaScript 算法
原生JS完成“一对一、一对多”矩形DIV碰撞检测、碰撞检查,通过计算接触面积(重叠覆盖面积)大小来判断接触对象DOM
原生JS完成“一对一、一对多”矩形DIV碰撞检测、碰撞检查,通过计算接触面积(重叠覆盖面积)大小来判断接触对象DOM
|
8月前
|
JavaScript 前端开发 大数据
数字太大了,计算加法、减法会报错,结果不正确?怎么办?用JavaScript实现大数据(超过20位的数字)相加减运算。
数字太大了,计算加法、减法会报错,结果不正确?怎么办?用JavaScript实现大数据(超过20位的数字)相加减运算。
|
2月前
|
JavaScript 前端开发
|
4月前
|
JavaScript
js计算时间差,包括计算,天,时,分,秒
js计算时间差,包括计算,天,时,分,秒
314 16
|
3月前
|
缓存 JavaScript 前端开发
探索Vue.js中的计算属性与侦听器
【10月更文挑战第5天】探索Vue.js中的计算属性与侦听器
31 1
|
3月前
|
缓存 JavaScript 前端开发
深入理解Vue.js中的计算属性与侦听属性
【10月更文挑战第5天】深入理解Vue.js中的计算属性与侦听属性
39 0
|
3月前
|
缓存 JavaScript 前端开发
探索Vue.js中的计算属性与侦听器:深入理解与实践
【10月更文挑战第5天】探索Vue.js中的计算属性与侦听器:深入理解与实践
27 0
|
8月前
|
缓存 JavaScript 前端开发
Vue.js计算属性:实现数据驱动的利器
Vue.js计算属性:实现数据驱动的利器
|
5月前
|
前端开发 JavaScript 大数据
React与Web Workers:开启前端多线程时代的钥匙——深入探索计算密集型任务的优化策略与最佳实践
【8月更文挑战第31天】随着Web应用复杂性的提升,单线程JavaScript已难以胜任高计算量任务。Web Workers通过多线程编程解决了这一问题,使耗时任务独立运行而不阻塞主线程。结合React的组件化与虚拟DOM优势,可将大数据处理等任务交由Web Workers完成,确保UI流畅。最佳实践包括定义清晰接口、加强错误处理及合理评估任务特性。这一结合不仅提升了用户体验,更为前端开发带来多线程时代的全新可能。
119 1
|
5月前
|
数据可视化 NoSQL Serverless
现代化 Web 应用构建问题之Serverless架构的Web站点费用计算如何解决
现代化 Web 应用构建问题之Serverless架构的Web站点费用计算如何解决
54 1