English 简体中文 繁體中文 한국 사람 日本語 Deutsch русский بالعربية TÜRKÇE português คนไทย french

简体中文 繁體中文 English 日本語 Deutsch 한국 사람 بالعربية TÜRKÇE คนไทย Français русский

回答

收藏

Tact 语言基础 |结构(Structs)和消息(Messages)

开源社区 开源社区 6681 人阅读 | 0 人回复 | 2025-03-26

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( \
  1. struct Point {
    ) [8 ]0 H+ z0 C2 ^. I- l' q
  2.     x: Int as int64;
    , N7 |3 a, ^" `" U5 t# Y& t: ?7 Y
  3.     y: Int as int64;3 h$ F2 \/ T) r8 m- K
  4. }
    8 D+ L# u5 [& B% M- o# A
  5. $ O8 G% C. `6 [2 K9 \
  6. struct Line {
    9 B6 j  q. J  ]) _, \
  7.     start: Point;7 c1 h+ E  P! E0 y# L
  8.     end: Point;
    4 ]% O6 C6 X' j: [
  9. }
复制代码
结构体还可以包含默认字段,并定义可选类型的字段。 结构体还可以包含默认字段和定义[可选类型]字段(/book/optionals)。 如果您有很多字段,但又不想一直在 new instances 中为它们指定通用值,那么这将非常有用。
* ?7 h; X% ^2 Q& P; O" C) C
  1. struct Params {
    ; q4 G  }+ b% O2 j3 K6 J
  2.     name: String = "Satoshi"; // default value( Q7 S: n# D/ G
  3. # q  G8 h1 D* c9 B( O
  4.     age: Int?; // field with an optional type Int?
    " o: s( U6 M# k5 Q, ]
  5.                // and default value of null; v1 S$ L' E8 R7 N3 \) P$ @
  6. ( L# p0 k+ z0 Q, I, c9 m$ j- v2 r' ]
  7.     point: Point; // nested Structs
    * Y  }) J( m; _( s8 f) z3 s
  8. }
复制代码
字段的顺序很重要,因为它与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
  1. struct Point {# ?9 {. L% @5 ]/ K- T3 q
  2.     x: Int;
    4 A. t( Y. T4 [3 f! }7 ?) L9 v
  3.     y: Int;+ p+ c4 K- L  j, z
  4. }
    & ?! @9 W  V3 C' r
  5. , c% B' W, Y9 k+ d5 o
  6. message Add {
    - |1 ?# l& W# m2 f$ a
  7.     point: Point; // holds a struct Point
    ' Q' D& d2 \. Q
  8. }
复制代码
  1. // This Message overwrites its unique id with 0x7362d09c& O/ y/ [% [( r2 ~% E$ z4 K3 R
  2. message(0x7362d09c) TokenNotification {
    , |) S5 r& [+ I6 X( v6 n, g
  3.     forwardPayload: Slice as remaining;
    ' d6 z' y5 w) K; X/ g' A
  4. }
复制代码
这对于要处理特定智能合约的某些操作码(如 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
  1. struct StA {
    : |; w2 I1 b6 L3 x* x
  2.     field1: Int;
    " O7 C: V. ?% T
  3.     field2: Int;
    8 K4 p0 x- \/ }! x+ b/ u6 S
  4. }% h3 d: J4 B3 h; e8 ~
  5. % o& J4 J. `  y* ^( K6 E% k
  6. message MsgB {
    4 `! D) k! K5 s
  7.     field1: String;; y1 \9 O& f  K* {( q
  8.     field2: String;
    5 u, G/ y4 H" ^/ T2 L( s
  9. }
    9 D( N: S, N- h9 M/ W) l
  10. : y6 g2 L7 w0 o( h' [
  11. fun example() {
    . Z1 A, Y+ M4 W& Y5 Y) I9 u
  12.     // Instance of a Struct StA! e8 c8 }& r2 p8 E' }0 I
  13.     StA{
    ' k; A; H7 J' O. {# Z
  14.         field1: 42,3 N( g. ]! q5 E, \
  15.         field2: 68 + 1, // trailing comma is allowed& F% I+ W: H1 D: ]/ ?8 q0 F. a
  16.     };: P$ H+ U7 B' y* y/ L) y6 I, m
  17. + X$ d+ }# G# S  B/ ?
  18.     // Instance of a Message MsgB, S* _0 d; W) ^8 \9 y
  19.     MsgB{& R- W! x$ @9 b/ n# _% x' i
  20.         field1: "May the 4th",' q0 V) q- j/ q: C4 V9 L
  21.         field2: "be with you!", // trailing comma is allowed
    2 f; Z! u6 @4 W  f; [5 W" H9 [
  22.     };
    8 Z9 r; o; g& `% ]& O& g: s1 G! C% Z
  23. }
