|
Tact 支持许多专为智能合约使用而定制的 原始数据类型。 不过,使用单独的存储方式往往会变得繁琐,因此有 Structs 和 Messages可以将类型组合在一起。 不过,使用单独的存储方式往往会变得繁琐,因此有 Structs 和 Messages可以将类型组合在一起。+ Q' R" |6 Q4 X, \" M
# a3 @9 ]4 s; x9 V( `4 w
, q* U( f) `" |) v9 ]% D2 v. T
! _5 Z+ E+ y. b1 @5 N2 k' M: o结构2 n, C0 m8 o& M4 p( W& k" T9 @
结构体可以定义包含多个不同类型字段的复杂数据类型。 它们还可以嵌套。 它们还可以嵌套。0 e$ \3 s+ L/ o" k3 y
- struct Point {
& r* |3 s' S( x( w - x: Int as int64;
% i+ c- e# |1 v4 S: z6 U7 Y3 y! X ~ - y: Int as int64;
5 n1 }; z" e: Z5 T" i - }
! p2 Y) i' @( t- G* t' g' [
' o7 H7 d, a9 W3 J- struct Line {- K; k( G7 ?; b) u/ v0 {
- start: Point;
1 j p; w% n- p/ e+ L( \ - end: Point;
5 H& K6 n2 I* o9 k! n1 ~& j K - }
复制代码 结构体还可以包含默认字段,并定义可选类型的字段。 结构体还可以包含默认字段和定义[可选类型]字段(/book/optionals)。 如果您有很多字段,但又不想一直在 new instances 中为它们指定通用值,那么这将非常有用。
! s0 y/ P G9 F2 O1 F- struct Params {
! M1 M$ j* P# @ - name: String = "Satoshi"; // default value! d' l8 J8 i; Y" M( {
2 x4 ]. U+ H' W: A; g$ M- age: Int?; // field with an optional type Int?$ V1 s* s; a/ M& Z& K
- // and default value of null
" e5 Z8 P4 w; F5 F9 A. k$ b - ( X8 e" [' C3 h9 |
- point: Point; // nested Structs
% M1 y7 Q* V; A7 M5 F" A - }
复制代码 字段的顺序很重要,因为它与TL-B 模式 中的内存布局一致。 不过,与某些采用手动内存管理的语言不同,Tact 在字段之间没有任何填充。" _: r8 M+ c4 U0 }* c* N
B6 D0 }# o4 J5 M2 Q- @
" Y. K; h- L/ I# h7 {) ~+ m. L) b1 |$ E5 | L, H0 q$ T
消息1 N1 T" Y1 I4 _9 B; q. y, ^
消息中可以包含 结构体:$ n6 Y! Y& I# B
- struct Point {
1 z9 G' k# Z( |, i8 N, F# w! I - x: Int;
$ ?. Z f$ W3 k( R* U - y: Int;3 _# y, L) ^& o; o* J7 [
- }
* d3 K& i/ `2 O9 I1 n- W - ( L; [( R, ?" E) r0 E4 V
- message Add {
0 L6 y& ~# n9 b2 @ - point: Point; // holds a struct Point5 B0 }' |" L5 a, E3 f) x
- }
复制代码- // This Message overwrites its unique id with 0x7362d09c
X! h4 G9 c( V. y, Z - message(0x7362d09c) TokenNotification {
7 E( i( R, ~ U- C# u7 p4 f0 ^* Z - forwardPayload: Slice as remaining;3 n' y- ]- k; ?$ O, q5 X( }
- }
复制代码 这对于要处理特定智能合约的某些操作码(如 Jetton standard)的情况非常有用。 该合约能够处理的操作码简表为此处以 FunC 表示。 它们是智能合约的接口。3 A, q- m1 _; i/ B. T
$ e( L; P9 N2 v4 k- S, l( [1 c7 J9 I
( |: n: k' i1 }4 Z u. Q' T操作 _) T1 E5 V) \! Y, Q, E
实例化
2 T2 w0 q( o/ P* R创建 Struct 和 Message 实例类似于 function calls,但需要用大括号 {}(大括号)代替小括号 ()指定参数:5 R4 c2 D" ]1 e- c1 Q& v7 N/ D/ o
- struct StA {6 w, k! S* d# J$ }. U3 M% n
- field1: Int;8 b3 G) I% t% P( T6 e# s/ |
- field2: Int;
. K& E6 M' v: G5 K% c# R - }
4 m# B# {5 S" o3 y' e5 V; M& @( j - 3 H; @/ |5 x0 g9 U
- message MsgB {" j+ w7 R4 |+ N( b& ]- p
- field1: String;' b1 l5 Z& ~' P, a1 s) v
- field2: String;6 U8 F2 `5 p, o! K5 Q4 M
- }
) c8 W# r8 L, I- U6 j% X
3 J( i- E" a% \, ?- fun example() {- U2 x8 i2 B) N
- // Instance of a Struct StA
( d% p3 U }2 e3 m - StA{
6 }& L* C1 S2 g1 z6 q - field1: 42,! M6 l) S* }' B9 }, G# U
- field2: 68 + 1, // trailing comma is allowed0 b( W5 x' B. u! i6 u
- };) W: E, D/ s% \! F5 O( g
- 0 g1 x0 y: [" N. W# i* E8 C
- // Instance of a Message MsgB7 @. o- B; K- e5 R% Z8 Z y' b
- MsgB{* |0 @% `' r6 e9 h* v; W1 m
- field1: "May the 4th",
3 x. n2 U o! v - field2: "be with you!", // trailing comma is allowed
N% l0 `& s q+ S* t6 f/ U3 i - };
% z3 ]9 s" _2 M) j+ O5 h. j - }
复制代码- struct PopQuiz {
r$ {! p3 i) P2 w. P7 p$ ` - vogonsCount: Int;
7 K) M* v: k# {$ t$ j - nicestNumber: Int;
M* ]- R1 [, U% `8 x" ] - }' I+ S6 |5 h" K6 _; |/ C: s
$ I9 m8 d9 Z# x0 E! |3 d3 Y- fun example() {
* N* Y& H& L* w/ | - // Let's introduce a couple of variables4 _4 e: D, N- K3 H( f) y% q! v
- let vogonsCount: Int = 42;# y& f- |+ g' g/ l; l" Q; z' p) g
- let nicestNumber: Int = 68 + 1;
- ?6 U5 t- u' w& U0 ^! E
6 F2 m+ b' T4 R& _+ M/ A- // You may instantiate the Struct as usual and assign variables to fields," `& m" b# w& P/ `2 x7 `4 A4 j: c
- // but that is a bit repetitive and tedious at times6 x" n2 m# u% F; W1 {" ?
- PopQuiz{ vogonsCount: vogonsCount, nicestNumber: nicestNumber };
: |/ v4 I p. V. o5 ]% `: r' M
# p3 b2 f( J" O; {8 u- // Let's use field punning and type less,: N( L ~( s3 z' M/ ~' l
- // because our variable names happen to be the same as field names$ T% Z5 D# q9 n9 u( M3 N. e" z
- PopQuiz{
" p, M+ o3 ~, \" X d7 X - vogonsCount,' u! ^6 w6 w# ^0 v9 |3 S
- nicestNumber, // trailing comma is allowed here too!
9 a* D% y: A! }& x7 N4 Q - };
2 f9 H; C6 Q$ {8 v: v* S( X w - }
复制代码- struct Big {
, S6 s q0 Y: A8 \& t3 X. \ O8 C" A - f1: Int;# Y4 A! W9 q. q
- f2: Int;+ t# B) B2 s5 T; B+ S# Z
- f3: Int;5 U6 a& I& \. }( F% Q6 ]4 w
- f4: Int;
% Z u# a, T4 W( w - f5: Int;, a8 u, b% G# A5 T# X
- f6: Int;
6 M1 C# Q* E3 [; e9 H' J - }" {7 O, c8 u: G+ g
- , j o8 a$ r! A M3 X
- fun conversionFun() {5 Z& V3 ?6 |9 ^; z
- dump(Big{( H! n) G8 O' q. @& k3 r
- f1: 10000000000, f2: 10000000000, f3: 10000000000,
, b% t! K/ E% R' { - f4: 10000000000, f5: 10000000000, f6: 10000000000,( d: Z. _ M& P
- }.toCell()); // x{...cell with references...}
8 A* B. g; N2 H- B - }
复制代码- struct Fizz { foo: Int }! O: S/ u8 N0 P! |- I5 f! |8 o
- message(100) Buzz { bar: Int }
\& @ l2 o! R' k6 j4 x
4 ^7 j2 D3 {9 ~1 a" M: c) R7 |; U2 c' E- fun constructThenParse() {
6 y! V" A9 _3 v6 n: ^0 Q7 } ~8 T - let fizzCell = Fizz{foo: 42}.toCell();
& k! G0 G4 l3 u& b5 e! n8 ? - let buzzCell = Buzz{bar: 27}.toCell();
1 x0 Z7 q/ `- f" } - " u3 c( J. y5 g: \5 v
- let parsedFizz: Fizz = Fizz.fromCell(fizzCell);
8 \* Z) p5 Z W - let parsedBuzz: Buzz = Buzz.fromCell(buzzCell);
x, B6 Q3 u8 \! \' g# @$ i$ f) n6 K - }
复制代码- struct ArbitraryStruct {}3 a/ n* x2 @; z
- message(0x2A) ArbitraryMessage {}( X. S/ ?- H. _2 Z2 u# q3 O
) m' n5 I/ u- V w/ D! v+ T- fun lawOne() {
3 E6 p2 K! K7 M$ ~0 P8 |0 d$ i - let structInst = ArbitraryStruct{};$ b; K! |/ L: O! s# d+ h$ d
- let messageInst = ArbitraryMessage{};# c* X) G. k2 b6 I. B
- $ [. b) M* ], b# R
- ArbitraryStruct.fromCell(structInst.toCell()); // = structInst
3 Q6 `/ ~; Y* R/ ~. U) D* Y - ArbitraryMessage.fromCell(messageInst.toCell()); // = messageInst& y, \4 C8 G% p D4 {* n
. ]2 t- {0 Z$ y- // Same goes for Slices, with .toCell().asSlice() and .fromSlice()) t0 q4 B( J! u' H. G8 ?$ j
- + y+ {+ ]' {- X; y7 C n S
- ArbitraryStruct.fromSlice(structInst.toCell().asSlice()); // = structInst
3 u Y8 ~9 h% R _5 {! m+ h - ArbitraryMessage.fromSlice(messageInst.toCell().asSlice()); // = messageInst- F* {% L# J. K; Z! ]
- }
复制代码 对于任何与给定 Struct/Message 具有相同 TL-B 布局的 cell,调用 Struct.fromCell()(或 Message.fromCell()),然后通过 .toCell() 将结果转换为 Cell,就会得到原始 cell 的副本:- struct ArbitraryStruct { val: Int as uint32 }
4 s2 ]! G" n7 M+ `1 y6 V" u4 u - message(0x2A) ArbitraryMessage {}0 J% B: W3 `- f5 i6 B3 F9 k9 ~# A
% [0 {" L( ~! W- fun lawTwo() {
8 R5 A# ^" p) U& h! M - // Using 32 bits to store 42 just so this cellInst can be
; p3 K$ v; ?' J) q' J - // re-used for working with both ArbitraryStruct and ArbitraryMessage
, G5 L& u! C3 k' c; X( g8 L* B - let cellInst = beginCell().storeUint(42, 32).endCell();
, n1 E/ b: h$ g - 6 `8 f9 T. Z/ ?: }' J3 f6 D
- ArbitraryStruct.fromCell(cellInst).toCell(); // = cellInst
- Y. F/ C3 P% r" S' z# ^ - ArbitraryMessage.fromCell(cellInst).toCell(); // = cellInst% [- e. J% @- N# @
x1 ?5 `1 w# M; N( x/ } }- h- // Same goes for Slices, with .fromSlice() and .toCell().asSlice()3 w! c' G3 W' u
- let sliceInst = cellInst.asSlice();2 m$ t4 H N W9 b
- , Z. t! }; v9 ^
- ArbitraryStruct.fromSlice(sliceInst).toCell().asSlice(); // = sliceInst
$ U0 z% H0 {( f - ArbitraryMessage.fromSlice(sliceInst).toCell().asSlice(); // = sliceInst. T8 _6 K. O: _4 X
- }
复制代码
5 \4 S$ r6 Q7 E' R1 F" f |
|