Solidity03 Solidity变量简述

devtools/2025/1/22 4:31:25/

文章目录

  • 一、变量简述
    • 1.1 状态变量
    • 1.2 局部变量
    • 1.3 全局变量
    • 1.4 注意问题
  • 二、变量可见性
    • 2.1 public
    • 2.2 private
    • 2.3 internal
    • 2.4 默认可见性
    • 2.5 可见性的用处
  • 三、变量初始值
    • 3.1 值类型初始值

一、变量简述

变量是指可以保存数据的内部存储单元,里面的数据可以在程序运行时引用或者修改。

变量都有一个名字,称为变量名。变量名是由字母、数字或下划线“_”组成的字符串,但不能包含空格或其它的特殊字符,也不能以数字开头。比如正确的变量名:symbol、totalSupply、_name、amount1 等等。

Solidity 是一种静态类型的编程语言,它要求在变量声明时,必须明确指定变量的类型。

变量声明时,如果没有赋给初值,都会有一个基于其类型的默认值,比如 int 类型的默认值为 0。

Solidity 提供了 3 种类型的变量:状态变量局部变量全局变量

1.1 状态变量

  • 状态变量是指在智能合约中声明的持久化存储的变量。它存在于合约的整个生命周期,直到合约被销毁。
  • 状态变量的变动是要记录在区块链上的,永久存储,也就是通常所说的“数据上链”。
  • 状态变量在合约的不同函数之间共享,可以通过调用函数来读取或修改它的数据值。
  • 状态变量类似于其它编程语言中“结构体(go)或类(java)”的成员变量。

Solidity 中,状态变量通常在合约的顶层进行声明,并且定义在全部函数之外。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract StatusVar {uint256 myStatus = 999;  // 声明状态变量function getStatus() public view returns(uint256) {return myStatus;   // 使用状态变量}
}

这里首先声明了一个状态变量 myStatus,然后在函数 getStatus 中,把它作为函数的返回值。

运行合约,调用函数 getStatus,输出结果:

0: uint256: 999

