|
Tact 支持许多专为智能合约使用而定制的 原始数据类型。 不过,使用单独的存储方式往往会变得繁琐,因此有 Structs 和 Messages可以将类型组合在一起。 不过,使用单独的存储方式往往会变得繁琐,因此有 Structs 和 Messages可以将类型组合在一起。3 |1 ]" t3 a" z$ F6 R2 {; e. \1 d: w. g
' x0 w2 l/ s2 g4 [5 M% I% t {
) V B H4 f g7 c, B- `5 Z" @4 _! s! r
结构
/ H" b% B: b! F) }& E8 g; W |结构体可以定义包含多个不同类型字段的复杂数据类型。 它们还可以嵌套。 它们还可以嵌套。! m! `. l1 ?: e' p( \
- struct Point {
) [8 ]0 H+ z0 C2 ^. I- l' q - x: Int as int64;
, N7 |3 a, ^" `" U5 t# Y& t: ?7 Y - y: Int as int64;3 h$ F2 \/ T) r8 m- K
- }
8 D+ L# u5 [& B% M- o# A - $ O8 G% C. `6 [2 K9 \
- struct Line {
9 B6 j q. J ]) _, \ - start: Point;7 c1 h+ E P! E0 y# L
- end: Point;
4 ]% O6 C6 X' j: [ - }
复制代码 结构体还可以包含默认字段,并定义可选类型的字段。 结构体还可以包含默认字段和定义[可选类型]字段(/book/optionals)。 如果您有很多字段,但又不想一直在 new instances 中为它们指定通用值,那么这将非常有用。
* ?7 h; X% ^2 Q& P; O" C) C- struct Params {
; q4 G }+ b% O2 j3 K6 J - name: String = "Satoshi"; // default value( Q7 S: n# D/ G
- # q G8 h1 D* c9 B( O
- age: Int?; // field with an optional type Int?
" o: s( U6 M# k5 Q, ] - // and default value of null; v1 S$ L' E8 R7 N3 \) P$ @
- ( L# p0 k+ z0 Q, I, c9 m$ j- v2 r' ]
- point: Point; // nested Structs
* Y }) J( m; _( s8 f) z3 s - }
复制代码 字段的顺序很重要,因为它与TL-B 模式 中的内存布局一致。 不过,与某些采用手动内存管理的语言不同,Tact 在字段之间没有任何填充。
1 a. m% Z( p8 G. |0 J+ V/ h- s9 o0 t9 @9 M# V# f: m
/ D6 q6 ?( [9 V3 n9 a6 ^) Q! p
+ i1 R0 x3 X) W+ Y5 L9 v消息6 } u8 h2 J2 G5 b
消息中可以包含 结构体:6 L& U! U5 i) |" Z
- struct Point {# ?9 {. L% @5 ]/ K- T3 q
- x: Int;
4 A. t( Y. T4 [3 f! }7 ?) L9 v - y: Int;+ p+ c4 K- L j, z
- }
& ?! @9 W V3 C' r - , c% B' W, Y9 k+ d5 o
- message Add {
- |1 ?# l& W# m2 f$ a - point: Point; // holds a struct Point
' Q' D& d2 \. Q - }
复制代码- // This Message overwrites its unique id with 0x7362d09c& O/ y/ [% [( r2 ~% E$ z4 K3 R
- message(0x7362d09c) TokenNotification {
, |) S5 r& [+ I6 X( v6 n, g - forwardPayload: Slice as remaining;
' d6 z' y5 w) K; X/ g' A - }
复制代码 这对于要处理特定智能合约的某些操作码(如 Jetton standard)的情况非常有用。 该合约能够处理的操作码简表为此处以 FunC 表示。 它们是智能合约的接口。) p5 b8 o( T% V# g0 v
4 n2 r6 f" {; B) z0 ] w/ o
; g0 L/ N: A& @: J* Z5 K
操作: s8 i# B1 G6 J# t8 O' F
实例化* F" f6 O/ F. c* P
创建 Struct 和 Message 实例类似于 function calls,但需要用大括号 {}(大括号)代替小括号 ()指定参数:
$ _- S0 p9 V$ N' b3 k5 v- struct StA {
: |; w2 I1 b6 L3 x* x - field1: Int;
" O7 C: V. ?% T - field2: Int;
8 K4 p0 x- \/ }! x+ b/ u6 S - }% h3 d: J4 B3 h; e8 ~
- % o& J4 J. ` y* ^( K6 E% k
- message MsgB {
4 `! D) k! K5 s - field1: String;; y1 \9 O& f K* {( q
- field2: String;
5 u, G/ y4 H" ^/ T2 L( s - }
9 D( N: S, N- h9 M/ W) l - : y6 g2 L7 w0 o( h' [
- fun example() {
. Z1 A, Y+ M4 W& Y5 Y) I9 u - // Instance of a Struct StA! e8 c8 }& r2 p8 E' }0 I
- StA{
' k; A; H7 J' O. {# Z - field1: 42,3 N( g. ]! q5 E, \
- field2: 68 + 1, // trailing comma is allowed& F% I+ W: H1 D: ]/ ?8 q0 F. a
- };: P$ H+ U7 B' y* y/ L) y6 I, m
- + X$ d+ }# G# S B/ ?
- // Instance of a Message MsgB, S* _0 d; W) ^8 \9 y
- MsgB{& R- W! x$ @9 b/ n# _% x' i
- field1: "May the 4th",' q0 V) q- j/ q: C4 V9 L
- field2: "be with you!", // trailing comma is allowed
2 f; Z! u6 @4 W f; [5 W" H9 [ - };
8 Z9 r; o; g& `% ]& O& g: s1 G! C% Z - }
复制代码- struct PopQuiz {3 B* f% N( [# u3 l* f
- vogonsCount: Int;4 @0 t0 R/ y! C2 F: D6 v- U
- nicestNumber: Int;
% u) @+ x/ I/ U. g1 y - }
. u8 B* x- o" `) O3 X
2 ?) T; A4 P" d! M" w* j8 O! {$ `- fun example() {' C' g/ F4 W+ w- S& o
- // Let's introduce a couple of variables
3 @- I2 \. j% R+ u - let vogonsCount: Int = 42;( `; @: g9 L' k4 ^
- let nicestNumber: Int = 68 + 1;
?1 ~( i( n* R1 \3 o- A - # `. y; q6 d0 h5 _
- // You may instantiate the Struct as usual and assign variables to fields,0 O: r2 b& r' ~) O/ g
- // but that is a bit repetitive and tedious at times
& M4 ?0 Q% I8 B& O - PopQuiz{ vogonsCount: vogonsCount, nicestNumber: nicestNumber };
8 I+ H0 N* Y7 g* i* r7 ~& D
9 \. H2 ]+ `4 O% r- }' ] x- // Let's use field punning and type less,
+ _- J0 O3 V$ A2 t# s! p - // because our variable names happen to be the same as field names
, ?+ E C1 i6 J! ]7 o B4 U - PopQuiz{% I" K8 _4 U5 C$ P2 H/ `0 |3 x$ A
- vogonsCount,# C! K( o' Z" x+ B$ ?8 d
- nicestNumber, // trailing comma is allowed here too!; ^8 W$ G& J* v% y
- };
) d0 m1 l$ E$ I* F - }
复制代码- struct Big {
/ o7 A6 T1 J8 a- f" ^9 H& V - f1: Int; ]7 @7 z6 k0 z4 e* p
- f2: Int;
* J" E6 C( S0 q' O' l6 u. \ - f3: Int;) I3 M& [3 x3 z
- f4: Int;( D: ]: P4 r: n/ t; w
- f5: Int;0 Z; s5 q1 e H) }1 @
- f6: Int;
! G$ y& Q( V, [5 }- \0 l, j8 P - }
# M+ `7 ]' u7 N. S7 }
0 R+ r& s% [8 G7 m# ?, t E- fun conversionFun() {+ I: S3 E, G1 C$ p. J O
- dump(Big{& h8 c; q7 I7 n) o+ z, B
- f1: 10000000000, f2: 10000000000, f3: 10000000000,. |: C: Z5 V" M2 p% l
- f4: 10000000000, f5: 10000000000, f6: 10000000000,' v8 x* c3 S, J0 u: J/ r
- }.toCell()); // x{...cell with references...}
+ g2 }$ G" k5 Q% v6 v9 A - }
复制代码- struct Fizz { foo: Int }
- I. x/ e; `1 ~$ g% Q- Z. _& H! r - message(100) Buzz { bar: Int }
6 Q% Z, k$ V; j- T d4 o - 2 X0 V. y+ {" v" S, `
- fun constructThenParse() {
+ R1 d) N) E9 x- ^3 o+ j6 r - let fizzCell = Fizz{foo: 42}.toCell();
8 y7 m' e* q. ]+ d/ y7 m/ g7 [ q - let buzzCell = Buzz{bar: 27}.toCell();% ~- F7 D! d! x- t5 [1 E' \9 s: Q
9 Z4 V* t V) T/ z- let parsedFizz: Fizz = Fizz.fromCell(fizzCell);
/ S! @: { ]9 M& `' L- z$ J - let parsedBuzz: Buzz = Buzz.fromCell(buzzCell);
0 ]/ O0 \ \6 k7 u4 m% @6 x: Y - }
复制代码- struct ArbitraryStruct {}) {" b) X% E# A. D2 P* p, z
- message(0x2A) ArbitraryMessage {}, T/ v- T) D( N
+ y9 F) K$ B: G7 s7 r- fun lawOne() {
; @, `7 H, L" l8 x+ T! ]. r - let structInst = ArbitraryStruct{};7 k1 `( @. ^! H% v" Z3 d
- let messageInst = ArbitraryMessage{};6 B' e# H" ]; h, p0 S1 B
4 v$ Z: b/ H4 x2 d- ArbitraryStruct.fromCell(structInst.toCell()); // = structInst- g; i6 b; H* G1 _) u9 \( t* ]7 K
- ArbitraryMessage.fromCell(messageInst.toCell()); // = messageInst; H) e5 S# b+ n0 o8 `5 e- {
- 3 N# f ~% h) Q' t2 E: F" H; x
- // Same goes for Slices, with .toCell().asSlice() and .fromSlice()
1 {# c) n& U$ D. W K1 s - - d$ r+ ]. I+ O+ d! L
- ArbitraryStruct.fromSlice(structInst.toCell().asSlice()); // = structInst
6 E! r6 j! k+ P) n! S - ArbitraryMessage.fromSlice(messageInst.toCell().asSlice()); // = messageInst
$ c3 m7 f+ n( m; \! h - }
复制代码 对于任何与给定 Struct/Message 具有相同 TL-B 布局的 cell,调用 Struct.fromCell()(或 Message.fromCell()),然后通过 .toCell() 将结果转换为 Cell,就会得到原始 cell 的副本:- struct ArbitraryStruct { val: Int as uint32 }( c# _$ B, ~* ]. X" D
- message(0x2A) ArbitraryMessage {}; `% P& F0 p( G' p& @" H* H$ y1 q
% |8 _+ R' T. S) e/ I- fun lawTwo() {
7 V# c+ i* i' o7 N7 z, M, Q( I - // Using 32 bits to store 42 just so this cellInst can be4 P- }0 s/ U: f( R
- // re-used for working with both ArbitraryStruct and ArbitraryMessage& Q& G: k7 ^6 T. p
- let cellInst = beginCell().storeUint(42, 32).endCell();' J! ~! R. o( W
- ' v3 W. M; U+ a( B# D
- ArbitraryStruct.fromCell(cellInst).toCell(); // = cellInst' }! \4 P/ K3 X9 o j# X4 H
- ArbitraryMessage.fromCell(cellInst).toCell(); // = cellInst/ ]2 d' A5 h$ q" w
- & C. n9 m$ k: K7 w
- // Same goes for Slices, with .fromSlice() and .toCell().asSlice(), x' p7 ]6 {1 U: ^/ U
- let sliceInst = cellInst.asSlice();& c8 y }# a1 w
- 3 j' i" L0 | }9 q
- ArbitraryStruct.fromSlice(sliceInst).toCell().asSlice(); // = sliceInst
/ q" ^. q/ a2 p( v' A# m: K - ArbitraryMessage.fromSlice(sliceInst).toCell().asSlice(); // = sliceInst
4 M- R% d$ ~9 @& `8 k( i7 P - }
复制代码
3 y( y3 q" @* q9 @1 r7 P% V |
|