在Solidity中,优化gas使用是非常重要的,因为每一笔交易都需要消耗gas。以下是一些可以帮助你优化gas使用的技巧:
•使用更小的数据类型:更小的数据类型消耗更少的gas。例如,使用uint8
代替uint256
可以节省gas。•使用calldata
代替memory
:在Solidity中,calldata
是一个非修改的、只读的数据存储位置,用于函数参数。它只在外部函数中可用,也就是说,只有被external
修饰符修饰的函数才能使用calldata
。与memory
相比,calldata
是在EVM的执行环境中,不需要从存储或内存中读取数据。•避免在循环中进行昂贵的操作:在循环中进行昂贵的操作(如调用外部合约或写入状态变量)会消耗大量的gas。如果可能,尽量在循环外部进行这些操作。
// 不推荐
for (uint i = 0; i < array.length; i ) {
array[i] = someExternalContract.calculate(i);
}
// 推荐
uint[] memory results = someExternalContract.calculateAll(array.length);
for (uint i = 0; i < array.length; i ) {
array[i] = results[i];
}
•使用事件而不是存储数据:如果你只需要在链外访问数据,那么使用事件比存储数据更加节省gas。事件的gas成本比存储数据的gas成本要低得多。
代码语言:javascript复制 // 不推荐
uint public lastUpdated;
function update() public {
lastUpdated = now;
}
// 推荐
event Updated(uint timestamp);
function update() public {
emit Updated(now);
}
•删除不需要的数据:当你不再需要某些数据时,使用delete
关键字删除它们可以返还gas。
mapping(address => uint) public balances;
function burn(address user, uint amount) public {
require(balances[user] >= amount, "Insufficient balance");
balances[user] -= amount;
if (balances[user] == 0) {
delete balances[user];
}
}
•避免冗余的存储数据:如果需要多次读取同一个存储变量,考虑将它存储在内存变量中。
代码语言:javascript复制 // 不推荐
function calculate() public {
uint a = balances[msg.sender];
uint b = balances[msg.sender];
uint c = a b;
}
// 推荐
function calculate() public {
uint balance = balances[msg.sender];
uint c = balance balance;
}
•使用库函数:在Solidity中,库函数可以帮助我们重用代码并优化gas消耗。这是因为库函数在EVM级别上被视为内联函数,所以它们通常比在合约中直接实现相同逻辑的函数更加节省gas。
代码语言:javascript复制 // 定义一个库 SafeMath
library SafeMath {
function mul(uint a, uint b) internal pure returns (uint) {
if (a == 0) {
return 0;
}
uint c = a * b;
require(c / a == b, "Multiplication overflow");
return c;
}
}
contract MyContract {
using SafeMath for uint;
uint public value;
function multiply(uint amount) public {
value = value.mul(amount); // 使用SafeMath库的mul函数
}
}
•使用静态调用:如果你只需要读取其他合约的数据,那么使用staticcall
比使用call
更加节省gas,因为staticcall
不会改变状态。•staticcall
比call
更加节省gas,原因有两点:•安全性:由于staticcall
不能修改状态,因此它不会引发复杂的状态变化,也就不会消耗大量的gas。另一方面,call
可以修改状态,因此它可能会引发复杂的状态变化,消耗大量的gas。•简单性:staticcall
只需要读取数据,因此它的计算量较小,消耗的gas也较少。另一方面,call
可以执行任何操作,包括计算密集型的操作,因此它可能会消耗大量的gas。•使用immutable
和constant
:如果有一个在合约生命周期内不会改变的值,那可以使用immutable
或constant
,这可以节省gas。
// 不推荐
address public owner;
constructor() {
owner = msg.sender;
}
// 推荐
address public immutable owner;
constructor() {
owner = msg.sender;
}
•优化数组:如果合约中使用了数组,可以考虑使用mapping来优化数组操作。
代码语言:javascript复制 // 不推荐
uint[] public array;
function remove(uint index) public {
for (uint i = index; i<array.length-1; i ){
array[i] = array[i 1];
}
array.pop();
}
// 推荐
mapping(uint => uint) public map;
uint public size = 0;
function remove(uint index) public {
map[index] = map[size-1];
delete map[size-1];
size--;
}
•使用require
而不是assert
:require
和assert
都可以用来检查条件,但require
在失败时会返还剩余的gas,而assert
则不会。
// 不推荐
function transfer(address to, uint amount) public {
assert(balances[msg.sender] >= amount);
balances[msg.sender] -= amount;
balances[to] = amount;
}
// 推荐
function transfer(address to, uint amount) public {
require(balances[msg.sender] >= amount, "Insufficient balance");
balances[msg.sender] -= amount;
balances[to] = amount;
}
声明:本作品采用署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0)[1]进行许可,使用时请注明出处。 Author: mengbin[2] blog: mengbin[3] Github: mengbin92[4] cnblogs: 恋水无意[5] 腾讯云开发者社区:孟斯特[6]
References
[1]
署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0): https://creativecommons.org/licenses/by-nc-sa/4.0/deed.zh
[2]
mengbin: mengbin1992@outlook.com
[3]
mengbin: https://mengbin.top
[4]
mengbin92: https://mengbin92.github.io/
[5]
恋水无意: https://www.cnblogs.com/lianshuiwuyi/
[6]
孟斯特: https://cloud.tencent.com/developer/user/6649301