timezone |
---|
Asia/Shanghai |
- 自我介绍 Hello everyone, I'm FFFFourwood, but you can call me "Arthur". I work as a software engineer. JS|React| C#
- 你认为你会完成本次残酷学习吗? Fighting!!!!
Solidity 简介
- Solidity 是用于编写以太坊智能合约的编程语言。
- 特点:基于对象、高级。
- 掌握 Solidity 可帮助理解区块链项目,规避风险,提升币圈参与感。
开发工具:Remix
- Remix 是以太坊官方推荐的智能合约开发工具,适合新手。
- 无需安装软件,直接在浏览器中编写、编译和部署合约。
- 网址:Remix
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.21;
contract HelloWeb3 {
string public _string = "Hello Web3!";
}
- 第一行:声明软件许可(MIT),以注释开头。
- 第二行:声明使用的 Solidity 版本(0.8.21)。
- 第三行:定义合约 HelloWeb3,并声明变量 _string,值为 "Hello Web3!"。
Solidity 值类型 个人笔记
- 布尔型
- 取值:true / false
- 运算符:!(非),&&(与),||(或),==(等于),!=(不等于)
- 短路规则:&& 遇到 false,|| 遇到 true,后续不执行。
- 整型
- 有符号整型 int,无符号整型 uint。
- 运算符:+,-,*,/,%(取余),**(幂运算)。
- 示例:uint256 public _number = 20220330;
- 地址类型
- 普通地址:address
- 可支付地址:payable address,可以接收以太币并调用 balance 查询余额。
- 定长字节数组
- 定长字节:bytes1,bytes32,最多 32 字节。
- 示例:bytes32 public _byte32 = "MiniSolidity";
- 枚举(enum)
- 定义多个状态,值从 0 开始。
- 可以与 uint 相互转换。
函数
- pure 函数不读不写链上状态。
- view 函数仅能读取状态变量。
- payable 函数可以接收以太币。
- 函数可见性(public、private、internal、external)决定函数的访问权限。
函数输出总结
- 返回值:return 和 returns
- returns:声明函数的返回类型,位于函数名后面。
- return:函数内部使用,返回具体值。
- 命名式返回
- 在 returns 关键字中声明返回变量的名称,Solidity 会自动初始化并返回这些变量。
- 你可以选择显式使用 return,也可以省略。
- 解构式赋值
- 可以通过解构赋值读取函数的所有或部分返回值。
- 不需要的返回值可以留空,用 , 隔开。
变量数据存储和作用域
- 引用类型
引用类型包括数组和结构体,由于其占用存储空间大,必须明确声明数据存储的位置。常见的存储位置有:
• storage:用于合约的状态变量,数据存储在链上,消耗较多的 gas。
• memory:用于函数的参数和临时变量,数据存储在内存中,不上链。
• calldata:类似于 memory,但不可修改,常用于函数参数。
-
数据位置的赋值规则
• storage to storage:创建引用,修改一个会影响另一个。 • memory to memory:创建引用,修改一个会影响另一个。 • 其他情况:创建副本,修改副本不会影响原始变量。
-
变量的作用域
Solidity 中变量的作用域分为三类:
• 状态变量:存储在链上的变量,所有函数可访问,gas 消耗高。
• 局部变量:仅在函数执行时有效,存储在内存中,gas 消耗低。
• 全局变量:Solidity 预定义的变量,如 msg.sender、block.number、msg.data 等,函数内无需声明即可使用。
-
以太单位与时间单位
• 以太单位:wei、gwei、ether,分别代表不同的精度单位,确保交易精度。 • 时间单位:seconds、minutes、hours、days、weeks,用于设定合约中的时间条件。
引用类型总结:数组(Array)和结构体(Struct)
- 数组(Array)
-
固定长度数组:在声明时指定长度,无法改变。
-
可变长度数组(动态数组):长度不固定,可以动态改变。
-
创建规则:
- 动态数组用
new
创建,并需指定长度。 - 数组字面常数初始化类型由第一个元素决定。
- 动态数组用
-
数组操作:
length
:获取数组长度。push()
:向数组末尾添加元素。pop()
:移除数组最后一个元素。
- 结构体(Struct)
-
结构体用于创建自定义数据类型,元素可以是原始类型或引用类型。
-
赋值方法:
- 通过
storage
引用赋值:Student storage _student = student; _student.id = 11;
- 直接引用状态变量:
student.id = 1;
- 构造函数式赋值:
student = Student(3, 90);
key-value
形式赋值:student = Student({id: 4, score: 60});
- 通过
-
数组和结构体是 Solidity 中的常用引用类型。数组支持固定和动态长度,常用成员方法包括
push()
和pop()
。结构体用于定义自定义数据类型,并有多种赋值方式。
映射类型(Mapping)总结
- 映射(Mapping)简介 映射是一种存储键值对(key-value)的数据结构,类似哈希表。通过映射,可以根据键(Key)查询对应的值(Value)。
映射声明格式:
mapping(_KeyType => _ValueType) public mappingName;
- 映射的规则
-
规则1:
_KeyType
必须是 Solidity 内置的值类型(如uint
、address
),不能是自定义类型。_ValueType
可以是自定义类型。 -
规则2:映射的存储位置必须是
storage
,因此映射适用于状态变量或函数中的storage
变量。映射不能作为函数的参数或返回类型。 -
规则3:声明为
public
的映射,Solidity 会自动创建一个 getter 函数来通过键查询值。 -
规则4:向映射添加键值对的语法:
idToAddress[_Key] = _Value;
- 映射的原理
-
原理1:映射不存储键的具体信息,也没有长度(
length
)属性。 -
原理2:映射通过
keccak256(abi.encodePacked(key, slot))
的哈希值作为偏移量存取对应的值。 -
原理3:所有未赋值的键会有默认值。例如,
uint
类型的默认值为0
。
- 总结
映射是 Solidity 中常用的哈希表数据结构,通过键查询值。它不存储键的信息且未赋值的键有默认值。映射的键类型必须是内置值类型,存储位置必须是
storage
。
变量初始值
- Solidity 中,声明但未赋值的变量会有默认初始值,具体根据变量类型而定。
- 值类型初始值
- boolean:
false
- string:
""
(空字符串) - int:
0
- uint:
0
- enum: 枚举的第一个元素
- address:
0x0000000000000000000000000000000000000000
(或address(0)
) - function
- internal: 空白函数
- external: 空白函数
- 引用类型初始值
- 映射(mapping): 所有元素都是默认值的映射。
- 结构体(struct): 所有成员设为默认值的结构体。
- 数组(array)
- 动态数组:
[]
(空数组) - 静态数组(定长): 所有成员设为默认值的数组。
- 动态数组:
delete
操作符
delete
操作符可以将变量的值重置为初始值。
常量:constant 和 immutable
常量:constant
和 immutable
总结
constant
关键字
- 特点:
constant
变量必须在声明时初始化,之后不能再改变。 - 作用:提高合约的安全性和节省 gas。
- 支持的类型:数值类型、
string
和bytes
可以声明为constant
,其他类型则不支持。
immutable
关键字
- 特点:
immutable
变量可以在声明时或构造函数(constructor
)中初始化,但初始化后不能再改变。 - 灵活性:可以通过构造函数或全局变量来初始化,例如
address(this)
、block.number
,或自定义函数返回值。 - Solidity 版本影响:在 Solidity v0.8.21 之后,
immutable
变量不需要显式初始化,否则需要显式初始化。
- 常见错误
-
修改
constant
变量:会抛出错误TypeError: Cannot assign to a constant variable.
。 -
修改
immutable
变量:会抛出错误TypeError: Immutable state variable already initialized.
。 -
constant
和immutable
关键字用于确保某些变量在初始化后不再改变,增加了合约的安全性和降低了 gas 消耗。
控制流与插入排序实现
- 控制流 Solidity 的控制流与其他编程语言类似,主要包含以下几种:
- if-else
- for 循环
- while 循环
- do-while 循环
- 三元运算符
- Solidity 插入排序实现
正确的做法是确保 j
不会小于 1,从而避免负值问题:
function insertionSort(uint[] memory a) public pure returns (uint[] memory) {
for (uint i = 1; i < a.length; i++) {
uint temp = a[i];
uint j = i;
while (j >= 1 && temp < a[j - 1]) {
a[j] = a[j - 1];
j--;
}
a[j] = temp;
}
return a;
}
运行结果:输入 [2,5,3,1]
,输出 [1,2,3,5]
。
构造函数和修饰器总结
构造函数 (Constructor)
-
构造函数是一种特殊的函数,在合约部署时自动运行一次,用来初始化合约的参数。
-
每个合约可以定义一个构造函数,只能在部署时调用一次。
-
历史变化: 在 0.4.22 之前,构造函数和合约同名。0.4.22 后改用
constructor
语法,避免函数名冲突导致的漏洞。
修饰器 (Modifier)
- 修饰器类似面向对象中的装饰器,主要用于函数运行前进行条件检查,减少代码冗余。
- 用法示例:定义一个
onlyOwner
修饰器,确保只有合约的owner
地址能调用某个函数。// 定义修饰器 modifier onlyOwner { require(msg.sender == owner); // 检查调用者是否是 owner 地址 _; // 继续执行函数主体 } // 函数使用 onlyOwner 修饰器 function changeOwner(address _newOwner) external onlyOwner { owner = _newOwner; }
OpenZeppelin的Ownable标准实现
-
OpenZeppelin 维护了一套标准的 Solidity 合约库,
Ownable.sol
是其中用于控制合约权限的标准实现。 -
通过定义
onlyOwner
修饰器来保护重要函数,仅允许合约所有者调用这些函数。 -
可以通过 Remix 测试 OpenZeppelin 的 Ownable 合约实现。
-
构造函数 在合约部署时初始化参数,只能调用一次。
-
修饰器 用于执行条件检查,例如权限控制,减少重复代码。
-
Ownable 合约 是常见的权限控制实现,确保合约的关键操作只能由特定地址执行。