火星链 火星链
Ctrl+D收藏火星链

TOR:探索以太坊合约委托调用(DelegateCall)

作者:

时间:1900/1/1 0:00:00

译文出自:登链翻译计划

译者:翻译小组

校对:Tiny熊

在本文中,我们看看如何调用另一个合约的函数,并更深入讨论delegatecall委托调用。

有时,需要在编写以太坊智能合约代码中,与其他合约进行交互。在Solidity中,有几种方法可以实现此目标:

如果知道目标合约的ABI,可以直接使用函数签名

假设已经部署了一个简单的合约,称为“Storage”,该合约允许用户保存val。

pragmasolidity^0

functionsetValue(uintv)public{val=v;}}

现在我们部署另一个称为“Machine”的合约,它是“Storage”合约的调用方。“Machine”引用“Storage”合约并更改其val。

pragmasolidity^0

functionsaveValue(uintx)publicreturns(bool){s

functiongetValue()publicviewreturns(uint){returns

}

在此案例中,我们知道Storage合约的ABI及其地址,以便我们可以使用该地址初始化现有的Storage合约,而ABI的作用是告诉我们如何调用Storage合约的函数。可以看到Machine合约调用了Storage

);describe('#saveValue()',()=>{it('shouldsuccessfullysavevalue',async()=>{awaitMachine

印度电商巨头Meesho宣布将探索区块链、元宇宙和NFT用例:5月28日消息,印度社交电商独角兽Meesho宣布将探索区块链、元宇宙、Web3 和 NFT用例重新定义线上销售。Meesho联合创始人兼首席技术官Sanjeev Barnwal表示,Meesho希望让买卖双方在元宇宙中互动,推出元宇宙业务只是时间和时机问题,但现在已经开始探索Web3及其扩展用例。

如果元宇宙成为常态,电商公司其实可以做很多事情,比如在线模拟线下购买体验。从根本上说,将会改变用户体验游戏规则。

值得一提的是,雷军旗下的顺为资本是Meesho公司股东,且参与过该公司早期阶段融资。[2022/5/28 3:47:13]

);});});

测试通过了!

Contract:MachineAfterinitalize#saveValue()?shouldsuccessfullysavevalue(56ms)1passing(56ms)

如果不知道目标合约的ABI,请使用call或delegatecall

但是,如果调用者(在本例中为“Machine”合约)不知道目标合约的ABI,该怎么办?

其实,我们仍然可以使用call()和delegatecall()来调用目标合约的函数。

在解释以太坊Solidity的call()和delegatecall()之前,了解EVM如何保存合约变量对于了解call()和delegatecall()会有所帮助。

EVM如何将字段变量保存到存储

在以太坊中,有两种空间可以保存合约的字段变量。一个是“内存”,另一个是“存储”。而且,“foo”保存到存储意味着“foo”的值会永久记录到区块链状态中。

外媒:PayPal已就稳定币与开发商进行探索性讨论:PayPal正在探索推出稳定币。四个了解情况的消息人士称,PayPal已经与该行业的一些稳定币协议开发商进行讨论,表明这家支付巨头可能倾向于与第三方公司合作。知情人士说,Avalanche区块链背后的团队Ava Labs是与PayPal就稳定币开发进行讨论的组织之一。目前尚不清楚讨论中还涉及了哪些其他协议。PayPal发言人称:“PayPal继续探索数字货币、数字金融服务基础设施的潜力,以及我们如何作为该领域值得信赖的合作伙伴来帮助增强数字商务。”(The Block)[2021/5/4 21:21:24]

那么,单个合约中的如此多的变量又是怎样让彼此不重叠呢?EVM将插槽号分配给字段变量。

