【问题标题】:Gas required exceeds block gas limit fallback function所需气体超过块气体限制后备功能
【发布时间】:2017-12-24 17:45:02
【问题描述】:

我正在研究智能合约并在此处观看此视频:https://www.youtube.com/watch?v=s677QFT6e4U&t=911s。我完全复制了代码,但是当我尝试调用回退函数时,出现以下错误:Gas required exceeds block gas limit: 300000000。即使后备功能如下(它什么都不做):

function () payable {

}

这怎么可能用了太多的气体?

合同代码:

pragma solidity ^0.4.11;

import './IERC20.sol';
import './SafeMath.sol';

contract AToken is IERC20 {
    
    using SafeMath for uint256;
    
    uint256 public _totalSupply = 0;
    uint256 public constant hardLimit = 45000000;
    string public constant symbol = "ABC";
    string public constant name = "Alphabet";
    uint8 public constant decimals = 18;
    
    //1 ETH = 25000 Alphabet
    uint256 public constant RATE = 25000;
    
    address public owner;
    
    mapping(address => uint256) balances;
    mapping(address => mapping(address => uint256)) allowed;
    
    function () payable {
        createTokens();
    }
    
    function SnapToken() {
        owner = msg.sender;
    }
    
    function createTokens() payable {
        //require(msg.value > 0);
        //uint256 tokens = msg.value.mul(RATE);
        //require(tokens.add(_totalSupply) <= hardLimit);
        //balances[msg.sender] = balances[msg.sender].add(tokens);
        //_totalSupply = _totalSupply.add(tokens);
        //owner.transfer(msg.value);
    }
    
    function totalSupply() constant returns (uint256 totalSupply) {
        return _totalSupply;
    }
    
    function balanceOf(address _owner) constant returns (uint256 balance) {
        return balances[_owner];
    }
    
    function transfer(address _to, uint256 _value) returns (bool success) {
        require(balances[msg.sender] >= _value && _value > 0);
        balances[msg.sender] = balances[msg.sender].sub(_value);
        balances[_to] = balances[_to].add(_value);
        Transfer(msg.sender, _to, _value);
        return true;
    }
    
    function transferFrom(address _from, address _to, uint256 _value) returns (bool success) {
        require(allowed[_from][msg.sender] >= _value && balances[_from] >= _value && _value > 0);
        balances[_from] = balances[_from].sub(_value);
        balances[_to] = balances[_to].add(_value);
        allowed[_from][msg.sender] = allowed[_from][msg.sender].sub(_value);
        Transfer(_from, _to, _value);
        return true;
    }
    
    function approve(address _spender, uint256 _value) returns (bool success) {
        //allowed[msg.sender][_spender] = allowed[msg.sender][_spender].add(_value);
        Approval(msg.sender, _spender, _value);
        return true;
    }
    
    function allowance(address _owner, address _spender) constant returns (uint256 remaining) {
        return allowed[_owner][_spender];
    }
    event Transfer(address indexed _from, address indexed _to, uint256 _value);
    event Approval(address indexed _owner, address indexed _spender, uint256 value);
}

我注释掉了一些东西,看看这是否会减少气体需求,但不幸的是不会。你以前遇到过这种情况吗?

谢谢

【问题讨论】:

  • 请附上合同和您的客户代码。
  • @AdamKipnis,请找到随附的代码。

标签: ethereum solidity smartcontracts ether


【解决方案1】:

正如发布的那样,合约在调用回退函数时会执行(在 Remix 中测试)。但是,一旦您取消注释createTokens() 中的逻辑,它将失败

后备函数的气体限制较低 (2300),因此它们的功能非常有限。你不能做诸如写入存储、调用外部函数或发送以太之类的事情,因为你会立即达到极限。它应该主要用于使您的合约能够接收以太币并可能记录事件。

在您上面发布的示例中,删除回退函数中对 createTokens() 的调用,然后直接从您的客户端调用该函数。

Documentation on Fallback Functions

示例客户端代码:

const abiDefinition = ...;
const contractAddress = ...;
const account = ...;
const amountInEther = ...;

const contract = web3.eth.contract(abiDefinition);
const contractInstance = contract.at(contractAddress);

const transactionObj = {
  from: account,
  value: web3.toWei(amountInEther, 'ether'),
};

contractInstance.createTokens.sendTransaction(transactionObj, (error, result) = {
  ...
};

另外,作为旁注,您的价值计算不正确。 msg.value 在 Wei,而不是 ether。发送 1 个以太币会使您远远超过您的 hardlimit。建议在合同中与 Wei 合作,因此您应该调整您的RATE

【讨论】:

  • 感谢您的回答。我今晚再试一次。你会建议做什么而不是调用 createTokens?毕竟,当投资者向合约付款时,这就是我需要调用的函数。谢谢!
  • 您可以通过sendTransaction() 呼叫发送以太币。我将通过一个示例 JS 客户端调用来更新答案。
  • 视频使用的是过时版本的 Solidity,现在还有其他更高效的函数调用,再次感谢!
猜你喜欢
  • 1970-01-01
  • 2022-11-14
  • 1970-01-01
  • 1970-01-01
  • 2017-08-12
  • 1970-01-01
  • 1970-01-01
  • 2018-11-02
  • 1970-01-01
相关资源
最近更新 更多