一.固定长度的数组(Arrays)

1.固定长度类型数组的声明

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. // 数组的长度为5,数组里面的存储的值的类型为uint类型
  6. uint [5] T = [1,2,3,4,5];
  7. }

2.通过length方法获取数组长度遍历数组求总和

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. // 数组的长度为5,数组里面的存储的值的类型为uint类型
  6. uint [5] T = [1,2,3,4,5];
  7.  
  8. // 通过for循环计算数组内部的值的总和
  9. function numbers() constant public returns (uint) {
  10. uint num = 0;
  11. for(uint i = 0; i < T.length; i++) {
  12. num = num + T[i];
  13. }
  14. return num;
  15. }
  16. }

3.尝试修改T数组的长度

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. uint [5] T = [1,2,3,4,5];
  6.  
  7. function setTLength(uint len) public {
  8.  
  9. T.length = len;
  10. }
  11. }

  PS:声明数组时,一旦长度固定,将不能再修改数组的长度。

 4.尝试修改T数组内部值

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. uint [5] T = [1,2,3,4,5];
  6.  
  7. function setTIndex0Value() public {
  8.  
  9. T[0] = 10;
  10. }
  11.  
  12. // 通过for循环计算数组内部的值的总和
  13. function numbers() constant public returns (uint) {
  14. uint num = 0;
  15. for(uint i = 0; i < T.length; i++) {
  16. num = num + T[i];
  17. }
  18. return num;
  19. }
  20. }

  T数组初始的内容为[1,2,3,4,5],总和为15 ,当我点击setTIndex0Value方法将第0个索引的1修改为10时,总和为24。

  PS:通过一个简单的试验可证明固定长度的数组只是不可修改它的长度,不过可以修改它内部的值,而bytes0 ~ bytes32固定大小字节数组中,大小固定,内容固定,长度和字节均不可修改。

5.尝试通过push往T数组中添加值

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4. uint [5] T = [1,2,3,4,5];
  5.  
  6. function pushUintToT() public {
  7.  
  8. T.push(6);
  9. }
  10. }

  PS:固定大小的数组不能调用push方法向里面添加存储内容,声明一个固定长度的数组,比如:uint [5] T,数组里面的默认值为[0,0,0,0,0],我们可以通过索引修改里面的值,但是不可修改数组长度以及不可通过push添加存储内容。

二.可变长度的Arrays

1.可变长度类型数组的声明

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. uint [] T = [1,2,3,4,5];
  6.  
  7. function T_Length() constant returns (uint) {
  8.  
  9. return T.length;
  10. }
  11. }

  uint [] T = [1,2,3,4,5],这句代码表示声明了一个可变长度的T数组,因为我们给它初始化了5个无符号整数,所以它的长度默认为5。

2.通过length方法获取数组长度遍历数组求总和

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. uint [] T = [1,2,3,4,5];
  6.  
  7. // 通过for循环计算数组内部的值的总和
  8. function numbers() constant public returns (uint) {
  9. uint num = 0;
  10. for(uint i = 0; i < T.length; i++) {
  11. num = num + T[i];
  12. }
  13. return num;
  14. }
  15. }

  通过length方法获取数组长度遍历数组求总和

3.尝试修改T数组的长度

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. uint [] T = [1,2,3,4,5];
  6.  
  7. function setTLength(uint len) public {
  8.  
  9. T.length = len;
  10. }
  11.  
  12. function TLength() constant returns (uint) {
  13.  
  14. return T.length;
  15. }
  16. }

  PS:对可变长度的数组而言,可随时通过length修改它的长度。

4.尝试通过push往T数组中添加值

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. uint [] T = [1,2,3,4,5];
  6.  
  7. function T_Length() constant public returns (uint) {
  8.  
  9. return T.length;
  10. }
  11.  
  12. function pushUintToT() public {
  13.  
  14. T.push(6);
  15. }
  16.  
  17. function numbers() constant public returns (uint) {
  18. uint num = 0;
  19. for(uint i = 0; i < T.length; i++) {
  20. num = num + T[i];
  21. }
  22. return num;
  23. }
  24. }

  PS:当往里面增加一个值,数组的个数就会加1,当求和时也会将新增的数字加起来。

三.二维数组 - 数组里面放数组

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. uint [2][3] T = [[1,2],[3,4],[5,6]];
  6.  
  7. function T_len() constant public returns (uint) {
  8.  
  9. return T.length; //
  10. }
  11. }

  uint [2][3] T = [[1,2],[3,4],[5,6]]这是一个三行两列的数组,你会发现和Java、C语言等的其它语言中二位数组里面的列和行之间的顺序刚好相反。在其它语言中,上面的内容应该是这么存储uint [2][3] T = [[1,2,3],[4,5,6]]。

  上面的数组T是storage类型的数组,对于storage类型的数组,数组里面可以存放任意类型的值(比如:其它数组,结构体,字典/映射等等)。对于memory类型的数组,如果它是一个public类型的函数的参数,那么它里面的内容不能是一个mapping(映射/字典),并且它必须是一个ABI类型。

