如何在Solidity库中使用状态变量函数

区块链研究实验室 view 143 2020-8-20 21:42
share to
Scan QR code with WeChat

如何在Solidity库中使用状态变量函数

众所周知,Solidity库不能有状态变量。

如果今天你在网上快速搜索一下Solidity库是否可以有状态变量,你会发现答案是否定的,它们不能。

以下是关于库的Solidity文档:

如何在Solidity库中使用状态变量函数

注意第一个限制:库不能具有状态变量。

但文档将显示,可以通过这种方式将存储指针传递到库函数并访问状态变量。

但是如果您想在库中定义、创建和使用新的状态变量,并且在不将它们作为参数传递的情况下使用它们呢?

如果您想随时随地修改所需的合约存储,而又不传递存储指针怎么办?

使用Solidity库可以做这些事情吗?

从Solidity文档看,答案似乎是否定的。如果您像我一样在网上搜索如何执行此操作,那么您可能会发现答案是否定的,除非您当然找到了此博客文章。

所以我会说:

Solidity库可以有状态变量!

我讨厌与Solidity文档发生冲突,而且几乎所有在这一点上了解Solidity的人都是如此。

请注意库限制底部的小行:

(这些可能会在以后解除。)

好吧,库不能具有状态变量的第一个限制已于2020年3月10日解除,没有人注意到把。

将状态变量添加到库中不仅仅是一个很好的技术技巧。具有状态变量的库非常有用。

如何向库中添加状态变量

通过使用Diamond(方块)存储,库可以拥有/创建/使用/修改状态变量。

从Solidity 0.6.4开始,可以在合约存储的任意位置创建指向结构的指针。

那就是Diamond(方块)存储。报价来自Diamond(方块)标准的合约存储部分。Diamond(方块)标准和实施Diamond(方块)的人们一直在引领Diamond(方块)存储的使用。

为了更好地理解如何使用Diamond(方块)存储向库添加状态变量,请参见下面的示例。

带有状态变量的库示例

这是带有状态变量的库的简单示例。它是为了易于阅读和理解而编写的。它编译时没有错误或警告。

// SPDX-License-Identifier: MITpragma solidity ^0.7.0;// This library has the state variables 'contractAddress' and 'name'library Library { // defining state variables struct DiamondStorage { address contractAddress; string name; // ... any number of other state variables } // return a struct storage pointer for accessing the state variables function diamondStorage()  internal  pure  returns (DiamondStorage storage ds)  { bytes32 position = keccak256("diamond.standard.diamond.storage"); assembly { ds.slot := position } } // set state variables function setStateVariables( address _contractAddress,  string memory _name )  internal  { DiamondStorage storage ds = diamondStorage(); ds.contractAddress = _contractAddress; ds.name = _name; } // get contractAddress state variable function contractAddress() internal view returns (address) { return diamondStorage().contractAddress; } // get name state variable function name() internal view returns (string memory) { return diamondStorage().name; }}// This contract uses the library to set and retrieve state variables contract ContractA { function setState() external { Library.setStateVariables(address(this), "My Name"); } function getState()  external  view  returns (address contractAddress, string memory name)  { contractAddress = Library.contractAddress(); name = Library.name(); }}

请注意,库函数setStateVariables、contractAddress和name()是内部函数。这些内部函数将被添加到ContractA的字节码中,从而增加它的大小。但是内部函数调用比外部调用使用更少的气体,所以这很好。

可以将库函数设置为外部函数,它们仍将起作用。在这种情况下,它们不会被添加到ContractA的字节码中。它们将使用委托代码操作码从外部调用。库函数就是这样工作的。

请注意,不同的库将需要使用不同的存储插槽,因此使用不同的keccak256ed字符串。这是为了防止两个或多个库在合同存储中写入相同的位置。

btcfans公众号

Scan QR code with WeChat

Link
Disclaimer:

Tags: Solidity
Previous: Cortex技术教程 | Cortex 全节点 Next: 为Aave DeFi协议创建清算脚本

Related