区块链 > 正文

以太坊如何优化存储以节省空间并降低燃气费

2019-07-29 17:22:44来源:

为了解决这个问题,让我们深入了解以太坊如何优化数据存储。但首先,请确保您知道如何读取区块链上的存储空间。

以太坊如何优化数据存储

从Solidity文档中,我们得到了这个定义:

静态大小的变量(除了映射和动态大小的数组类型之外的所有内容)在从位置0开始的存储中连续布局。如果可能,需要少于32个字节的多个项目被打包到单个存储槽中,根据以下规则

以下是低效存储使用的示例。请注意较小的大小变量(如boolVar和bytes4Var)不是按顺序初始化的,当它们可以打包在一起时会占用新的插槽0和2:

更有效的存储方法是按顺序声明bool(1字节大小)和bytes4(4字节大小)变量。然后,EVM将这两个模块有效地打包到一个存储插槽中。

同样,在Object结构中,更有效的方法是将两个uint8组合在一起,占用1个插槽。这样,Object的所有未来实例只需要存储2个插槽,而不是3个插槽。存储优化在结构中尤其重要,因为存储可以快速增长:

注意:插槽索引为0从右到左。Bytes4Var在boolVar之后被初始化,所以它存储在boolVar的左边,正好是1个字节。

例外情况:

1、常量不存储在存储器中。从以太坊文档中,编译器不为常量变量保留存储槽。这意味着您将无法在任何存储槽中找到以下内容:

contract A {

uint public constant number = ...; //not stored in storage

}

2、映射和动态大小的数组不遵循这些约定。稍后将详细介绍这一点。

你现在有能力解决这个问题!

详情演练

要解决这个级别的问题,您必须找出数据[2]中存储的内容,将其转换为bytes16变量,并将其作为unlock()privacy.sol的密钥提交。

0.注意privacy.sol中的以下变量声明。让我们来计算一下这些占用的存储插槽:

// boolean values take up 1 byte

bool public locked = true;

// IGNORE: as constant uints are not stored in storage

uint256 public constant ID = block.timestamp;

// uint8 vars take up 1 byte

uint8 private flattening = 10;

uint8 private denomination = 255;

// uint16 takes up 2 bytes

uint16 private awkwardness = uint16(now);

// each bytes32 takes up one slot

// so an array of length 3 takes up are 3 slots

bytes32[3] private data;

你应该期望unlock、flattening和denomination三个字段一共占用5个字节,仅共享1个存储槽(备用27个字节的空闲存储空间)。

您应该期望data阵列占用3个剩余的插槽,每个数据有一个data[index]。

1、要获取数据[2],请读取插槽4的存储:

在Truffle控制台-Network Ropsten中,访问您的级别实例,并调用GetStorageAt(…,3)获取数据[2]。

2、使用Remix,将bytes32结果转换为bytes16值。

3、使用Remix,使用bytes16值调用unlock()来解锁此级别!

关键安全要素:

· 通常,过多的插槽使用会浪费gas,特别是如果您声明了将复制许多实例的结构。请记住优化存储以节省gas!

· 如果您不需要保持智能合约状态,请将变量保存到memory中。SSTORE <> SLOAD是非常耗气的操作码。

· 所有存储在区块链上都是公开可见的,甚至是您的private变量!

· 不要在没有哈希的情况下存储密码和私钥

相关阅读

  • 电影花絮
  • 电影情报
  • 圈子新闻
  • 电影新闻
  • 电影搜罗
  • 电视剧
  • 影视演员
推荐阅读