四.创建 Memory Arrays
  创建一个长度为length的memory类型的数组可以通过new关键字来创建。memory数组一旦创建,它不可通过length修改其长度。

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. function f(uint len) {
  6. uint[] memory a = new uint[](7);
  7. bytes memory b = new bytes(len);
  8. // 在这段代码中 a.length == 7 、b.length == len
  9. a[6] = 8;
  10. }
  11. }

五.数组字面量 Array Literals / 内联数组 Inline Arrays

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. function f() public {
  6. g([1, 2, 3]);
  7. }
  8.  
  9. function g(uint[3] _data) public {
  10. // ...
  11. }
  12. }

  在上面的代码中,[1, 2, 3]是 uint8[3] memory 类型,因为1、2、3都是uint8类型,他们的个数为3,所以[1, 2, 3]是 uint8[3] memory 类型。但是在g函数中,参数类型为uint[3]类型,显然我们传入的数组类型不匹配,所以会报错。

  正确的写法如下:

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. function f() public {
  6. g([uint(1), 2, 3]);
  7. }
  8.  
  9. function g(uint[3] _data) public {
  10. // ...
  11. }
  12. }

  在这段代码中,我们将[1, 2, 3]里面的第0个参数的类型强制转换为uint类型,所以整个[uint(1), 2, 3]的类型就匹配了g函数中的uint[3]类型。

memory类型的固定长度的数组不可直接赋值给storage/memory类型的可变数组

  TypeError: Type uint256[3] memory is not implicitly convertible to expected type uint256[] memory.

  TypeError: Type uint256[3] memory is not implicitly convertible to expected type uint256[] storage pointer

  正确使用

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4. function f() public {
  5.  
  6. uint[3] memory x = [uint(1), 3, 4];
  7. }
  8. }

六.创建固定大小字节数组/可变大小字节数组
  之前我们的文章中深入讲解了bytes0 ~ bytes32、bytes以及string的使用。bytes0 ~ bytes32创建的是固定字节大小的字节数组,长度不可变,内容不可修改。而string是特殊的可变字节数组,它可以转换为bytes以通过length获取它的字节长度,亦可通过索引修改相对应的字节内容。

  创建可变字节数组除了可以通过bytes b = new bytes(len)来创建外,我们亦可以通过byte[] b来进行声明。

  而bytes0 ~ bytes32我们可以通过byte[len] b来创建,len 的范围为0 ~ 32。不过这两种方式创建的不可变字节数组有一小点区别,bytes0 ~ bytes32直接声明的不可变字节数组中,长度不可变,内容不可修改。而byte[len] b创建的字节数组中,长度不可变,但是内容可修改。

  1. pragma solidity ^0.4.4;
  2.  
  3. contract C {
  4.  
  5. bytes9 a = 0x6c697975656368756e;
  6. byte[9] aa = [byte(0x6c),0x69,0x79,0x75,0x65,0x63,0x68,0x75,0x6e];
  7.  
  8. byte[] cc = new byte[](10);
  9.  
  10. function setAIndex0Byte() public {
  11. // 错误,不可修改
  12. a[0] = 0x89;
  13. }
  14.  
  15. function setAAIndex0Byte() public {
  16.  
  17. aa[0] = 0x89;
  18. }
  19.  
  20. function setCC() public {
  21.  
  22. for(uint i = 0; i < a.length; i++) {
  23.  
  24. cc.push(a[i]);
  25. }
  26. }
  27. }

七.总结

  本篇文章系统讲解了可变与不可变数组的创建、以及二位数组与其它语言中二位数组的区别,同时讲解了如何创建memory类型的数组以及对bytes0 ~ bytes32、bytes与byte[]对比分析。

文章来源:https://blog.csdn.net/liyuechun520/article/details/78410733