复制代码
  1. struct PopQuiz {3 B* f% N( [# u3 l* f
  2.     vogonsCount: Int;4 @0 t0 R/ y! C2 F: D6 v- U
  3.     nicestNumber: Int;
    % u) @+ x/ I/ U. g1 y
  4. }
    . u8 B* x- o" `) O3 X

  5. 2 ?) T; A4 P" d! M" w* j8 O! {$ `
  6. fun example() {' C' g/ F4 W+ w- S& o
  7.     // Let's introduce a couple of variables
    3 @- I2 \. j% R+ u
  8.     let vogonsCount: Int = 42;( `; @: g9 L' k4 ^
  9.     let nicestNumber: Int = 68 + 1;
      ?1 ~( i( n* R1 \3 o- A
  10. # `. y; q6 d0 h5 _
  11.     // You may instantiate the Struct as usual and assign variables to fields,0 O: r2 b& r' ~) O/ g
  12.     // but that is a bit repetitive and tedious at times
    & M4 ?0 Q% I8 B& O
  13.     PopQuiz{ vogonsCount: vogonsCount, nicestNumber: nicestNumber };
    8 I+ H0 N* Y7 g* i* r7 ~& D

  14. 9 \. H2 ]+ `4 O% r- }' ]  x
  15.     // Let's use field punning and type less,
    + _- J0 O3 V$ A2 t# s! p
  16.     // because our variable names happen to be the same as field names
    , ?+ E  C1 i6 J! ]7 o  B4 U
  17.     PopQuiz{% I" K8 _4 U5 C$ P2 H/ `0 |3 x$ A
  18.         vogonsCount,# C! K( o' Z" x+ B$ ?8 d
  19.         nicestNumber, // trailing comma is allowed here too!; ^8 W$ G& J* v% y
  20.     };
    ) d0 m1 l$ E$ I* F
  21. }
复制代码
  1. struct Big {
    / o7 A6 T1 J8 a- f" ^9 H& V
  2.     f1: Int;  ]7 @7 z6 k0 z4 e* p
  3.     f2: Int;
    * J" E6 C( S0 q' O' l6 u. \
  4.     f3: Int;) I3 M& [3 x3 z
  5.     f4: Int;( D: ]: P4 r: n/ t; w
  6.     f5: Int;0 Z; s5 q1 e  H) }1 @
  7.     f6: Int;
    ! G$ y& Q( V, [5 }- \0 l, j8 P
  8. }
    # M+ `7 ]' u7 N. S7 }

  9. 0 R+ r& s% [8 G7 m# ?, t  E
  10. fun conversionFun() {+ I: S3 E, G1 C$ p. J  O
  11.     dump(Big{& h8 c; q7 I7 n) o+ z, B
  12.         f1: 10000000000, f2: 10000000000, f3: 10000000000,. |: C: Z5 V" M2 p% l
  13.         f4: 10000000000, f5: 10000000000, f6: 10000000000,' v8 x* c3 S, J0 u: J/ r
  14.     }.toCell()); // x{...cell with references...}
    + g2 }$ G" k5 Q% v6 v9 A
  15. }
