:2026-03-06 20:09 点击:1
在以太坊智能合约开发中,处理数据是核心任务之一,当需要存储一段固定长度的二进制数据时,例如哈希值(如 keccak256 的结果)、公钥、加密密文或任何其他已知长度的原始字节序列,bytes 类型便派上了用场,与可变长度的字节数组 bytes 不同,bytes1 到 bytes32 这种固定大小的字节数组在存储和计算上更具效率,是 Solidity 开发者工具箱中一个非常重要的工具。
本文将详细讲解在以太坊智能合约中如何定义、使用和操作固定字节数组。
固定字节数组是一种在声明时就必须指定其长度(从 1 到 32 个字节)的数据类型,一旦声明,其长度在合约的整个生命周期内都不可改变。
这种类型的优势在于:
bytes32 直接赋值给一个 bytes20 变量,需要显式转换。声明固定字节数组非常简单,使用 bytes1 到 bytes32 的关键字,后跟变量名。
语法:
bytes1 到 bytes32 变量名;
示例:
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract FixedBytesExample {
// 声明一个 32 字节的字节数组,通常用于存储哈希值
bytes32 public myHash;
// 声明一个 20 字节的字节数组,通常用于存储以太坊地址
bytes20 public myAddress;
// 声明一个 16 字节的字节数组,例如用于 AES-
128 的密钥
bytes16 public mySecretKey;
// 声明一个 1 字节的字节数组
bytes1 public myFlag;
// 构造函数中进行初始化
constructor() {
// 可以通过字面量直接赋值
// 字面量必须是有效的十六进制字符串,且长度必须匹配
myHash = 0x608060405234801561001057600080fd5b5061015a806100206000396000f3fe608060405234801561001057600080fd5b50600436106100415760003560e01c806360fe47b1146100465780636d4ce63c14610064575b600080fd5b61004e61008a565b60405161005b91906100f9565b60405180910390f35b61007e6004803603810190610079919061014d565b610093565b60405161008b91906100f9565b60405180910390f35b60008054905090565b8060008190555050565b6000813590506100b6816101a4565b92915050565b6000602082840312156100d2576100d161019c565b5b60006100e0848285016100a7565b91505092915050565b6100f18161017c565b82525050565b600060208201905061010c60008301846100e8565b92915050565b6000819050919050565b600061012661012161011d565b61017c565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b60005b8381101561016a57808201518184015260208101905061014f565b83811115610179576000848401525b50505050565b6000600282049050600182168061019757607f821691505b602082108114156101ab576101aa6101c9565b5b50919050565b6101ba8161017c565b81146101c557600080fd5b5056fea2646970667358221220c1b5b3c4d2e4a8a5b6c7d8e9f0a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6e7f8a64736f6c63430008070033";
// 也可以使用十六进制字符串
myAddress = 0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045; // Vitalik Buterin's address
// 初始化 mySecretKey
mySecretKey = 0x1234567890abcdef1234567890abcdef;
// 初始化 myFlag
myFlag = 0xff; // 设置为全 1,表示 true
}
}
关键点:
0x 开头的十六进制字面量直接赋值,编译器会检查其长度是否与变量类型匹配。固定字节数组的元素可以通过索引从 0 开始访问,每个索引对应一个字节(8位)。
语法: 数组名[索引位置];
示例:
contract BytesAccessExample {
bytes32 public myData = 0x68656c6c6f20776f726c6421deadbeefcafe00000000000000000000000000000000;
// 获取第 0 个字节 (0x68, 即 'h' 的 ASCII 码)
function getByte0() public pure returns (bytes1) {
return myData[0];
}
// 获取第 4 个字节 (0x6f, 即 'o' 的 ASCII 码)
function getByte4() public pure returns (bytes1) {
return myData[4];
}
// 修改第 30 个字节
function modifyByte30(bytes1 newValue) public {
myData[30] = newValue;
}
// 修改多个字节(通过拼接)
function modifyFirst5Bytes(bytes5 newValue) public {
// 将 myData 的前 5 个字节替换为 newValue
myData = bytes32(bytes32(newValue) & 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF00000000000000000000000000000000 | bytes32(myData) & 0x000000000000000000000000000000000000000000000000000000000000FFFF);
// 注意:直接修改多个字节需要位运算,比较繁琐。
// 更好的方法是先将整个数组读出,修改内存中的副本,然后重新赋值。
}
}
注意:
&, , <<)来完成,这可能会使代码变得复杂,一个更清晰的方法是先将 bytes32 读取到内存中,创建一个副本进行修改,然后将整个值写回存储。固定字节数组拥有一些内置的成员方法,非常实用。
.length:返回字节数组的长度本文由用户投稿上传,若侵权请提供版权资料并联系删除!