Solidity数组的更多相关文章

  1. 从零构建以太坊(Ethereum)智能合约到项目实战——学习笔记10

    P57 .1-Solidity Types - 玩转 Solidity 数组 (Arrays) 学习目标 1.掌握Arrays的可变不可变的创建 2.深度理解可变数组和不可变数组之间的区别 3.二维数 ...

  2. 智能合约语言 Solidity 教程系列5 - 数组介绍

    写在前面 Solidity 是以太坊智能合约编程语言,阅读本文前,你应该对以太坊.智能合约有所了解, 如果你还不了解,建议你先看以太坊是什么 本文前半部分是参考Solidity官方文档(当前最新版本: ...

  3. Solidity的地址 数组如何判断是否包含一个给定的地址?

    Q: given address[] wallets. What is the correct method to check that the list contains a given addre ...

  4. solidity 学习笔记 2 (二维数组)

    solidity 二维数组: pragma solidity ^0.4.23; contract twoArray{ uint[2][3] grade =[[20,30],[40,50],[45,60 ...

  5. solidity定长数组和动态数组

    固定长度的数组 固定长度数组声明 直接在定义数组的时候声明固定长度数组的值: uint[5] fixedArr = [1,2,3,4,5]; 可通过数组的length属性来获得数组的长度,进而进行遍历 ...

  6. [Contract] Solidity address payable 转换与数组地址

    address payable  -->  address address payable addr1 = msg.sender; address addr2 = addr1; // 隐式转 a ...

  7. Solidity教程系列1 - 类型介绍

    现在的Solidity中文文档,要么翻译的太烂,要么太旧,决定重新翻译下,再加上代码事例讲解. 写在前面 Solidity是以太坊智能合约编程语言,阅读本文前,你应该对以太坊.智能合约有所了解, 如果 ...

  8. 智能合约语言 Solidity 教程系列4 - 数据存储位置分析

    写在前面 Solidity 是以太坊智能合约编程语言,阅读本文前,你应该对以太坊.智能合约有所了解, 如果你还不了解,建议你先看以太坊是什么 这部分的内容官方英文文档讲的不是很透,因此我在参考Soli ...

  9. 智能合约语言 Solidity 教程系列6 - 结构体与映射

    写在前面 Solidity 是以太坊智能合约编程语言,阅读本文前,你应该对以太坊.智能合约有所了解, 如果你还不了解,建议你先看以太坊是什么 本系列文章一部分是参考Solidity官方文档(当前最新版 ...

随机推荐

  1. Codeforces821C Okabe and Boxes 2017-06-28 15:24 35人阅读 评论(0) 收藏

    C. Okabe and Boxes time limit per test 3 seconds memory limit per test 256 megabytes input standard ...

  2. All Start Here.

    缘由 本博客是为天大软院 2016 级研一课程"现代软件工程"的课程设计而开设.同时借此机会和同学们进行技术交流与分享. 我们小组共有四位成员: 陈岩岩 2016218020 刘莞 ...

  3. Implementation of WC in JAVA

    Implementation of WC in JAVA github地址 相关要求 基本功能 -c [文件名] 返回文件的字符数 (实现) -w [文件名] 返回文件的词的数目 (实现) -l [文 ...

  4. Win7下无法启动sql server fulltext search (mssqlserver)的问题

    在Win7下安装了SQL Server 2005, 但启动“SQL Server FullText Search (MSSQLSERVER)”服务时启动不成功,系统日志显示“SQL Server Fu ...

  5. .net core 分布式配置中心

    github地址 https://github.com/wangchengqun/ratel 配置文件 数据同步端口 Server: ip: 127.0.0.1 port: 7890 浏览器访问 ht ...

  6. AJPFX平台讲述买卖、点差、单位,外汇的交易时间以及外汇交易者的参与者

    AJPFX平台讲解:买(多).卖(空).点差.单位 外汇保交易也就是通过外汇的升值和贬值来赚取利润.以EURUSD(欧元/美元)为例.假设目前价格为1.3820左右,即1欧元兑换1.3820美元.这个 ...

  7. Ubuntu18.04 - 安装深度桌面(Deepin Linux Desktop)

    深度Linux官网:https://www.deepin.org 深度桌面感觉确实不错,所以打算安装上来,以后体验一下,下面是安装的命令: 1, sudo add-apt-repository ppa ...

  8. Java中goto和break、continue实现区别

    goto 关键字很早就在程序设计语言中出现.事实上,goto 是汇编语言的程序控制结构的始祖:“若条件 A,则跳到这里:否则跳到那里”.若阅读由几乎所有编译器生成的汇编代码,就会发现程序控制里包含了许 ...

  9. Ubuntu系统常见问题解决

    我本人使用Ubuntu16.04LTS已经有一段时间了.实话来说,ubuntu虽然确实是开源世界的一款优秀的桌面操作系统,但是易用性和稳定性相比Windows还是差太多.本人从安装系统到逐步配置生产环 ...

  10. [POI2015]LOG(树状数组)

    今天考试考了这题,所以来贡献\([POI2015]LOG\)的第一篇题解.代码略丑,调了快三个小时才调出来\(AC\)代码. 对于这种小清新数据结构题,所以我觉得树状数组才是这道题的正确打开方式. 首 ...