contractSample1{uint256first;//slot0uint256second;//slot1}

ImageforpostEVM使用插槽保存字段变量

因为first在Sample1合约中最先声明,所以分配了0个插槽。每个不同的变量都通过其插槽号来区分。

在EVM中,智能合约存储中具有2256个插槽,每个插槽可以保存32字节大小的数据。

如何调用智能合约函数

像Java,Python这样的通用编程代码一样,Solidity函数可以看作是一组命令。当我们说“函数被调用”时,这意味着我们将特定的上下文(如参数)注入到该组命令(函数)中,并且在此上下文中一个接一个地执行命令。

函数、命令组、地址空间可以通过其名称找到。

在以太坊函数中,调用可以用字节码表示,使用432*N个字节表达。这个字节码由两部分组成。

灰度探索增加新的加密货币投资产品 包含 AAVE、DOT 等 23 个币种:据官方消息,灰度投资(Grayscale Investments)宣布考虑增加新的加密货币投资产品。当前正在考虑的数字资产包括AAVE、BAT、ADA、LINK、COMP、ATOM、MANA、EOS、FIL、FLOW、LPT、MKR、DOT、RSR、STX、SUSHI、SNX、XTZ、GRT、UNI及YFI。

不过灰度创建结构性投资产品需要进行审查和考虑并受严格的内部控制,需要足够安全的托管安排和监管考虑的制约,因此,不能保证上述列出的资产会推出相应的投资产品。灰度投资目前为获得认证的投资者提供 8 个单一资产投资信托和 1 个多元化基金。[2021/2/26 17:57:15]

函数选择器:这是函数调用字节码的前4个字节。函数选择器是通过对目标函数的名称加上其参数类型(不包括空格)进行哈希取前4个字节得到,例如bytes4(keccak-256(“saveValue(uint)”))。基于此函数选择器,EVM可以决定应在合约中调用哪个函数。

函数参数:将参数的每个值转换为固定长度为32bytes的十六进制字符串。如果有多个参数,则串联在一起。

如果用户将此432*N字节字节代码传递给交易的数据字段。EVM可以找到应执行的函数,然后将参数注入该函数。

用测试用例解释DelegateCall

上下文

当我们谈论智能合约函数的调用方式时,有一个“上下文”一词。实际上,“上下文”一词在软件中是很笼统的概念,其含义根据场合不同有所改变。

当我们谈论程序的执行时,我们可以说“上下文”是指执行时所有环境(如变量或状态)。例如,在执行程序“A”时,执行该程序的用户名是“zeroFruit”,则用户名“zeroFruit”可以是程序“A”的上下文。

日本著名传统金融机构进军交易所 探索房地产代币化:日本著名传统金融机构东海东京金融控股有限公司(Tokai Tokyo Financial Holdings)正努力在日本启动数字安全交易所。

日经新闻称,Tokai Tokyo正在投资区块链公司Hash Dash Holdings,以利用该公司来发展交易所。Hash Dash旨在将区块链技术应用于金融行业,以发行数字证券并在智能手机上提供交易服务。 计划中的交易所将进行代币化房地产交易,Tokai Tokyo、Hash Dash和ICHX TECH将通过对日本房地产进行数字化来探索这一领域,并在其投资的新加坡数字安全交易所iSTOX进行交易。

Tokai Tokyo希望将其计划中的日本交易所与iSTOX连接起来,以便投资者可以在任一交易所无缝地买卖数字化资产,从而使日本公司有可能在新加坡的交易所筹集资金。(Cointelegraph)[2020/6/26]

在以太坊智能合约中,有很多上下文,其中一个代表性的事情是谁执行这个合约。你可能会在很多Solidity代码中看到msg

..

functionaddValuesWithCall(addresscalculator,uint256a,uint256b)publicreturns(uint256){(boolsuccess,bytesmemoryresult)=calculator

}

下面是目标合约Calculator,它也有calculateResult和user。

pragmasolidity^0

}

声音 | 王锋:恒生研究院目前主要集中在区块链等方面的探索:恒生电子副总裁王锋日前接受中国证券报记者采访时表示,2016年,恒生成立研究院。恒生研究院主要研究3-5年左右可以为金融行业使用的基础性技术。目前主要集中在AI、大数据、云、区块链等方面的探索。(中证网)[2019/12/15]

测试addValuesWithCall

下面是addValuesWithCall的测试代码。需要测试的有:

由于上下文位于“Calculator”而非“Machine”上,因此add结果应保存到“Calculator”合约存储中

因此,Calculator的calculateResult应该为3,而user的地址应该设置为Machine的地址。

并且Machine的calculateResult应该为0,user为零地址。

