solidity 函数修饰器 modifier

news/2024/11/23 20:24:06/

函数 修饰器modifier

使用 修饰器modifier 可以轻松改变函数的行为。 例如,它们可以在执行函数之前自动检查某个条件。 修饰器modifier 是合约的可继承属性, 并可能被派生合约覆盖。

pragma solidity ^0.4.11;contract owned {function owned() public { owner = msg.sender; }address owner;// 这个合约只定义一个修饰器,但并未使用: 它将会在派生合约中用到。// 修饰器所修饰的函数体会被插入到特殊符号 _; 的位置。// 这意味着如果是 owner 调用这个函数,则函数会被执行,否则会抛出异常。modifier onlyOwner {require(msg.sender == owner);_;}
}contract mortal is owned {// 这个合约从 `owned` 继承了 `onlyOwner` 修饰符,并将其应用于 `close` 函数,// 只有在合约里保存的 owner 调用 `close` 函数,才会生效。function close() public onlyOwner {selfdestruct(owner);}
}contract priced {// 修改器可以接收参数:modifier costs(uint price) {if (msg.value >= price) {_;}}
}contract Register is priced, owned {mapping (address => bool) registeredAddresses;uint price;function Register(uint initialPrice) public { price = initialPrice; }// 在这里也使用关键字 `payable` 非常重要,否则函数会自动拒绝所有发送给它的以太币。function register() public payable costs(price) {registeredAddresses[msg.sender] = true;}function changePrice(uint _price) public onlyOwner {price = _price;}
}contract Mutex {bool locked;modifier noReentrancy() {require(!locked);locked = true;_;locked = false;}// 这个函数受互斥量保护,这意味着 `msg.sender.call` 中的重入调用不能再次调用  `f`。// `return 7` 语句指定返回值为 7,但修改器中的语句 `locked = false` 仍会执行。function f() public noReentrancy returns (uint) {require(msg.sender.call());return 7;}
}

如果同一个函数有多个 修饰器modifier,它们之间以空格隔开,修饰器modifier 会依次检查执行。

警告: 在早期的 Solidity 版本中,有 修饰器modifier 的函数,return 语句的行为表现不同。

修饰器modifier 或函数体中显式的 return 语句仅仅跳出当前的 修饰器modifier 和函数体。 返回变量会被赋值,但整个执行逻辑会从前一个 修饰器modifier 中的定义的 “_” 之后继续执行。

修饰器modifier 的参数可以是任意表达式,在此上下文中,所有在函数中可见的符号,在 修饰器modifier 中均可见。 在 修饰器modifier 中引入的符号在函数中不可见(可能被重载改变)。

reference: https://solidity-cn.readthedocs.io/zh/develop/contracts.html#modifiers

编译器源码:

void ContractCompiler::appendModifierOrFunctionCode()
{solAssert(m_currentFunction, "");unsigned stackSurplus = 0;Block const* codeBlock = nullptr;vector<VariableDeclaration const*> addedVariables;m_modifierDepth++;if (m_modifierDepth >= m_currentFunction->modifiers().size()){solAssert(m_currentFunction->isImplemented(), "");codeBlock = &m_currentFunction->body();}else{ASTPointer<ModifierInvocation> const& modifierInvocation = m_currentFunction->modifiers()[m_modifierDepth];// constructor call should be excludedif (dynamic_cast<ContractDefinition const*>(modifierInvocation->name()->annotation().referencedDeclaration))appendModifierOrFunctionCode();else{ModifierDefinition const& modifier = m_context.functionModifier(modifierInvocation->name()->name());CompilerContext::LocationSetter locationSetter(m_context, modifier);solAssert(modifier.parameters().size() == modifierInvocation->arguments().size(), "");for (unsigned i = 0; i < modifier.parameters().size(); ++i){m_context.addVariable(*modifier.parameters()[i]);addedVariables.push_back(modifier.parameters()[i].get());compileExpression(*modifierInvocation->arguments()[i],modifier.parameters()[i]->annotation().type);}for (VariableDeclaration const* localVariable: modifier.localVariables()){addedVariables.push_back(localVariable);appendStackVariableInitialisation(*localVariable);}stackSurplus =CompilerUtils::sizeOnStack(modifier.parameters()) +CompilerUtils::sizeOnStack(modifier.localVariables());codeBlock = &modifier.body();}}if (codeBlock){m_returnTags.push_back(m_context.newTag());codeBlock->accept(*this);solAssert(!m_returnTags.empty(), "");m_context << m_returnTags.back();m_returnTags.pop_back();CompilerUtils(m_context).popStackSlots(stackSurplus);for (auto var: addedVariables)m_context.removeVariable(*var);}m_modifierDepth--;
}

http://www.ppmy.cn/news/883752.html

相关文章

【Tinymce富文本】如何实现disabled和readonly效果

一&#xff1a;项目产品需求 提前说一下 “tinymce”: “^5.9.2”,我用的5.9版本哈 要实现根据条件判断这个富文本是否可以编辑 这里我们最开始尝试了一下:disabled“true” 和readonly都没有用 &#xff0c;刷新以后全部失效 然后google一番&#xff0c;并且看了官方文档都没有…

动态调整对话框属性(去掉标题栏,去掉边框,修改类似成Border:NONE样式)

// 动态修改对话框属性&#xff0c;去掉对话框标题栏&#xff0c;设置Border为NONE属性。 if(dlg.GetSafeHwnd()) {dlg.ModifyStyle(WS_CAPTION, 0); // 去掉标题栏dlg.ModifyStyleEx(WS_EX_DLGMODALFRAME, 0); // 去掉边框dlg.SetWindowPos(NULL, 0, 0, 0, 0, SWP_NOSIZE | S…

v-model修饰符.lazy详解

官网相关内容&#xff1a; 在默认情况下&#xff0c;v-model 在 input 事件中同步输入框的值与数据 (除了 上述 IME 部分)&#xff0c;但你可以添加一个修饰符 lazy &#xff0c;从而转变为在 change 事件中同步&#xff1a; <!-- 在 "change" 而不是 "inp…

WS_BORDER以及WS_EX_CLIENTEDGE的研究

每次研究一点东西&#xff0c;都没有记录&#xff0c;下次需要的时候又浪费时间重新去研究。本人就是个没有总结习惯的人&#xff0c;这样等于没用积累&#xff0c;实在不行。 这2天研究了一下WS_BORDER和WX_EX_CLIENTEDGE的不同&#xff0c;以及在动态创建&#xff0c;和窗口…

【solidity】函数修饰器(Function Modifiers)

修饰符可用于以声明方式更改函数的行为。例如&#xff0c;您可以使用修饰符在执行函数之前自动检查条件。 检查调用者权限 // SPDX-License-Identifier: GPL-3.0 pragma solidity >0.7.0 <0.9.0;contract Owned {address public owner;uint256 public account;construc…

ModifyStyle()调用不起作用

现象&#xff1a;在使用WTL做CListBox子类化时&#xff0c;使用 ModifyStyle(0, LBS_OWNERDRAWFIXED)不起作用; 原因&#xff1a;并不是所有的风格都可以动态利用ModifyStyle/ModifyStyleEx()函数增加和去除&#xff0c;有些风格比如 LBS_HASSTRINGS | LBS_OWNERDRAWFIXED| LB…

$(...).modal is not a function的解决办法

最近在练习ssm框架的时候遇到了一个小问题&#xff0c;无法识别bootstrap.min.js的modal方法&#xff0c;控制台报错$(...).modal is not a function。 于是测试测试了一下 发现可以正常弹框&#xff0c;然后又检查了id是不是写错了&#xff0c;也没写错。突然没了头绪。为什么…

user-modify

最近入职开始工作了&#xff0c;碰到一个很有意思的问题&#xff0c;后端传回来的文章内容<a>标签内的内容可以编辑。 很不理解&#xff0c;我从后端取到的数据这该怎么改呢。 问了同事&#xff08;导师&#xff09;&#xff0c;让我试试contenteditable属性&#xff0c;…