区块链100讲:Solidity语法constant/view/pure关键字定义

Posted by

区块链100讲:Solidity语法constant/view/pure关键字定义

本文作者:HiBlock区块链社区技术布道者辉哥

原文发布于简书

区块链100讲:Solidity语法constant/view/pure关键字定义

通过本文学习,熟悉了解以太坊智能合约语言Solidity语法中constant,view,pure的区别。

1

区别总结

在Solidity中constant,view,pure三个函数修饰词的作用是告诉编译器,函数不改变/不读取状态变量,这样函数执行就可以不消耗gas了(是完全不消耗!),因为不需要矿工来验证。

在Solidity v4.17之前,只有constant,后来有人嫌constant这个词本身代表变量中的常量,不适合用来修饰函数,所以将constant拆成了view和pure。

  • view的作用和constant一模一样,可以读取状态变量但是不能改;

  • pure则更为严格,pure修饰的函数不能改也不能读状态变量,否则编译通不过。

大家可以运行以下测试代码来加深这3个关键字的理解。

contract constantViewPure{

string name;

uint public age;

function constantViewPure() public{

name = “liushiming”;

age = 29;

}

function getAgeByConstant() public constant returns(uint){

age += 1; //声明为constant,在函数体中又试图去改变状态变量的值,编译会报warning, 但是可以通过

return age; // return 30, 但是!状态变量age的值不会改变,仍然为29!

}

function getAgeByView() public view returns(uint){

age += 1; //view和constant效果一致,编译会报warning,但是可以通过

return age; // return 30,但是!状态变量age的值不会改变,仍然为29!

}

function getAgeByPure() public pure returns(uint){

return age; //编译报错!pure比constant和view都要严格,pure完全禁止读写状态变量!

return 1;

}

}

2

详细描述

2.1 Constant 状态变量

状态变量可以被声明为 constant。在这种情况下,只能使用那些在编译时有确定值的表达式来给它们赋值。 任何通过访问 storage,区块链数据(例如 now, this.balance 或者 block.number)或执行数据( msg.gas ) 或对外部合约的调用来给它们赋值都是不允许的。 在内存分配上有边界效应(side-effect)的表达式是允许的,但对其他内存对象产生边界效应的表达式则不行。 内建(built-in)函数 keccak256,sha256,ripemd160,ecrecover,addmod 和 mulmod 是允许的(即使他们确实会调用外部合约)。

允许带有边界效应的内存分配器的原因是这将允许构建复杂的对象,比如查找表(lookup-table)。 此功能尚未完全可用。

编译器不会为这些变量预留存储,它们的每次出现都会被替换为相应的常量表达式(这将可能被优化器计算为实际的某个值)。

不是所有类型的状态变量都支持用 constant 来修饰,当前支持的仅有值类型和字符串。

pragma solidity ^0.4.0;

contract C {

uint constant x = 32**22 + 8;

string constant text = “abc”;

bytes32 constant myHash = keccak256(“abc”);

}

2.2 View 函数

可以将函数声明为 view 类型,这种情况下要保证不修改状态。

下面的语句被认为是修改状态:

  • 修改状态变量。

  • 产生事件(https://solidity-cn.readthedocs.io/zh/develop/contracts.html?highlight=view#events)。

  • 创建其它合约(https://solidity-cn.readthedocs.io/zh/develop/control-structures.html#creating-contracts)。

  • 使用 selfdestruct。

  • 通过调用发送以太币。

  • 调用任何没有标记为 view 或者 pure 的函数。

  • 使用低级调用。

  • 使用包含特定操作码的内联汇编。

pragma solidity ^0.4.16;

contract C {

function f(uint a, uint b) public view returns (uint) {

return a * (b + 42) + now;

}

}

注解:

  • onstant 是 view 的别名。

  • Getter 方法被标记为 view。

  • 编译器没有强制 view 方法不能修改状态。

2.3 Pure 函数

函数可以声明为 pure ,在这种情况下,承诺不读取或修改状态。

除了上面解释的状态修改语句列表之外,以下被认为是从状态中读取:

  • 读取状态变量。

  • 访问 this.balance 或者 <address>.balance。

  • 访问 block,tx, msg 中任意成员 (除 msg.sig 和 msg.data 之外)。

  • 调用任何未标记为 pure 的函数。

  • 使用包含某些操作码的内联汇编。

pragma solidity ^0.4.16;contract C {

function f(uint a, uint b) public pure returns (uint) {

return a * (b + 42); }}

警告

编译器没有强制 pure 方法不能读取状态。

区块链100讲:Solidity语法constant/view/pure关键字定义

区块链马拉松|Blockathon(2018)上海邀请你来观看路演啦~

时间:2018年10月21日15:00-17:00

地点:上海市黄浦区柳泉弄1号2楼(P2·露香园)

免费参加,还有礼品哦

报名方式:识别下图二维码或者点击“阅读原文即可报名。

区块链100讲:Solidity语法constant/view/pure关键字定义

北京blockathon回顾:

Blockathon(北京):48小时极客开发,区块松11个现场交付项目创意公开

成都blockathon回顾:

Blockathon2018(成都站)比赛落幕,留给我们这些区块链应用思考

原文始发于微信公众号( 区块链社区HiBlock ):区块链100讲:Solidity语法constant/view/pure关键字定义