describe('#addValuesWithCall()',()=>{letCalculator;beforeEach(async()=>{Calculator=awaitCalculatorFactory

);it('shouldsuccessfullyaddvalueswithcall',async()=>{constresult=awaitMachine

);});

按预期通过了所有测试:

Contract:MachineAfterinitalize#addValuesWithCall()?shouldsuccessfullyaddvalueswithcall(116ms)1passing(116ms)

测试addValuesWithDelegateCall

下面是我们的addValuesWithCall测试代码。我们需要测试的有:

由于上下文位于“Machine”而非“Calculator”上,因此add结果应保存到“Machine”存储中。

因此,Calculator的calculateResult应该为0,而user的地址应为0地址。

而Machine的calculateResult应为3,而user的则为用户地址。

describe('#addValuesWithDelegateCall()',()=>{letCalculator;beforeEach(async()=>{Calculator=awaitCalculatorFactory

);it('shouldsuccessfullyaddvalueswithdelegatecall',async()=>{constresult=awaitMachine

);});

但是失败了!什么呢?“562046206989085878832492993516240920558397288279”来自哪里?

0passing(236ms)1failing1)Contract:MachineAfterinitalize#addValuesWithDelegateCall()shouldsuccessfullyaddvalueswithdelegatecall:AssertionError:expected'562046206989085878832492993516240920558397288279'toequal'3'expected-actual-5620462069890858788324929935162409205583972882793

如前所述,每个字段变量都有其自己的插槽。当我们委托调用Calculator时,上下文位于Machine上,但是插槽编号基于Calculator。

因此,由于Calculator用calculateResult覆盖了Storage地址,而user覆盖了calculateResult,因此测试失败。

基于此知识,我们可以找到“562046206989085875878832492993516240920558397288279”的来源。它是EOA的十进制版本。

Imageforpost“Calculator”合约字段变量将覆盖“Machine”合约字段变量

因此,要解决此问题,我们需要更改“Machine”字段变量的顺序。

contractMachine{uint256publiccalculateResult;addresspublicuser;Storagepublics;...}

最后,测试通过了!

Contract:MachineAfterinitalize#addValuesWithDelegateCall()?shouldsuccessfullyaddvalueswithdelegatecall(106ms)1passing(247ms)

总结一下

在本文中,我们已经看到了如何从合约中调用另一个合约的函数。

如果我们知道目标函数的ABI,就可以直接使用目标函数签名

如果我们不知道目标函数的ABI,可以使用call()或delegatecall()。但是在delegatecall()的情况下,我们需要关心字段变量的顺序。

源代码

如果你想自己进行测试,可以在此代码库中找到代码。

本翻译由CellNetwork赞助支持。

来源:https://medium.com/coinmonks/delegatecall-calling-another-contract-function-in-solidity-b579f804178c

参考资料

登链翻译计划:https://github.com/lbc-team/Pioneer

翻译小组:https://learnblockchain.cn/people/412

Tiny熊:https://learnblockchain.cn/people/15

ABI:https://learnblockchain.cn/docs/solidity/abi-spec.html

此代码库:https://github.com/zeroFruit/upgradable-contract/tree/feat/delegatecall

CellNetwork:https://www.cellnetwork.io/?utm_souce=learnblockchain

本文来源于非小号媒体平台:

登链社区

现已在非小号资讯平台发布105篇作品,

非小号开放平台欢迎币圈作者入驻

入驻指南:

/apply_guide/

本文网址:

/news/9569963.html

免责声明:

1.资讯内容不构成投资建议,投资者应独立决策并自行承担风险

2.本文版权归属原作所有,仅代表作者本人观点,不代表非小号的观点或立场

上一篇:

币安为何推出第三条链?这对BNB意味着什么?

标签:CALALLTORVALcalo币最新消息SATSALL币HistoriaInterValue

币安交易所app下载热门资讯
CIR:金色说明书 | 火币生态链HECO轻社交DeFi项目CircleSwap挖矿教程(移动端和网页端)

为了方便投资者及时了解热门项目的相关信息和使用流程,金色财经推出了“金色说明书”系列教程。本期由金色财经和火币生态链的circleswap联合推出.

1900/1/1 0:00:00
LOEx进取区1月06日09:00上线SUNT

尊敬的LOEx用户:LOEx进取区上线SUNT!并开放SUNT/USDT交易对具体时间如下:充提币:1月06日10:00交易对:1月06日09:00注:未到充值开放时间请勿提前充值.

1900/1/1 0:00:00
FIL:Gate.io 已经完成当日为用户转化分发1.3万枚FIL代币公告

Gate.io今日已经根据用户FIL6持仓情况完成FIL分发,总计约1.3万枚,用户可在账户账单明细中查看详情.

1900/1/1 0:00:00
DFT:关于ZT网站交易系统恢复的公告

尊敬的ZT用户:????ZT现已完成系统升级,WEB及APP端交易均已恢复正常使用。升级期间给您造成的不便,敬请谅解!温馨提示:新资产上线时,通常伴随较为剧烈的价格波动,请评估风险后谨慎参与交易.

1900/1/1 0:00:00
OIN:炒币教父: 短期回调 后市行情怎么看?

早间5时比特币在33860一线遇压回调,震荡下挫,触底反弹,午间12时涨至33591一线受阻延续震荡下行,触底二次反弹向上试探遇压回调,随后短线下挫跌破32000关口,跌至31715一线反弹.

1900/1/1 0:00:00
BTC:沸腾 2020:前三十加密货币趋势分析 异军突起与掉队者是谁

前言:2020是沸腾的一年。从312暴跌到比特币新高,从DeFi狂潮到Filecoin上线。“一个牛市可以赚10年的钱”,但也有人合约亏损酿成惨案.

1900/1/1 0:00:00