-
Notifications
You must be signed in to change notification settings - Fork 2.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Wtf Solidity japanese 101 Lesson 15 (Last Lesson) #774
Changes from 8 commits
701adbd
fc127b6
db8583f
9e3367a
a77bbad
223cae2
615059f
9c1f758
37995e8
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -73,6 +73,6 @@ | |
|
||
|
||
|
||
## Summary | ||
## まとめ | ||
|
||
この章では、Solidityの`mapping`型を紹介しました。ここまでで、一般的な変数型の全ての種類について学んできましたよ。 |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
// SPDX-License-Identifier: MIT | ||
pragma solidity ^0.8.21; | ||
contract Events { | ||
// define _balances mapping variable to record number of tokens held at each address | ||
//(各アドレスで保有されているトークン数を記録するmapping変数として_balanceを定義します) | ||
mapping(address => uint256) public _balances; | ||
|
||
// define Transfer event to record transfer address, receiving address and transfer number of a transfer transfaction | ||
//(転送を行うトランザクションの送信元アドレスや受信アドレス、転送数を記録する為のTransferイベントを定義します) | ||
event Transfer(address indexed from, address indexed to, uint256 value); | ||
|
||
|
||
// define _transfer function,execute transfer logic | ||
//(_transfer関数を定義して、転送ロジックを実行する) | ||
function _transfer( | ||
address from, | ||
address to, | ||
uint256 amount | ||
) external { | ||
|
||
_balances[from] = 10000000; // give some initial tokens to transfer address(送信元アドレスにいくつかの初期トークンを付与します) | ||
|
||
_balances[from] -= amount; // "from" address minus the number of transfer("from"アドレスから転送数を減算します) | ||
_balances[to] += amount; // "to" address adds the number of transfer("to"アドレスに転送数を加算します) | ||
|
||
// emit event | ||
emit Transfer(from, to, amount); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,106 @@ | ||
--- | ||
title: 12. Events(イベント) | ||
tags: | ||
- solidity | ||
- basic | ||
- wtfacademy | ||
- event | ||
--- | ||
|
||
# WTF Solidity Tutorial: 12. Events(イベント) | ||
|
||
最近、Solidity の学習を再開し、詳細を確認しながら「Solidity 超シンプル入門」を作っています。これは初心者向けのガイドで、プログラミングの達人向けの教材ではありません。毎週 1〜3 レッスンのペースで更新していきます。 | ||
|
||
僕のツイッター:[@0xAA_Science](https://twitter.com/0xAA_Science)|[@WTFAcademy\_](https://twitter.com/WTFAcademy_) | ||
|
||
コミュニティ:[Discord](https://discord.gg/5akcruXrsk)|[Wechat](https://docs.google.com/forms/d/e/1FAIpQLSe4KGT8Sh6sJ7hedQRuIYirOoZK_85miz3dw7vA1-YjodgJ-A/viewform?usp=sf_link)|[公式サイト wtf.academy](https://wtf.academy) | ||
|
||
すべてのソースコードやレッスンは github にて公開: [github.com/AmazingAng/WTFSolidity](https://github.com/AmazingAng/WTFSolidity) | ||
|
||
----- | ||
|
||
この章では、ERC20のトークンを例として使い、Solidityにおける`event`を紹介します。 | ||
|
||
## Events(イベント) | ||
`solidity`におけるイベントは`EVM`(Ethereum Virtual Machine)に格納されるトランザクションログです。それらは関数が呼び出される際に発生し、コントラクトアドレスでアクセスできます。イベントは2つの性質を持っています : | ||
|
||
- 即応性がある: アプリケーション(例えば[`ether.js`](https://learnblockchain.cn/docs/ethers.js/api-contract.html#id18))は`RPC`インターフェースを通して、購読し(検知)し監視(追跡)することができ、フロントエンドで応答することができる。 | ||
- 経済的である: イベントにデータを格納するのは安く、大体一度に2,000`gas`位掛かります。それに比べて、オンチェーンに新しい変数を格納するのは少なくとも20,000`gas`かかります。 | ||
|
||
### Declare events(イベントを宣言する) | ||
イベントは`event`キーワードで宣言され、イベントの名前、そして記録されるそれぞれのパラメーターの型と名前が続きます。例として`ERC20`トークンのコントラクトから`Transfer`イベントを取り上げてみましょう。 | ||
```solidity | ||
event Transfer(address indexed from, address indexed to, uint256 value); | ||
``` | ||
`Transfer`イベントは3つのパラメーターを記録します: `from`と`to`、そして`value`です。それぞれ、それらはトークンが送信される元のアドレス、受信アドレス、そして送金されるトークンの数に相当しています。パラメーターの`from`と`to`は`indexed`キーワードで印が付けられており、`topics`として知られている特別なデータ構造に格納され、プログラムによって容易に照会されます。 | ||
|
||
### Emit events(イベントを発生させる) | ||
|
||
関数の中でイベントを発生させることが出来ます。次に続く例において、`_transfer()`関数が呼び出される度に、`Transfer`イベントは発生させられ、対応するパラメーターが記録されます。 | ||
```solidity | ||
// define _transfer function, execute transfer logic(_transfer関数を定義し、転送ロジックを実行する) | ||
function _transfer( | ||
address from, | ||
address to, | ||
uint256 amount | ||
) external { | ||
|
||
_balances[from] = 10000000; // give some initial tokens to transfer address(転送元アドレスに幾らかの初期トークンを付与する) | ||
|
||
_balances[from] -= amount; // "from" address minus the number of transfer("from"アドレスは転送させる数を減算する) | ||
_balances[to] += amount; // "to" address adds the number of transfer("to"アドレスは転送させる数を加算する) | ||
|
||
// emit event(イベントを発生させる) | ||
emit Transfer(from, to, amount); | ||
} | ||
``` | ||
|
||
## EVM Log | ||
|
||
EVMはSolidityのイベントを保管する為に`Log`を使用します。各ログは2つの部品を含んでいます: `topics`と`data`です。 | ||
|
||
![](img/12-3.png) | ||
|
||
### `Topics` | ||
|
||
`Topics`はイベントを記述する為に使用されます。各イベントは最大4つの`topics`を含んでいます。典型的に、最初の`topic`はイベントハッシュです: イベントシグネチャのハッシュです。`Transfer`イベントのイベントハッシュは次のように計算されます: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 「最大で4つの |
||
|
||
```solidity | ||
keccak256("Transfer(addrses,address,uint256)") | ||
|
||
//0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef | ||
``` | ||
|
||
イベントハッシュに加えて、`topics`は3つの`indexed`パラメーターを含んでいます。例えば、`Transfer`イベントにおける`from`と`to`パラメーターのようなものです。匿名イベント(イベントの記述に"anonymous"を付与)は特別です: イベント名を持ちませんし、最大で4つの`indexed`パラメーターを持つことが出来ます。 | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ここも同じく「最大で3つの |
||
|
||
`indexed`パラメーターはイベントの索引付けされた"key"として理解することが出来ます。そしてそれは容易にプログラムによって照会されることが出来ます。各`indexed`パラメーターのサイズは32バイトです。`array`や`string`のような32バイトよりも大きなパラメーターに関しては、基礎となるデータのハッシュが格納されます。(非基本型(例:配列や構造体)の場合、そのデータ自体ではなく、そのデータのKeccak-256ハッシュが格納される) | ||
|
||
### `Data` | ||
|
||
索引の無いパラメーターはログの`data`セクションに格納されます。それらはイベントの"value"として解釈され、直接取得することは出来ません。しかしそれらはより大きいサイズのデータを格納することが出来ます。それゆえに、`data`セクションは、`array`や`string`と言った複雑なデータ構造を格納する為に使用されます。その上、`data`は`topic`に較べてより少ないガスを消費します。 | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 中文の 「因此一般 data 部分可以用来存储复杂的数据结构,例如数组和字符串等等,因为这些数据超过了256比特,即使存储在事件的 topics 部分中,也是以哈希的方式存储。」の翻訳が抜けてる感じがするな There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 「(非基本型(例:配列や構造体)の場合、そのデータ自体ではなく、そのデータのKeccak-256ハッシュが格納される)」 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
@thurendous There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
@thurendous |
||
|
||
## Remix Demo | ||
`Event.sol`コントラクトを例として見てみましょう。 | ||
|
||
1. `Event`コントラクトをデプロイします。 | ||
|
||
2. `Transfer`イベントを発生させる為に、`_transfer`関数を呼び出します。 | ||
|
||
![](./img/12-1_ja.png) | ||
|
||
3. 発生させられたイベントをチェックする為に、トランザクションの詳細情報を確認します。 | ||
|
||
![](./img/12-2_ja.png) | ||
|
||
### Etherscanでイベントを照会する | ||
|
||
イーサスキャン(Etherscan)はイーサリアムのブロックチェーン上にあるトランザクションやスマートコントラクト、それ以上のことを見ることが出来るブロックエクスプローラーです。先ず、私はRinkebyやGoerliなどのイーサリアムテストネットにコントラクトをデプロイしました。次に、私は100トークン送金する為に`_transfer`関数を呼び出しました。その後で、トランザクションの詳細情報を`etherscan`で確認することが出来ます:[URL](https://rinkeby.etherscan.io/tx/0x8cf87215b23055896d93004112bbd8ab754f081b4491cb48c37592ca8f8a36c7) | ||
|
||
イベントの詳細情報を確認する為に`Logs`ボタンをクリックします: | ||
|
||
![details of event](./img/12-3_ja.png) | ||
|
||
`Topics`には3つの要素があります: `[0]`はイベントのハッシュであり、`[1]`と`[2]`は`Transfer`イベントで定義されている`indexed`パラメーターです(即ち、`from`と`to`です)。`Data`にある要素は索引の無いパラメーター`amount`です。 | ||
|
||
## まとめ | ||
このレクチャーでは、`Solidity`におけるイベントを使用し、照会する方法について紹介しました。`Dune Analytics`のような、多くのオンチェーンの分析ツールは、solidityのイベントに基いています。 |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
// SPDX-License-Identifier: MIT | ||
pragma solidity ^0.8.13; | ||
|
||
/* Inheritance tree(継承ツリー): | ||
God | ||
/ \ | ||
Adam Eve | ||
\ / | ||
people | ||
*/ | ||
|
||
contract God { | ||
event Log(string message); | ||
|
||
function foo() public virtual { | ||
emit Log("God.foo called"); | ||
} | ||
|
||
function bar() public virtual { | ||
emit Log("God.bar called"); | ||
} | ||
} | ||
|
||
contract Adam is God { | ||
function foo() public virtual override { | ||
emit Log("Adam.foo called"); | ||
} | ||
|
||
function bar() public virtual override { | ||
emit Log("Adam.bar called"); | ||
super.bar(); | ||
} | ||
} | ||
|
||
contract Eve is God { | ||
function foo() public virtual override { | ||
emit Log("Eve.foo called"); | ||
} | ||
|
||
function bar() public virtual override { | ||
emit Log("Eve.bar called"); | ||
super.bar(); | ||
} | ||
} | ||
|
||
contract people is Adam, Eve { | ||
function foo() public override(Adam, Eve) { | ||
super.foo(); | ||
} | ||
|
||
function bar() public override(Adam, Eve) { | ||
super.bar(); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,74 @@ | ||
// SPDX-License-Identifier: MIT | ||
pragma solidity ^0.8.21; | ||
|
||
// Inheritance contract(継承コントラクト) | ||
contract Grandfather { | ||
event Log(string msg); | ||
|
||
// Apply inheritance to the following 3 functions: hip(), pop(), Grandfather(),then log "Grandfather". | ||
//(継承を次の3つの関数に対して適用します: hip()、pop()、Grandfather()、そしてログは"Grandfather"です。) | ||
function hip() public virtual{ | ||
emit Log("Grandfather"); | ||
} | ||
|
||
function pop() public virtual{ | ||
emit Log("Grandfather"); | ||
} | ||
|
||
function grandfather() public virtual { | ||
emit Log("Grandfather"); | ||
} | ||
} | ||
|
||
contract Father is Grandfather{ | ||
// Apply inheritance to the following 2 functions: hip() and pop(), then change the log value to "Father". | ||
//(継承を次の2つの関数に対して適用します: hip()、pop()、そしてログの値を"Father"に変更します。) | ||
function hip() public virtual override{ | ||
emit Log("Father"); | ||
} | ||
|
||
function pop() public virtual override{ | ||
emit Log("Father"); | ||
} | ||
|
||
function father() public virtual{ | ||
emit Log("Father"); | ||
} | ||
} | ||
|
||
contract Son is Grandfather, Father{ | ||
// Apply inheritance to the following 2 functions: hip() and pop(), then change the log value to "Son". | ||
//(継承を次の2つの関数に対して適用します: hip()、pop()、そしてログの値を"Son"に変更します。) | ||
function hip() public virtual override(Grandfather, Father){ | ||
emit Log("Son"); | ||
} | ||
|
||
function pop() public virtual override(Grandfather, Father) { | ||
emit Log("Son"); | ||
} | ||
|
||
function callParent() public{ | ||
Grandfather.pop(); | ||
} | ||
|
||
function callParentSuper() public{ | ||
super.pop(); | ||
} | ||
} | ||
|
||
// Applying inheritance to the constructor functions | ||
//(コンストラクター関数に対して継承を適用します) | ||
abstract contract A { | ||
uint public a; | ||
|
||
constructor(uint _a) { | ||
a = _a; | ||
} | ||
} | ||
|
||
contract B is A(1) { | ||
} | ||
|
||
contract C is A { | ||
constructor(uint _c) A(_c * _c) {} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
// SPDX-License-Identifier: MIT | ||
pragma solidity ^0.8.21; | ||
|
||
contract Base1 { | ||
modifier exactDividedBy2And3(uint _a) virtual { | ||
require(_a % 2 == 0 && _a % 3 == 0); | ||
_; | ||
} | ||
} | ||
|
||
contract Identifier is Base1 { | ||
|
||
// Calculate the value of a number divided by 2 and divided by 3, respectively, but the parameters passed in must be multiples of 2 and 3 | ||
//(2で除算された数値と3で除算された数値をそれぞれ計算しますが、渡される引数は2と3の倍数でなければなりません) | ||
function getExactDividedBy2And3(uint _dividend) public exactDividedBy2And3(_dividend) pure returns(uint, uint) { | ||
return getExactDividedBy2And3WithoutModifier(_dividend); | ||
} | ||
|
||
// Calculate the value of a number divided by 2 and divided by 3, respectively | ||
//(2で除算された数値と3で除算された数値をそれぞれ計算します) | ||
function getExactDividedBy2And3WithoutModifier(uint _dividend) public pure returns(uint, uint){ | ||
uint div2 = _dividend / 2; | ||
uint div3 = _dividend / 3; | ||
return (div2, div3); | ||
} | ||
|
||
|
||
// Rewrite the modifier: when not rewriting, enter 9 to call getExactDividedBy2And3, it will be reverted because it cannot pass the check | ||
// Delete the following three lines of comments and rewrite the modifier function. At this time, enter 9 to call getExactDividedBy2And3, and the call will be successful. | ||
//(修飾子を書き換える: 書き換えない時には、getExactDividedBy2And3を呼び指す際に9を入れます。そうすればチェックを通らないので、リバーとされます。) | ||
//(次の3行のコメントを消して、修飾子関数を書き換えてください。この時にはgetExactDividedBy2And3に9を代入すれば、呼び出しは成功するでしょう。) | ||
|
||
// modifier exactDividedBy2And3(uint _a) override { | ||
// _; | ||
// } | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
これはなんですっけ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@thurendous
<code> </code>
これはマークダウンのスペースですね。
タイトルの和訳をセットするに当たって、PCで閲覧した時に英文タイトルの直下に来るように間隔を作成。
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@PolymetisOutis
what is this?
<code> </code>(定数と不変)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
spaceだよね。これ要るんだっけ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@thurendous
「 タイトルの和訳をセットするに当たって、PCで閲覧した時に英文タイトルの直下に来るように間隔を作成。」だよ。
レイアウトを整えてるんだね!