复制代码
  1. struct Fizz { foo: Int }
    - I. x/ e; `1 ~$ g% Q- Z. _& H! r
  2. message(100) Buzz { bar: Int }
    6 Q% Z, k$ V; j- T  d4 o
  3. 2 X0 V. y+ {" v" S, `
  4. fun constructThenParse() {
    + R1 d) N) E9 x- ^3 o+ j6 r
  5.     let fizzCell = Fizz{foo: 42}.toCell();
    8 y7 m' e* q. ]+ d/ y7 m/ g7 [  q
  6.     let buzzCell = Buzz{bar: 27}.toCell();% ~- F7 D! d! x- t5 [1 E' \9 s: Q

  7. 9 Z4 V* t  V) T/ z
  8.     let parsedFizz: Fizz = Fizz.fromCell(fizzCell);
    / S! @: {  ]9 M& `' L- z$ J
  9.     let parsedBuzz: Buzz = Buzz.fromCell(buzzCell);
    0 ]/ O0 \  \6 k7 u4 m% @6 x: Y
  10. }
复制代码
  1. struct ArbitraryStruct {}) {" b) X% E# A. D2 P* p, z
  2. message(0x2A) ArbitraryMessage {}, T/ v- T) D( N

  3. + y9 F) K$ B: G7 s7 r
  4. fun lawOne() {
    ; @, `7 H, L" l8 x+ T! ]. r
  5.     let structInst = ArbitraryStruct{};7 k1 `( @. ^! H% v" Z3 d
  6.     let messageInst = ArbitraryMessage{};6 B' e# H" ]; h, p0 S1 B

  7. 4 v$ Z: b/ H4 x2 d
  8.     ArbitraryStruct.fromCell(structInst.toCell());   // = structInst- g; i6 b; H* G1 _) u9 \( t* ]7 K
  9.     ArbitraryMessage.fromCell(messageInst.toCell()); // = messageInst; H) e5 S# b+ n0 o8 `5 e- {
  10. 3 N# f  ~% h) Q' t2 E: F" H; x
  11.     // Same goes for Slices, with .toCell().asSlice() and .fromSlice()
    1 {# c) n& U$ D. W  K1 s
  12. - d$ r+ ]. I+ O+ d! L
  13.     ArbitraryStruct.fromSlice(structInst.toCell().asSlice());   // = structInst
    6 E! r6 j! k+ P) n! S
  14.     ArbitraryMessage.fromSlice(messageInst.toCell().asSlice()); // = messageInst
    $ c3 m7 f+ n( m; \! h
  15. }
复制代码
对于任何与给定 Struct/Message 具有相同 TL-B 布局的 cell,调用 Struct.fromCell()(或 Message.fromCell()),然后通过 .toCell() 将结果转换为 Cell,就会得到原始 cell 的副本:
  1. struct ArbitraryStruct { val: Int as uint32 }( c# _$ B, ~* ]. X" D
  2. message(0x2A) ArbitraryMessage {}; `% P& F0 p( G' p& @" H* H$ y1 q

  3. % |8 _+ R' T. S) e/ I
  4. fun lawTwo() {
    7 V# c+ i* i' o7 N7 z, M, Q( I
  5.     // Using 32 bits to store 42 just so this cellInst can be4 P- }0 s/ U: f( R
  6.     // re-used for working with both ArbitraryStruct and ArbitraryMessage& Q& G: k7 ^6 T. p
  7.     let cellInst = beginCell().storeUint(42, 32).endCell();' J! ~! R. o( W
  8. ' v3 W. M; U+ a( B# D
  9.     ArbitraryStruct.fromCell(cellInst).toCell();  // = cellInst' }! \4 P/ K3 X9 o  j# X4 H
  10.     ArbitraryMessage.fromCell(cellInst).toCell(); // = cellInst/ ]2 d' A5 h$ q" w
  11. & C. n9 m$ k: K7 w
  12.     // Same goes for Slices, with .fromSlice() and .toCell().asSlice(), x' p7 ]6 {1 U: ^/ U
  13.     let sliceInst = cellInst.asSlice();& c8 y  }# a1 w
  14. 3 j' i" L0 |  }9 q
  15.     ArbitraryStruct.fromSlice(sliceInst).toCell().asSlice();  // = sliceInst
    / q" ^. q/ a2 p( v' A# m: K
  16.     ArbitraryMessage.fromSlice(sliceInst).toCell().asSlice(); // = sliceInst
    4 M- R% d$ ~9 @& `8 k( i7 P
  17. }
复制代码

3 y( y3 q" @* q9 @1 r7 P% V
分享到:
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则