许多开发人员在开发Solidity之前实现了Java,Go,Python 感觉就像回到80年代后期的DeLorean一样。 但是Solidity的稳定性非常有限。
我正在使用名为#ScriptIt的队长的NodeJS oracle用于以下用例:
1. 新用户获得256分
2. 每次新呼叫,用户的积分将减少log2
船长将直接从Docker容器中的Solidity运行NodeJS调用,并将结果返回给您的合约。
智能合约
异步联系将派生自usingCaptainJS,其中包括异步调用和回调功能。
要在回调发生时记住异步调用,您需要一个JobCounter以及作业ID和发件人地址的映射:
uint JobCounter = 0;
mapping (uint =》 address) JobToSenderMap;
事件
在以太坊中,当同步事务处于挂起状态,事务只有失败或成功两种状态。异步事务将要求发出事件时,通知用户事务是否挂起、成功或失败。
因此,您定义三个这些事件,并且每个事件至少应包含发件人地址:
event GetPoints_Success(address Sender, uint Points);
event GetPoints_Pending(address Sender);
event GetPoints_Failed(address Sender, string ErrorMsg);
函数
以太坊的默认模式是每个用户调用一个合约函数,并支付在一个同步事务环境中执行代码所需的GAS。
但现在我们有了一个异步事务环境, 这意味着在同步函数调用终止后将需要额外的气体。
因此,您的函数必须是payable,您的首次检查必须是验证用户是否转移了足够的额外gas费用:
uint GasRequired = DEFAULT_GAS_UNITS * tx.gasprice + 70 szabo;
require(msg.value 》= GasRequired, “please send some extra gas.。.”);
在这个演示用例中,我们将要求使用usingCaptainJS中定义的默认gas单位乘以当前的交易gas价格加上70 Szabo的交易费。
一旦用户输送了足够gas,你可以根据船长在GitHub上的描述来调用mathjs的log2函数:
Run(
JobCounter,
concat(“math:log2(”,uintToString(PointsPerUser[msg.sender]), “)”),
“”, “”, 1, DEFAULT_GAS_UNITS, tx.gasprice
);
emit GetPoints_Pending(msg.sender);
在调用Run(。..)之后,您必须发出pending事件。如果调用Run(。..)失败,则同步调用将失败。
回调
一旦船长计算了用户积分的log2值,他就会通过调用CaptainsResult函数将结果发送回合约。通过仅添加CaptainsOrdersAllowed确保只有队长调用此功能。
确保在函数结束时发出成功事件。
function CaptainsResult(uint JobCounter, string Log2Result)
external onlyCaptainsOrdersAllowed {
// the return of the async call
address sender = JobToSenderMap[JobCounter];
uint Points = StringToUint(Log2Result);
PointsPerUser[sender] = Points;
emit GetPoints_Success(sender, Points);
}
果队长无法调用您提交的代码(也许您的JavaScript代码中有拼写错误),他会通过调用合同的CaptainsError函数通知您。
确保在函数结束时发出失败的事件。
function CaptainsError(uint JobCounter, string ErrorMsg)
external onlyCaptainsOrdersAllowed {
// the return of the async call
address sender = JobToSenderMap[JobCounter];
emit GetPoints_Failed(sender, ErrorMsg);
}
这是完整的代码:
pragma solidity ^0.4.25;
import “。/usingCaptainJS_v2.sol”;
contract AsyncPattern is usingCaptainJS {
// to identify async calls
uint JobCounter = 0;
mapping (uint =》 address) JobToSenderMap;
// demo use case: points per sender
mapping (address =》 uint) PointsPerUser;
event GetPoints_Success(address Sender, uint Points);
event GetPoints_Pending(address Sender);
event GetPoints_Failed(address Sender, string ErrorMsg);
function GetPoints() public payable {
// make sure to have enough gas for the async callback
uint GasRequired = DEFAULT_GAS_UNITS * tx.gasprice + 70 szabo;
require(msg.value 》= GasRequired, “please send some extra gas.。.”);
// remember this call
JobToSenderMap[++JobCounter] = msg.sender;
// now do the math - but mix async + async.。.
// every user has 256 points at the beginning and with every next
// call it is log2 of his points
if(PointsPerUser[msg.sender] == 0) {
// first call!
PointsPerUser[msg.sender] = 256;
emit GetPoints_Success(msg.sender, 256);
}
else {
// every other call
Run(
JobCounter, concat(“math:log2(”, uintToString(PointsPerUser[msg.sender]), “)”),
“”, “”, 1, DEFAULT_GAS_UNITS, tx.gasprice
);
emit GetPoints_Pending(msg.sender);
}
}
function CaptainsResult(uint JobCounter, string Log2Result) external onlyCaptainsOrdersAllowed {
// the return of the async call
address sender = JobToSenderMap[JobCounter];
uint Points = StringToUint(Log2Result);
PointsPerUser[sender] = Points;
emit GetPoints_Success(sender, Points);
}
function CaptainsError(uint JobCounter, string ErrorMsg) external onlyCaptainsOrdersAllowed {
// the return of the async call
address sender = JobToSenderMap[JobCounter];
emit GetPoints_Failed(sender, ErrorMsg);
}
}
相关热词:#区块链
为什么需要MOSFET栅极电阻?MOSFET栅极电阻...
时间:2026-03-05
NTC/PTC/CTR热敏电阻是什么?热敏电阻的使用...
时间:2026-03-05
解析单电阻采样的原理以及注意点
时间:2026-03-05
共源极放大器的设计方法
时间:2026-03-05
关于STM32WL LSE 添加反馈电阻后无法起振的...
时间:2026-03-05
如何直观地判断两级放大器的零点位置呢?
时间:2026-03-05
时序分析基本概念介绍<wire load model&...
时间:2026-03-05
电子元器件解析—电阻
时间:2026-03-05
3PEAK高压零漂放大器契合精密应用
时间:2026-03-05
助力绿色5G数字式电流和功率监测芯片-TPA62...
时间:2026-03-05
未来什么技术占主流?会是区块链吗?
时间:2026-03-05
浅谈区块链下的智能合约
时间:2026-03-05
浅显易懂地揭开.Net生态系统的神秘面纱!
时间:2026-03-05
对技术的执着和美的追求变成了程序员的“诗...
时间:2026-03-05
什么是区块链技术_区块链技术解析
时间:2026-03-05
脑洞大开:玻璃窗也能发电?
时间:2026-03-05
如何布置无人值守变电站?
时间:2026-03-05
SCADA系统懂多少?
时间:2026-03-05
灰尘对光伏电站效率影响究竟多大
时间:2026-03-05
一种基于Agent的智能电网集成优化控制策略
时间:2026-03-05