Solidity07 常数 constant和immutable

这一个文档,我们介绍Solidity中和常量相关的两个关键字,constant(常量)和immutable(不变量)。状态变量声明这两个关键字之后,不能在初始化后更改数值。这样做的好处是提升合约的安全性并节省gas

另外,只有数值变量可以声明constantimmutablestringbytes可以声明为constant,但不能为immutable

一、常量的定义

在智能合约中,如果一个状态变量的值恒定不变,就可以使用关键字 constant 或者 immutable 进行修饰,把它定义为常量。

string constant SYMBOL = "WETH";
uint256 immutable TOTAL_SUPPLY = 1000;

常量的命名规则与变量相同,但通常使用大写字母表示,单词之间用下划线 "_" 连接,这样与变量更容易区分。这不是强制性的规定,而是大家约定俗成的编程规范。

状态变量一旦声明为 constantimmutable 后,就不能更改它的值了。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

contract ConstType { 
  string constant SYMBOL = "WETH"; // 定义常量 SYMBOL
 
  function getSymbol() public view returns(string memory) {
    return SYMBOL; // 可以读取常量 SYMBOL 的值
  }
  
  function setSymbol(string memory symbol) public {
    SYMBOL = symbol; // 编译时报错,不能修改常量 SYMBOL 的值
  }
}

我们将以上合约代码复制到 Remix,编译时就会报错,错误信息为:不能给一个常量重新赋值。

二、constant 和 immutable 的区别

constantimmutable 虽然都能限制对状态变量的修改,但两者还是有区别的,表现在两个方面:

2.1 初始化时机

constant 关键字修饰的状态变量,必须在声明时就立即显式赋值,然后就不再允许修改了。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

contract ConstType {
    uint8  public constant DECIMALS = 18; // 声明时赋值
}

immutable 关键字修饰的状态变量,既可以在声明时显式赋值,还可以在合约部署时赋值,也就是在合约的构造函数 constructor 中赋值。但是,一旦赋值后就不能再修改了。所以,immutableconstant 多了一个设置初值的机会,但是它花费的 gas 费要高一点。

比如,在 ERC20 代币合约中,通常将铸造总量 totalSupply 定义为 immutable,这样可以在合约部署时,由部署者按照实际需求填入数量,不用再去修改代码,从而提高了合约的通用性和灵活性。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

contract ConstType {
    uint8  public immutable DECIMALS = 18; // 声明时赋值
    uint256 public immutable TOTAL_SUPPLY;

     constructor() {
        TOTAL_SUPPLY = 1000; // 合约部署时赋值
    }
}

2.2 适用的数据类型

constant 可以修饰任何数据类型。

immutable 只能修饰值类型,比如:int、uint、bool、address 等,不能修饰 string、bytes 等引用类型。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

contract ConstType {
    uint256 constant TOTAL_SUPPLY = 100; // 正确
    uint256 immutable TOTAL_SUPPLY_I = 100; // 正确
    string constant SYMBOL = "WETH"; // 正确
    string immutable SYMBOL_I = "WETH"; // 错误
}

三、使用常量的好处

  1. 代码可读性

通过使用常量,可以为代码中使用的特定数值或字符串提供有意义的名称,增加代码的可读性,使代码更易于理解和维护。

  1. 代码重用

通过将常量定义为可复用的值,避免了在代码中重复输入。如果需要更改常量的值,只需修改它的定义,而不必在整个代码中寻找和替换多个出现的具体值。

  1. 预防错误

常量的值在编译时就被确定,并且在运行时不可修改。这可以帮助预防意外的值更改和错误的赋值操作,减少人为错误和意外行为。

  1. 节省 Gas

使用常量比变量更节省 gas 成本,这也是非常重要的一点。常量的值在编译时就已知,且不可改变,编译器会将其值直接嵌入到合约代码中,避免了在运行时进行存储和访问的开销。

四、代码示例

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.21;
contract Constant {
    // constant变量必须在声明的时候初始化,之后不能改变
    uint256 public constant CONSTANT_NUM = 10;
    string public constant CONSTANT_STRING = "0xAA";
    bytes public constant CONSTANT_BYTES = "LJ";
    address public constant CONSTANT_ADDRESS = 0x0000000000000000000000000000000000000000;

    // immutable变量可以在constructor里初始化,之后不能改变
    uint256 public immutable IMMUTABLE_NUM = 9999999999;
    address public immutable IMMUTABLE_ADDRESS;
    uint256 public immutable IMMUTABLE_BLOCK;
    uint256 public immutable IMMUTABLE_TEST;

    // 利用constructor初始化immutable变量,因此可以利用
    constructor(){
        IMMUTABLE_ADDRESS = address(this);
        IMMUTABLE_NUM = 1118;
        IMMUTABLE_TEST = test();
    }

    function test() public pure returns(uint256){
        uint256 what = 9;
        return(what);
    }
}
  • 部署好合约之后,通过remix上的getter函数,能获取到constantimmutable变量初始化好的值。
  • constant变量初始化之后,尝试改变它的值,会编译不通过并抛出TypeError: Cannot assign to a constant variable.的错误。

  • immutable变量初始化之后,尝试改变它的值,会编译不通过并抛出TypeError: Immutable state variable already initialized.的错误。

这一章,我们介绍了Solidity中两个关键字,constant(常量)和immutable(不变量),让不应该变的变量保持不变。这样的做法能在节省gas的同时提升合约的安全性。

相关推荐
C7211BA2 小时前
比特币中的相关技术
区块链
软件工程小施同学11 小时前
【最新区块链论文录用资讯】CCF A—NDSS 2025 (二) 附pdf下载
pdf·区块链
#咸菜2 天前
RuleOS:区块链开发的“新引擎”,点燃Web3创新之火
区块链
一面千人2 天前
Avalanche公链深度解析:创新共识、亚秒级最终性与生态竞争力
区块链·arbitrum·公链·polygon·solana·亚稳态·sui·aptos·dag·bsc·optimism·avalanche·雪崩共识·亚秒级·最终性·雪崩效应
元宇宙时间2 天前
TON基金会确认冠名赞助2025香港Web3嘉年华,并将于4月8日重磅呈现“TON生态日”
人工智能·去中心化·区块链·gpu算力
#咸菜2 天前
RuleOS:区块链开发的“破局者”,开启Web3新纪元
web3·区块链
qiquandongkh2 天前
期权帮|中证1000股指期权交割结算价怎么算?
大数据·金融·区块链
#咸菜2 天前
RuleOS:DApp开发的“破局者”,区块链创新的“加速器
区块链
加密新世界2 天前
如何在 Uniswap V4 上创建 Hook
区块链·智能合约
worxfr2 天前
利率债、信用债、可转债区别与优势
区块链