1.2 局部变量

  • 局部变量是指在函数内部声明的变量,其作用域仅限于该函数内部
  • 局部变量的生命周期仅限于该函数的执行过程中,函数执行完毕后,局部变量将被销毁,它的值也将无法使用。
  • 局部变量与状态变量的区别在于:状态变量的数据是要上链的,而局部变量不上链
  • 声明局部变量的语法与状态变量类似,只是要在函数的内部进行声明。
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract LocalVar {function sum() public pure returns(uint256){uint256 a = 1;  // 声明局部变量 auint256 b = 2;  // 声明局部变量 buint256 result = a + b; // 声明局部变量 result,并使用局部变量a, breturn result; // 使用局部变量 result 作为返回值}
}

运行合约,调用函数 sum,输出结果:

0: uint256: 3

1.3 全局变量

全局变量是指在合约的顶层预先定义的特殊变量,用于提供有关区块链和交易属性的信息

全局变量是由 Solidity 语言本身提供,用户无权定义或者修改,但可以直接在任何位置使用。

例如,常用的全局变量有:当前区块的时间戳 block.timestamp、区块高度 block.number、函数调用者地址 msg.sender 等。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract GlobalVar {// 返回当前函数调用所在的区块号、时间戳、调用者地址function getGlobalVars() public view returns(uint,uint,address,uint){return (block.number, block.timestamp, msg.sender,msg.sender.balance);}
}

运行以上合约,调用函数 getGlobalVars,输出下面的结果:

0:uint256: 12             // 当前区块号
1:uint256: 1735107832     // 当前区块的时间戳
2:address: 0x5B38Da6a701c568545dCfcB03FcB875f56beddC4   //调用者的地址
3:uint256: 99999999999992885246 //调用者地址的余额

常用的全局变量(更完整的列表请看这个链接):

名称返回
blockhash(uint blockNumber) returns (bytes32)指定区块的哈希值
block.coinbase (address payable)当前区块的矿工地址
block.difficulty (uint)当前区块的难度值
block.gaslimit (uint)当前区块的gaslimit
block.number (uint)当前区块的区块号
block.timestamp (uint)当前区块的时间戳
gasleft() returns (uint256)当前剩余的gas
msg.data (bytes calldata)完成 calldata
msg.sender (address payable)消息发送者的地址,也就是函数调用者的地址
msg.sig (bytes4)消息的函数签名,也就是calldata的前四个字节
msg.value (uint)消息携带的token数量,也就是函数调用时,同时存入的token量
tx.gasprice (uint)交易的gas价格
tx.origin (address payable)交易的发送方地址

1.4 注意问题

在声明变量时,应该需要注意以下问题:

  • 变量名规范的写法是采用小驼峰形式:每个单词的首字母大写,但整体以小写字母或者下划线“_”开头。比如:symbol、totalSupply、_name。
  • 变量名不应该以数字开头,必须以字母或下划线开头。例如:123test 是一个无效的变量名,但是 _123test 是一个有效的变量名。
  • 变量名是区分大小写的。例如:Name 和 name 是两个不同的变量。
  • 变量声明时,如果没有赋给初值,它就会有一个基于其类型的默认值,比如 int 类型变量的默认值为 0。
  • Solidity 的变量没有 undefinednull 等空值的概念。

二、变量可见性

状态变量的可见性是用来控制状态变量在合约内部或外部的访问权限。

Solidity 为状态变量提供了 3 种可见性修饰符,分别是 publicprivateinternal,用于限制状态变量的访问权限。

2.1 public

如果状态变量的可见性声明为 public,也就是公开变量,那么在合约内部和外部都可以访问这个状态变量。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract PublicVisibility {uint256 public delta = 8; // 可见性声明为 publicfunction addDelta(uint256 num) external view returns(uint256) {uint256 sum = num + delta; // 函数内可以使用状态变量 deltareturn sum;}
}

我们把合约代码复制到 Remix,进行编译,并部署到区块链上:

在这里插入图片描述

我们从外部可以看到状态变量 delta,并且还可以从外部访问它,点击 delta 获得返回值为 8。

2.2 private

如果状态变量的可见性声明为 private,也就是私有变量,那么就只能在合约内部访问这个状态变量,而不能从外部访问。

通常,私有变量只能由合约内部的函数使用,在合约外部是不可见的,也无法使用。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract PrivateVisibility {uint256 private delta = 8; // 可见性声明为 privatefunction addDelta(uint256 num) external view returns(uint256) {uint256 sum = num + delta; // 函数内可以使用状态变量 deltareturn sum;}
}

我们把合约代码复制到 Remix,进行编译,并部署到区块链上:
在这里插入图片描述

我们从外部看不到状态变量 delta ,所以从外部也就无法访问它。

私有变量将数据隐藏在合约内部,不直接暴露给外部访问,可以防止外部代码直接修改或访问合约的内部状态,提高了合约的安全性。

所以,在 Solidity 编程中,无需外部使用的状态变量,尽可能使用 private 可见性

2.3 internal

如果状态变量的可见性声明为 internal,也就是内部变量,那么可以在当前合约和它的继承合约的内部访问这个状态变量,而不能从外部访问。

internalprivate 相比,就是 internal 变量可以在派生的合约中使用private 变量不能在派生合约中使用,只能在当前的合约中使用。

关于合约继承的相关知识,我们在此处不再延伸。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract InternalVisibility {uint256 internal delta = 8; // 可见性声明为 internalfunction addDelta(uint256 num) external view returns(uint256) {uint256 sum = num + delta; // 函数内可以使用状态变量 deltareturn sum;}
}// VarVisibility 继承合约 InternalVisibility
contract VarVisibility is InternalVisibility {function getDelta() external view returns(uint256) {return delta; // 继承合约中可以使用状态变量 delta}
}

我们把合约代码复制到 Remix,进行编译,并部署到区块链上:

在这里插入图片描述

我们从外部无法看到状态变量 delta,所以从外部也无法使用它。但 delta 可以在合约 InternalVisibility 内部和它的继承合约 VarVisibility 中使用。

2.4 默认可见性

如果状态变量在声明的时候,没有指定可见性,那么它的可见性就为默认值 internal

也就是说:

uint status = 1;等于uint internal status = 1;

2.5 可见性的用处

  1. 访问控制

    通过在状态变量上设置适当的可见性,可以实现对合约功能的访问控制。例如,将敏感数据存储在私有状态变量中,只允许合约的内部特定函数访问,从而确保外部只有通过授权才可以访问这些数据。

  2. 封装数据

    通过将状态变量声明为 private,可以限制对其数据的直接访问,只能通过合约内部的函数来访问,这样可以有效地封装内部数据,避免外部访问者直接操作状态变量,确保数据的安全性和一致性。

    另外,除了状态变量外,函数也有可见性的限制,而且函数的可见性增加了一个 external,我们将在后面的函数中详细讲解。

三、变量初始值

Solidity中,声明但没赋值的变量都有它的初始值或默认值。这一讲,我们将介绍常用变量的初始值。

3.1 值类型初始值

  • bool 类型变量的默认值为 false;

  • int 类型变量的默认值为 0,包括各种长度的有符号整型 ,比如 int8、int16…;

  • uint 类型变量的默认值为 0,包括各种长度的无符号整型,比如 uint8、uint16…;

  • string 类型变量的默认值为:“”;

  • address 类型变量的默认值为:0x0000000000000000000000000000000000000000 (或 address(0));

  • bytes32 类型变量的默认值为:0x0000…,共 64个 0;

  • enum 类型变量的默认值是它列表中的第一个元素。

  • Function 函数

    • internal: 空白函数

    • external: 空白函数

我们可以通过一个合约来学习变量的默认值:

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract VarDefault { enum status {normal, deleted} // 定义枚举类型 bool public a; // false int public b; // 0uint public c; // 0address public d; // 0x0000...,共 40 个 0 bytes32 public e; // 0x0000...,共 64 个 0string public f; // ""status public g; // 0,也就是 status.normalfunction fi() internal{} // internal空白函数 //可见性为 `internal` 的变量和函数可以被主合约和子合约访问
function fe() external{} // external空白函数  //可见性为 `external` 的函数只能被第三方合约访问。
}

我们将合约代码复制到 Remix,进行编译,并部署到区块链上:
在这里插入图片描述


http://www.ppmy.cn/devtools/152498.html

相关文章

jenkins-api操作

一. 简述: 在一个比较复杂的环境中, 往往会有自己开发的运维管理平台。在代码发布这块,尽管jenkins有一个比较方便的UI, 但很多团队还是喜欢集中式管理, 将发布功能(仅仅把jenkins作为一个发布组件使用)嵌入运维管理平…

【Spring MVC】如何运用应用分层思想实现简单图书管理系统前后端交互工作

前言 🌟🌟本期讲解关于SpringMVC的编程思想之应用分层~~~ 🌈感兴趣的小伙伴看一看小编主页:GGBondlctrl-CSDN博客 🔥 你的点赞就是小编不断更新的最大动力 🎆那…

数据库的DQL(3)

数据库的DQL(3) 分组查询 在MySQL中,group by关键字可以根据一个或多个字段对查询结果进行分组 group by 字段名1.分组函数 有时也叫聚合函数 count(): 查询表中的记录数量avg(): 求平均值sum(): 求和max():求最大值min():求最小值 案例1: mysql&g…

深入解析人工智能中的协同过滤算法及其在推荐系统中的应用与优化

目录 什么是协同过滤算法核心原理基本步骤相似度计算代码实现详解1.流程图2.创建基础的数据结构存储用户评分数据3.计算用户相似度4.获取相似用户5.推荐方法 算法优化建议1. 数据预处理优化去除异常值和噪声数据进行数据标准化使用稀疏矩阵优化存储 2. 相似度计算优化使用局部敏…

STM32使用DSP库 Keil方式添加

文章目录 前言一、添加DSP库二、使能FPU及配置1. 使能FPU2. 增加编译的宏3.增加头文件的检索路径三. 验证1. 源码中添加2.代码测试前言 添加DSP有两种方案,本文采用的是是Keil 中添加。 一、添加DSP库 在创建好的工程中添加DSP库:步骤如下: 步骤1:选择运行环境管理; 步…

Spring参数校验,数组入参校验 :List<E>

1、程序内直接校验 参数进入控制层方法,使用具体的参数方法来验证,灵活不优雅 Objects.isNull() String.isEmpty()2、验证 单对象(Obj)使用 Validated 优雅,不灵活,对象类配置多 首先在对象类中 1、校验…

kafka学习

kafka (消息、生产者、消费者、toptic、分区、偏移量、broker、集群) Kafka是什么? 学习Kafka的目的,为了解决高吞吐量项目的需求,Kafka号称大数据的杀手锏,这款为大数据而生的消息中间件,以其百亿级tps的吞吐量名声…

nacos安装及SpringCloud整合

参考资料: 参考视频 参考demo SpringCloud-Alibaba基础框架搭建 nacos官网 nacos部署教程: 1.准备环境 - 1.64 bit OS,支持 Linux/Unix/Mac/Windows,推荐选用 Linux/Unix/Mac。 - 2.64 bit JDK 1.8+;下载 & 配置。 - 3.Maven 3.2.x+;下载 & 配置。 2.版本选…