欢迎光临
黎跃春区块链技术博客

【Solidity智能合约开发第16篇】3分钟了解Solidity Types – Dynamically-sized byte array

一、Dynamically-sized byte array

  • string 是一个动态尺寸的UTF-8编码字符串,它其实是一个特殊的可变字节数组,string是引用类型,而非值类型。
  • bytes 动态字节数组,引用类型。

根据经验,在我们不确定字节数据大小的情况下,我们可以使用string或者bytes,而如果我们清楚的知道或者能够将字节书控制在bytes1 ~ bytes32,那么我们就使用bytes1 ~ bytes32,这样的话能够降低存储成本。

二、常规字符串 sting 转换为 bytes

string字符串中没有提供length方法获取字符串长度,也没有提供方法修改某个索引的字节码,不过我们可以将string转换为bytes,再调用length方法获取字节长度,当然可以修改某个索引的字节码。

1、源码

// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0 <0.7.0;

contract C {

    bytes9 public g = 0x6c697975656368756e;

    string public name = "liyuechun";

    function gByteLength() view public returns (uint) {

        return g.length;
    }

    function nameBytes() view public returns (bytes memory) {

        return bytes(name);
    }

    function nameLength() view public returns (uint) {

        return bytes(name).length;
    }

    function setNameFirstByteForL(bytes1 z)  public {

        // 0x4c => "L"
        bytes(name)[0] = z;
    }
}

2、说明

function nameBytes() view public returns (bytes memory) {

    return bytes(name);
}

nameBytes这个函数的功能是将字符串name转换为bytes,并且返回的结果为0x6c697975656368756e0x6c697975656368756e一共为9字节,也就是一个英文字母对应一个字节。

function nameLength() view public returns (uint) {

    return bytes(name).length;
}

我们之前讲过,string字符串它并不提供length方法帮助我们返回字符串的长度,所以在nameLength方法中,我们将name转换为bytes,然后再调用length方法来返回字节数,因为一个字节对应一个英文字母,所以返回的字节数量刚好等于字符串的长度。

function setNameFirstByteForL(bytes1 z) {

    // 0x4c => "L"
    bytes(name)[0] = z;
}

如果我们想将name字符串中的某个字母进行修改,那么我们直接通过x[k] = z的形式进行修改即可。x是bytes类型的字节数组,k是索引,zbytes1类型的变量值。

setNameFirstByteForL方法中,我就将liyuechun中的首字母修改成L,我传入的z的值为0x4c,即大写的L

三、汉字字符串或特殊字符的字符串转换为bytes

1、特殊字符

// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0 <0.7.0;

contract C {

    string public name = "a!+&520";

    function nameBytes() view public returns (bytes memory) {

        return bytes(name);
    }

    function nameLength() view public returns (uint) {

        return bytes(name).length;
    }

}

在这个案例中,我们声明了一个name字符串,值为a!+&520,根据nameBytesnameLength返回的结果中,我们不难看出,不管是字母数字还是特殊符号,每个字母对应一个byte(字节)

2、中文字符串

// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0 <0.7.0;

contract C {

    string public name = "黎跃春";

    function nameBytes() view public returns (bytes memory) {

        return bytes(name);
    }

    function nameLength() view public returns (uint) {

        return bytes(name).length;
    }

}

在上面的代码中,我们不难看出,黎跃春转换为bytes以后的内容为0xe9bb8ee8b783e698a5,一共9个字节。也就是一个汉字需要通过3个字节来进行存储。那么问题来了,以后我们取字符串时,字符串中最好不要带汉字,否则计算字符串长度时还得特殊处理。

四、bytes字节数组

// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0 <0.7.0;

contract C {

    bytes public name = new bytes(1); // 创建一个长度为1的字节数组

    // function setNameLength(uint length) public {

    //     name.length = length; // 现在新版本的length是只读状态,不可修改字节数组长度,5.0以下版本的solidity可修改长度。
    // }

    function nameLength() view public returns (uint) {

        return name.length; // 读取字节长度
    }

    // 往字节数组中添加字节
    function pushOneByte(byte b) public {

        // name.push(b); 
        name.push() = b; // 往数组里面添加一个字节,name.push(b) 等价于 name.push() = b;
    }

    function popOneByte() public {

        name.pop(); // 删除一个字节
    }

    function setIndexByte(uint index,byte b) public {

        name[index] = b; // 修改某个索引下的字节
    }

    function deleteByte() public {

        delete name; // 清空字节数组
    }

}

五、总结

对比分析:

  • 不可变字节数组

我们之前的文章中提到过如果我们清楚我们存储的字节大小,那么我们可以通过bytes1,bytes2,bytes3,bytes4,......,bytes32来声明字节数组变量,不过通过bytesI来声明的字节数组为不可变字节数组,字节不可修改,字节数组长度不可修改。

  • 可变字节数组

我们可以通过bytes name = new bytes(length) , length为字节数组长度,来声明可变大小和可修改字节内容的可变字节数组。

赞(0) 打赏
未经允许不得转载:黎跃春区块链技术博客 » 【Solidity智能合约开发第16篇】3分钟了解Solidity Types – Dynamically-sized byte array

评论 抢沙发

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址

区块链在线课程、区块链职业技术水平认证考试一网打尽

区块链技术在线课程区块链技术职业认证

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

微信扫一扫打赏