|
Tact 支持许多专为智能合约使用而定制的 原始数据类型。 不过,使用单独的存储方式往往会变得繁琐,因此有 Structs 和 Messages可以将类型组合在一起。 不过,使用单独的存储方式往往会变得繁琐,因此有 Structs 和 Messages可以将类型组合在一起。0 D! L) _# D4 H; @% [
h( C7 y. z1 `/ }4 E4 ]) \/ Y. v9 g
# V! w8 w2 A5 Z) [
结构# h! P, ?+ L1 P/ h! I
结构体可以定义包含多个不同类型字段的复杂数据类型。 它们还可以嵌套。 它们还可以嵌套。
; O' b( Z& j" k9 K) y- L- struct Point {' K: B9 ~2 j7 \4 N6 j& F
- x: Int as int64;: D: E% @- T5 |$ a3 }4 j
- y: Int as int64;5 I! U8 @, i4 l; L$ I# g6 G x9 U
- }
3 l: Z0 T9 Q& r$ l2 R1 T( {& {. r - ' J( g4 n1 a: C4 m6 j
- struct Line {
+ M& D7 k2 W$ E" ]" b7 Z - start: Point;
6 t3 I) i4 u9 F - end: Point;
! O2 y8 R/ _( W4 s7 O - }
复制代码 结构体还可以包含默认字段,并定义可选类型的字段。 结构体还可以包含默认字段和定义[可选类型]字段(/book/optionals)。 如果您有很多字段,但又不想一直在 new instances 中为它们指定通用值,那么这将非常有用。
# s( H2 t# C; |9 L$ N- struct Params {* z. ?. H* y) s
- name: String = "Satoshi"; // default value
9 y B( j- I" p) O L* V; i5 e
* o) @3 }" }/ h3 Y. Q/ w9 L+ W% w- age: Int?; // field with an optional type Int?
8 H, [) B5 e/ N( U - // and default value of null9 m! Y+ a0 K- T; n' J& U. u
- : ~" D$ x; \8 C; |
- point: Point; // nested Structs
& [; b! N+ |/ ] H7 U* ?+ L - }
复制代码 字段的顺序很重要,因为它与TL-B 模式 中的内存布局一致。 不过,与某些采用手动内存管理的语言不同,Tact 在字段之间没有任何填充。
" b. L4 Q8 ^; O% f& S6 }
5 O: o) i# F$ J0 l$ I& y4 x. d m( T9 n% g" T5 z, j+ {
: }9 B1 C8 c# \! y
消息
S9 d8 ?6 {" \6 \- e消息中可以包含 结构体:
$ G. p4 O* K. b. r- struct Point {* ]# x, }* e, [
- x: Int;0 {& P# c1 v! B5 F' Y9 h8 M1 R
- y: Int;
% c, S. M) E( h9 O( G; g - }
) ` m6 M2 t( R - 4 [$ ?3 }: @4 i8 _9 ]6 p, P
- message Add {9 k1 d) V. h" k4 x
- point: Point; // holds a struct Point& n0 ?* C& Q2 i' r; _. V- b
- }
复制代码- // This Message overwrites its unique id with 0x7362d09c
# Y/ j% k0 S1 F- @ - message(0x7362d09c) TokenNotification {: Q9 ~+ [* Y5 a, D1 @% i0 O
- forwardPayload: Slice as remaining;
* H2 N/ G/ R t5 b2 Z - }
复制代码 这对于要处理特定智能合约的某些操作码(如 Jetton standard)的情况非常有用。 该合约能够处理的操作码简表为此处以 FunC 表示。 它们是智能合约的接口。
6 H$ m, X9 H( R* Z1 z" I' I* C7 D; n: I. f V
8 O/ W4 g- w" p/ {+ x( y操作0 s& o" r1 S% V/ f% L8 i; K9 a
实例化$ j5 w t/ j' l# w" e$ m3 B# Y+ }
创建 Struct 和 Message 实例类似于 function calls,但需要用大括号 {}(大括号)代替小括号 ()指定参数:+ A# \6 {6 u+ ~0 ]( t! y7 s& M
- struct StA {1 u2 A7 K4 W6 v$ M' h
- field1: Int;
- V& O/ p% m. @; y" r! s2 ^+ |3 ^3 X7 \ - field2: Int;
/ e" `% s% j4 K! \; H O; \) s - }
7 Y/ d0 s Z, Z% s& ~( i* M
" _2 a+ B, ?0 B; p- message MsgB {
2 H, l( {9 F% V6 W% V# ^ - field1: String;4 k* h4 \/ K W' |- C, _% o
- field2: String;
/ K& x2 r h0 P- r+ S - } [6 R# \, k, ]
- 2 { Q7 `8 n: \; D; B8 z. w; G. e
- fun example() {
2 c7 g* ]/ S' _: S6 H - // Instance of a Struct StA
( s6 Y1 D/ c) l1 c6 U# h - StA{
9 P! y, h9 ~5 ]! Y0 P( i# ?) x+ a; W - field1: 42,. _" l7 I8 q$ _/ d, w: J+ b
- field2: 68 + 1, // trailing comma is allowed
$ G- Q$ u% C8 M0 o - };
2 s7 c, C+ f1 @ y2 r8 H4 S$ u
0 \( u) F# d0 }! [- // Instance of a Message MsgB8 W* V' c/ G0 H: z
- MsgB{; |2 B/ |7 E9 k/ o0 L! W
- field1: "May the 4th",! i T) g9 S6 ?) `* }* c& e
- field2: "be with you!", // trailing comma is allowed
9 m0 f! o" n( Y/ o8 R0 } - };. t& Q& q0 q9 Z2 {" o
- }
复制代码- struct PopQuiz {2 _- S1 P) i i8 D" f
- vogonsCount: Int;
" x0 s" F6 q; x6 R" P7 P - nicestNumber: Int;
r( U: R- m" m3 O1 ?& u - }6 @' e1 E; H! ]$ _4 D
- 1 E, A+ ?9 n: r4 N
- fun example() { G& j3 Y/ C: h
- // Let's introduce a couple of variables4 \8 E2 F/ ~2 ^' g: k
- let vogonsCount: Int = 42;
- q4 n; x7 n8 ^ - let nicestNumber: Int = 68 + 1;$ d: @# L K2 ~; }" U
( @6 T! \3 g" h; ~/ S: v5 y: _- // You may instantiate the Struct as usual and assign variables to fields,
- G ]: s- e6 T/ D: E - // but that is a bit repetitive and tedious at times
+ v1 y" m. A' x' `4 X - PopQuiz{ vogonsCount: vogonsCount, nicestNumber: nicestNumber };! a6 h" ]# R1 ~$ |! m- Q# S9 ]
4 N& H, t" f; G8 y3 g# s1 D- // Let's use field punning and type less,
0 N/ ]8 H4 B2 k; g+ T2 D" }, M+ { - // because our variable names happen to be the same as field names
/ D+ Y/ [) q+ K" _ - PopQuiz{
9 a6 h/ q! T' V$ u' c - vogonsCount,/ D Y' c* |0 b& b' \% V: S) r
- nicestNumber, // trailing comma is allowed here too!" }& w: H3 A+ _$ U9 ?& K5 D
- };
$ j& `" G4 K8 w' y; @ - }
复制代码- struct Big {4 _" x, d* h$ Z
- f1: Int;2 w! Q$ Q( o8 R# _" v2 _8 J
- f2: Int;8 f2 q2 O5 X+ W; g, v! L
- f3: Int;
; x+ H4 R1 `8 t - f4: Int;( Q! @& O F* U, q+ i
- f5: Int;
% R# s0 V" C9 L3 o - f6: Int;
1 {) `6 e- i; J, ?8 [, R - }
6 c4 B( Z: e$ x5 s z1 C
9 e' B# [) M6 k: s5 D: ^2 Y, D- fun conversionFun() {
& k5 q$ x5 t g/ e% F$ R9 U - dump(Big{ i1 K8 D& X- P, x
- f1: 10000000000, f2: 10000000000, f3: 10000000000,0 R6 L9 h% L& J# g( E4 K7 i: f
- f4: 10000000000, f5: 10000000000, f6: 10000000000,5 {) F3 h5 R, g: K6 m) d
- }.toCell()); // x{...cell with references...}
3 D( w+ [" M, `" `9 f - }
复制代码- struct Fizz { foo: Int }5 o, U+ S' C5 d7 ^/ s
- message(100) Buzz { bar: Int }
. U. q# b/ U! z: `+ U
4 j* F5 U, _8 g- fun constructThenParse() {
& _; D" x/ f1 u - let fizzCell = Fizz{foo: 42}.toCell();) X, ?7 x* o+ G; @& r
- let buzzCell = Buzz{bar: 27}.toCell();+ @0 Z A' w9 Y" b
+ Y" v) `' c2 V) g- let parsedFizz: Fizz = Fizz.fromCell(fizzCell);3 ^( L5 J6 M& R: X0 k* P4 b6 H
- let parsedBuzz: Buzz = Buzz.fromCell(buzzCell);7 l8 }2 L% W& w- G0 G% ~1 e* Q( B( Q
- }
复制代码- struct ArbitraryStruct {}
8 ~. _5 `5 `/ l9 W3 J, d4 | - message(0x2A) ArbitraryMessage {} h0 i4 n: ?+ K/ T' Y i1 a
- 6 @( N V' Y3 q
- fun lawOne() {( D# T- x M( s+ r, U- E
- let structInst = ArbitraryStruct{};" y3 g) d% Z; @" i! y9 Q- N
- let messageInst = ArbitraryMessage{};
+ ?& g4 u5 t0 }; A. Q) B
' V, E3 T6 ]# s# o& n4 U) X1 x- ArbitraryStruct.fromCell(structInst.toCell()); // = structInst
1 K, k! I: W; t I( r - ArbitraryMessage.fromCell(messageInst.toCell()); // = messageInst$ ]9 p% w4 R7 e" }% t7 L: G1 Q
0 ~( r3 F W: x* V! {& b% t5 w6 d1 B- // Same goes for Slices, with .toCell().asSlice() and .fromSlice()$ i- U3 t6 v, Z
- 4 ]& Y; y0 W) \* S
- ArbitraryStruct.fromSlice(structInst.toCell().asSlice()); // = structInst
: D/ M( h3 v5 q& k. z5 A# D5 j - ArbitraryMessage.fromSlice(messageInst.toCell().asSlice()); // = messageInst
, w; g* F! x, {( v - }
复制代码 对于任何与给定 Struct/Message 具有相同 TL-B 布局的 cell,调用 Struct.fromCell()(或 Message.fromCell()),然后通过 .toCell() 将结果转换为 Cell,就会得到原始 cell 的副本:- struct ArbitraryStruct { val: Int as uint32 }
' B5 o7 v$ }( B( @! k, x - message(0x2A) ArbitraryMessage {}: A( O" q3 A" J. L: Y
7 w: [. t! q% ~3 h4 k- fun lawTwo() {
, h4 D7 |# u$ J0 h - // Using 32 bits to store 42 just so this cellInst can be
3 z- }2 c5 |7 _) `2 J - // re-used for working with both ArbitraryStruct and ArbitraryMessage! c! L- ]# q; |
- let cellInst = beginCell().storeUint(42, 32).endCell();
! O1 N: S& @3 \, z8 P- c( ~7 V - ' t9 k' v/ o/ f, L( ^8 G) m
- ArbitraryStruct.fromCell(cellInst).toCell(); // = cellInst1 h0 p; n3 Y3 p% u
- ArbitraryMessage.fromCell(cellInst).toCell(); // = cellInst. z' X3 t$ {9 O4 K: y
- " E! ` k; V& o) M5 ^ W
- // Same goes for Slices, with .fromSlice() and .toCell().asSlice()
8 W: ]8 z$ P8 ?" p - let sliceInst = cellInst.asSlice();7 F9 `& z+ z5 Z0 k! G8 J6 C6 @
- 8 I: G3 v C, z" T0 j. F
- ArbitraryStruct.fromSlice(sliceInst).toCell().asSlice(); // = sliceInst/ Q! _+ P3 Y" J) V" g6 H
- ArbitraryMessage.fromSlice(sliceInst).toCell().asSlice(); // = sliceInst
) N4 @0 e5 W5 h+ N/ b - }
复制代码
2 p. k% b0 t+ X |
|