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

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

回答

收藏

7.5 NFT 项目/SBT 项目智能合约

开源社区 开源社区 7006 人阅读 | 0 人回复 | 2025-03-15

在本课中,我们将回顾经典的 NFT 项目的代码。此外,我们还将学习 SoulBound (SBT) Item。
" M4 m; T  e7 _4 n+ Z7 L' K  k
NFT Item合约
让我们打开并查看 contract.
  1. int min_tons_for_storage() asm "50000000 PUSHINT"; ;; 0.05 TON* M5 w8 r9 _' p  Y$ V2 j
复制代码
就像在前面的课程中一样,我们看到有一些常量显示了我们应该在合约上存储的最低代币数量,以便能够支付租金。
7 o5 z" c3 E3 m7 A5 x
  1. (int, int, slice, slice, cell) load_data() {
    # I, J: |: k, Z
  2.     slice ds = get_data().begin_parse();' K4 g: \. N4 M+ o3 z5 v4 v9 L
  3.     var (index, collection_address) = (ds~load_uint(64), ds~load_msg_addr());' n9 v: |5 Z& C/ s# U# \1 K" u
  4.     if (ds.slice_bits() > 0) {" i/ r- _' a& i  S7 K
  5.       return (-1, index, collection_address, ds~load_msg_addr(), ds~load_ref());
    6 Z  D7 ~$ N% b7 n$ z2 M
  6.     } else {  + v; t% }" T% _3 _
  7.       return (0, index, collection_address, null(), null()); ;; nft not initialized yet
    ( x. [  N3 e/ g
  8.     }$ b7 V3 H5 {# i# z) Z  v3 j
  9. }
复制代码
然后我们看到 load_data. 这个 load_data 与我们以前看到的有些不同。通常,我们只是从本地存储区读取数据,但在这里,我们有一些更复杂的逻辑。因此,我们打开本地存储的Cell进行读取,然后读出 index 和 collection_address. 我们知道什么是索引和 Collections 地址。  
然后,我们尝试读取 NFT Item的元数据。首先,我们检查它是否存在。如果是,我们返回 –1, 以及Collections地址、所有者地址和元数据。或者我们只返回 0, 索引和Collections地址时,没有所有者和元数据。这意味着 NFT 尚未初始化。基于这一点,我们将在后面的代码中做出一些决定。
  1. () store_data(int index, slice collection_address, slice owner_address, cell content) impure {( b* f: k: ?6 O/ S; n
  2.     set_data(2 z/ E5 T) Y+ c
  3.         begin_cell()
    * s5 N: n% w% E5 u+ ^& `% W
  4.             .store_uint(index, 64); p- B  O, c/ j, x0 a5 X& ?, M
  5.             .store_slice(collection_address)4 U# F" x) A7 l& q! R
  6.             .store_slice(owner_address)
    3 n" S  J! b/ c# i/ f
  7.             .store_ref(content)
    # d6 }: V# w( D( L2 y7 r& [- ]& u+ b
  8.             .end_cell()* B1 |: }/ a+ J9 }
  9.     );& m. }# R, f, h* w$ o- y
  10. }
复制代码
我们还看到 store_data 就像我们调查的其他合约一样,这只是正常情况。

( |# _# U% z/ h! T
  1. () send_msg(slice to_address, int amount, int op, int query_id, builder payload, int send_mode) impure inline {
    ; k2 c- @- @3 |5 ]
  2.   var msg = begin_cell()- B) D9 l; K! ]  c' ^; }
  3.     .store_uint(0x10, 6) ;; nobounce - int_msg_info$0 ihr_disabled:Bool bounce:Bool bounced:Bool src:MsgAddress -> 010000
    " U1 @! u% i2 z: l. G
  4.     .store_slice(to_address), z" \" G7 F* U( F+ N
  5.     .store_coins(amount)7 [& S+ u# I0 D
  6.     .store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1)1 F0 |* w( e3 ]1 P9 z3 W, a! e
  7.     .store_uint(op, 32)
      P  ?( q0 ?- ?$ {: T3 |
  8.     .store_uint(query_id, 64);- l. L# U6 ?$ b7 V; c# ^
  9. " }; ]! Y" \( }2 U
  10.   if (~ builder_null?(payload)) {
    - P8 A2 T- o6 R( ?
  11.     msg = msg.store_builder(payload);# Q5 G7 k7 e* b# K# ]
  12.   }
复制代码
在这里,我们封装了发送消息的逻辑,因此我们可以在合约的其他逻辑中使用这个函数。因此,只要我们想发送信息,就可以使用这个函数。它接受目标地址、我们应附加的金额、操作代码、查询 ID、可能的有效载荷(这次是以生成器格式)以及消息的发送模式。这就是我们发送信息的逻辑。
我们的另一个函数是 transfer_ownership. 让我们稍后再研究,一旦我们在 recv_internal 逻辑,因为它是这里最重要的功能。
  1. () recv_internal(int my_balance, int msg_value, cell in_msg_full, slice in_msg_body) impure {
    . ?, A) `' ~, O$ Z7 h8 [' K9 n
  2.     if (in_msg_body.slice_empty?()) { ;; ignore empty messages
    9 B0 V7 G$ G5 O' \  o$ A
  3.         return ();
    % o2 q& p. M3 d  q# T
  4.     }
    3 L9 J5 ^4 g4 G0 R! H8 a4 n1 z

  5. ( g( p( t- V! [0 P0 V  i$ J
  6.     slice cs = in_msg_full.begin_parse();
      `8 [- O. \0 @
  7.     int flags = cs~load_uint(4);. a+ c; C7 t1 l

  8. : a+ p1 ^7 s! ^* B- v" U
  9.     if (flags & 1) { ;; ignore all bounced messages: s" M  u- j' J+ }& i" h: F; {
  10.         return ();
    0 T* ]: u9 o3 q/ K1 f, s8 R
  11.     }$ O* @$ i. h& t2 r7 N$ T
  12.     slice sender_address = cs~load_msg_addr();
    . V5 ]) B, M( `8 u
  13. ; I- j6 ]9 y# X
  14.     cs~load_msg_addr(); ;; skip dst9 O' K4 T& d, K& ^. K: b4 Z
  15.     cs~load_coins(); ;; skip value6 l$ {  u0 Y) O9 ]$ ^9 @; ^
  16.     cs~skip_bits(1); ;; skip extracurrency collection
    & f1 H5 Y, l! f
  17.     cs~load_coins(); ;; skip ihr_fee& o: @6 ]- v8 g/ T# o
  18.     int fwd_fee = muldiv(cs~load_coins(), 3, 2); ;; we use message fwd_fee for estimation of forward_payload costs# j$ Q  @* j# a5 g

  19. ) V4 K) o4 D) q3 \7 D$ a  n2 N! P
  20.     (int init?, int index, slice collection_address, slice owner_address, cell content) = load_data();6 f! q  x+ J# E
  21.     if (~ init?) {. M2 ^- Q: R& I& n
  22.       throw_unless(405, equal_slices(collection_address, sender_address));
    ( M8 R% p! Q( w1 h6 Y7 h- X9 u
  23.       store_data(index, collection_address, in_msg_body~load_msg_addr(), in_msg_body~load_ref());% U) P, K. ^2 t) V8 W$ M1 e: M
  24.       return ();
      Z: x* d1 _4 g
  25.     }
复制代码
我们来看看它是怎么做的。我们再次忽略空的 in_msg_body 消息,读取标志,忽略退回的信息,并检查发件人是谁。现在我们跳过一些内容,比如信息的目标地址之类的,因为你已经知道这个Cell的内容是什么了。接下来是 fwd_fee 再次根据上一笔交易估算发送下一笔交易的成本。
然后加载数据。正如你所记得的,我们加载数据的方式与这里的实现方式相同。因此,我们要检查它是否未初始化,然后我们期待实际获得这些数据,期待 in_msg_body 以获得所有者的地址和带有元数据的Cell。但这只能通过 collection_address, 因此我们要检查它是否等于发件人地址。然后我们就可以初始化 NFT Item了。
  1. int op = in_msg_body~load_uint(32);* ?( s+ Q7 F, ?4 o
  2.   int query_id = in_msg_body~load_uint(64);
    , M3 ^8 _/ p. A" C7 Q  Y

  3. . s; n$ o7 j  k5 M
  4.   if (op == op::transfer()) {
    ) q1 g. L, F. q; Z4 f
  5.     transfer_ownership(my_balance, index, collection_address, owner_address, content, sender_address, query_id, in_msg_body, fwd_fee);
    . Z4 h# u! M1 q& I1 F& x* I' S
  6.     return ();
      k' U$ r4 q) f8 G  w" W
  7.   }
    - H4 e4 G; l+ Z
  8.   if (op == op::get_static_data()) {
    4 R1 N) i- e) w" A/ }
  9.     send_msg(sender_address, 0, op::report_static_data(), query_id, begin_cell().store_uint(index, 256).store_slice(collection_address), 64);  ;; carry all the remaining value of the inbound message
    9 _2 E! v; l4 o& X
  10.     return ();& \9 ], }! o- Q3 [" Z
  11.   }
复制代码
接下来是操作代码,可以是 op::transfer 或 op::get_static_data. 这是我们要处理的两个可能值。当我们说 get_static_data, 我们会立即向发送静态数据请求的用户发送一条信息。我们只是报告数据,所以我们会向他发送 index 和 collection_address. 这是 get_static_data 函数。正如你所看到的,这是我们的第一个 send_ msg 用例
1 L  i4 k% J- F; d5 b$ l& x5 e$ q
  1. () transfer_ownership(int my_balance, int index, slice collection_address, slice owner_address, cell content, slice sender_address, int query_id, slice in_msg_body, int fwd_fees) impure inline {
      D5 ~$ w  C' Q/ N+ G
  2.     throw_unless(401, equal_slices(sender_address, owner_address));
    # J  ?" j$ v/ D5 v6 k) W3 N3 f( ~0 C- U
  3. & C' x( c4 v7 x- v. Z: g
  4.     slice new_owner_address = in_msg_body~load_msg_addr();
    ! r. H: q# Z* ?0 S0 b- Z
  5.     force_chain(new_owner_address);5 ~' T+ g  v8 [2 U9 r# N
  6.     slice response_destination = in_msg_body~load_msg_addr();
    1 T. U+ I3 O; Z0 J1 [8 h) w
  7.     in_msg_body~load_int(1); ;; this nft don't use custom_payload7 G0 k2 F8 b0 Y8 x9 O
  8.     int forward_amount = in_msg_body~load_coins();% |$ Y, x" _$ Q6 t  E& x- V; [
  9.     throw_unless(708, slice_bits(in_msg_body) >= 1);
    ; d7 Z. y& x; X5 Q$ n

  10. # h: b  N) |$ G/ Q
  11.     int rest_amount = my_balance - min_tons_for_storage();
    5 z: ^) C" r. q* o
  12.     if (forward_amount) {5 `3 l" j! A1 R  S0 w; k# R
  13.       rest_amount -= (forward_amount + fwd_fees);- l& @! |3 @& Y3 R. y) ^
  14.     }
      M& C0 X: j" e* Y. Z% G
  15.     int need_response = response_destination.preload_uint(2) != 0; ;; if NOT addr_none: 00
    / w" G* n! E5 E7 k1 x
  16.     if (need_response) {: p# \( |4 w  R; a9 {  f: j+ L6 w% \
  17.       rest_amount -= fwd_fees;
    4 _! r! Y6 n& ?% }( T! o' F
  18.     }1 }# k& z- R! I7 \3 |) A
  19.   s6 P1 z3 z" b
  20.     throw_unless(402, rest_amount >= 0); ;; base nft spends fixed amount of gas, will not check for response
    5 Q8 H  q1 ]9 S2 Y1 E4 y
  21. 2 G2 G8 i9 ^$ s6 N
  22.     if (forward_amount) {+ c7 K8 f3 ~8 _- p
  23.       send_msg(new_owner_address, forward_amount, op::ownership_assigned(), query_id, begin_cell().store_slice(owner_address).store_slice(in_msg_body), 1);  ;; paying fees, revert on errors
    1 i. u+ I' f  N+ S
  24.     }* {( q9 h' Y$ K; a% s- b" C7 f3 v+ c
  25.     if (need_response) {# {1 t. @( C% R: U( `
  26.       force_chain(response_destination);7 L: h  e) d6 Z- O$ j8 D0 W
  27.       send_msg(response_destination, rest_amount, op::excesses(), query_id, null(), 1); ;; paying fees, revert on errors* f$ v: i0 d4 `' {" _' {
  28.     }' L: [* i! E+ x5 d; x
  29. , T1 O( q, S' I. q/ i1 `
  30.     store_data(index, collection_address, new_owner_address, content);% D4 X$ z, }; R! b
  31. }
复制代码
这里有一个转移所有权的功能。首先,我们要检查物品的所有者地址是否就是发送信息的人,因为只有他才能转移所有权。然后,我们读出新所有者的地址。我们 force_chain 以确保他在同一条链上。然后我们检查是否有 response_destination, 因此,一旦我们转移了这个所有权,我们是否应该告诉某人向另一个人发送信息。我们检查 forward_amount 是多少,如果有人要求的话。然后,我们就可以计算出,在发出信息之后,我们就会有足够的钱 forward_amount 到目的地。然后,我们要弄清楚是否需要回应,如果存在 response_destination 。如果存在,其剩余金额必须大于 0。
下面我就向大家简单介绍一下它的工作原理。如果存在 forward_amount ,我们将发送一条信息,通知用户所有权已由该 forward_amount 分配。如果我们需要回应,我们将 force_chain 以确保响应目的地在同一链上。我们发送的回复通常包括超额费用部分和所有额外资金;它们只是被转发到 response_destination. 然后,我们只需使用 new_owner_address 保存数据。基本上,这是唯一会发生变化的地方、 transfer_ownership 是这里的核心函数之一。您拥有一件物品,有时您会将其所有权转让给其他人或出售,这取决于交易的类型。

9 h5 a# E" _+ \& Q$ {* i' x- t
SBT item合约
NFT 可以在另一个层面运作。查看 Getgems (https://getgems.io/) 市场。它们有自己的合约,可以出售这类资产。我们现在不研究这些市场合约。不过,我还是要告诉你一个地方,在那里你可以学到很多关于 NFT 的知识 — GetGems github — 在这里,我还想为大家介绍一种 NFT。
GetGems repository 正在主办一些不同的 NFT 合约、藏品、销售和市场以及一些拍卖活动。这里是了解更多与 NFT 藏品、物品及其相关的合约的好地方。但我想向您展示一份确切的合约,我们可以对其进行深入研究。
sbt-item 是一种灵魂绑定代币(SBT)。灵魂绑定与 NFT 非常相似,但不同之处在于灵魂绑定代币不能擅自转让给其他人。我们仔细看看它的代码。
  1. global int storage::index;) [) c# k- D7 O  b
  2. global int init?;# {; I3 P* g4 {* e2 O4 i
  3. global slice storage::collection_address;
    / [+ }( ?( ]& A5 Q" k
  4. global slice storage::owner_address;
    , |7 V( c, ^2 B# r# l
  5. global slice storage::authority_address;9 o! Z, e* O6 ^
  6. global cell storage::content;
    7 z2 l2 s; t  c, C# f% }
  7. global int storage::revoked_at;
    ! f! L  V# B) }6 b
  8. () load_data() impure {$ p4 `% \: a, O" p$ |9 r  V3 Z
  9.     slice ds = get_data().begin_parse();
    % S/ H. x+ c) L+ p) |( w  N+ t
  10.   L0 |2 x2 |+ H7 ?+ n  v1 ]
  11.     storage::index              = ds~load_uint(64);+ I. A2 k) O! `& F3 j$ G
  12.     storage::collection_address = ds~load_msg_addr();
    . Y/ c9 @, b  L! `% s% `8 j
  13.     init?                       = false;$ Z. s5 O) w  t9 |& n( c, t5 p: M

  14. ! W$ F' }8 j% M" e0 x! ?$ P
  15.     if (ds.slice_bits() > 0) {
      p  b, B. B7 u
  16.         init?                      = true;
    0 a, O6 I$ n' {+ f1 ~6 K: Y
  17.         storage::owner_address     = ds~load_msg_addr();
    ! p' S9 R& K) P1 i7 e
  18.         storage::content           = ds~load_ref();3 p/ h) d7 W7 ~
  19.         storage::authority_address = ds~load_msg_addr();( G% g6 M6 p1 E, \& K8 y. c
  20.         storage::revoked_at        = ds~load_uint(64);6 r! h2 l7 V6 d( I3 e* V% C
  21.     }
    / B- N/ J- q7 a+ V* K
  22. }
复制代码
这里有一些变量: storage::index, storage::collection_address. 我们还定义了其他全局变量,这只是 GetGems 实现的一个例子。你不必用同样的方法。我想在这里重点谈谈其他部分:

: d0 M0 o  u  Q4 F, f
  1. if (op == op::request_owner()) {
    7 k- a1 M/ G1 ]: t
  2.     slice dest = in_msg_body~load_msg_addr();# ]7 ]3 H& S' `9 G4 U
  3.     cell body = in_msg_body~load_ref();$ @5 H/ @5 ~% s; M, j5 E; d/ {
  4.     int with_content = in_msg_body~load_uint(1);
    + }1 N+ _6 J7 i  H
  5. 0 E' a! u- O$ q3 ?; s7 R
  6.     var msg = begin_cell()# ?3 T! T# U7 J
  7.             .store_uint(storage::index, 256)
    $ \. ~8 c  `4 X" g' {: o
  8.             .store_slice(sender_address)
    5 i/ J4 L- O* i8 T3 t  T
  9.             .store_slice(storage::owner_address)
    - b9 f# g, C# H( M5 _4 r
  10.             .store_ref(body)3 u! D, ^: i/ O- Z9 D9 M
  11.             .store_uint(storage::revoked_at, 64)
    : H. f, d2 f0 [$ ^
  12.             .store_uint(with_content, 1);
    ' [6 o5 _* _: f( r% C% ~: a/ L

  13. % D- e, b1 _: F9 x6 J1 J, G3 g- v
  14.     if (with_content != 0) {1 p- w/ e$ p( `# }8 c6 O
  15.         msg = msg.store_ref(storage::content);
    ; W5 p$ C7 M9 G
  16.     }
    0 T; j. Y+ W- p6 R! G5 d5 m+ d

  17. ( u$ |7 k' l4 q$ T* C
  18.     ;; mode 64 = carry all the remaining value of the inbound message1 c3 @/ z2 j) @" N: @
  19.     send_msg(flag::regular() | flag::bounce(), dest, 0, op::owner_info(), query_id, msg, 64);
    - p, E/ v+ A# N9 r0 o
  20.     return ();- T, h7 v2 c- F, |
  21. }% w( S' k0 W( v; W% }% P) _
  22. if (op == op::prove_ownership()) {
    9 @4 _! L+ @; c& W3 s7 }
  23.     throw_unless(401, equal_slices(storage::owner_address, sender_address));& J; \/ e# j. u6 f) Z
  24. ) x. G# s" `7 f' q, X
  25.     slice dest = in_msg_body~load_msg_addr();  `7 h( b. l8 j" Z; g
  26.     cell body = in_msg_body~load_ref();
    0 g4 z8 E* f' U6 k, o
  27.     int with_content = in_msg_body~load_uint(1);( T5 @0 i/ e8 M" u3 @( {( n6 Y
  28. 8 v7 q$ i0 ~. j  c) a
  29.     var msg = begin_cell()& p8 q4 M- i2 ^5 H& o4 m# V
  30.             .store_uint(storage::index, 256)
    4 P# H% {3 j; h- T
  31.             .store_slice(storage::owner_address)' ?, `. c8 U1 Z3 H5 y) s7 f% J
  32.             .store_ref(body)+ o6 }4 \" Q" P: @
  33.             .store_uint(storage::revoked_at, 64)
    ( \  R( i  W2 k4 R
  34.             .store_uint(with_content, 1);8 z% O' S! p' L3 h* p" @

  35. 4 b( s, i. M- h: i
  36.     if (with_content != 0) {2 d! B; _( S$ p% h+ G9 T) P
  37.         msg = msg.store_ref(storage::content);0 p1 g8 o& u5 L- t5 A
  38.     }3 c) k& L+ g3 k1 T7 i* U
  39. 7 e! m9 L5 J' Y9 |. |' R' V
  40.     ;; mode 64 = carry all the remaining value of the inbound message. M3 d. ]2 G0 G- Y, T0 O2 J6 B0 @
  41.     send_msg(flag::regular() | flag::bounce(), dest, 0, op::ownership_proof(), query_id, msg, 64);- B/ H3 ?: J. K  ]. C+ d$ I
  42.     return ();
    - Q/ S& s0 I6 Q- M1 e; r# c
  43. }. e: J" F" J* Q" B2 ]: T
  44. if (op == op::get_static_data()) {
    * K& n; M- i1 Z' c- l6 I
  45.     var msg = begin_cell().store_uint(storage::index, 256).store_slice(storage::collection_address);/ j; R9 H. b$ ]8 s
  46. ) s: H) E' B7 S' s
  47.     ;; mode 64 = carry all the remaining value of the inbound message: K/ ?3 L6 {% Q4 o
  48.     send_msg(flag::regular(), sender_address, 0, op::report_static_data(), query_id, msg, 64);
    4 q+ h- b' Y  Y# p
  49.     return ();
    7 v& n* e! n* s
  50. }
    " _: n' O' T  Z; f
  51. if (op == op::destroy()) {, I& A3 m( o: }
  52.     throw_unless(401, equal_slices(storage::owner_address, sender_address));- x* g! ]* G+ @, E; a. d4 L0 K) V
  53. + t, {/ o; }1 k3 a+ a% ?
  54.     send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);
    5 k3 b. q+ C* R7 Q
  55. 2 x$ }& S) H! I) I
  56.     storage::owner_address = null_addr();4 ?2 G$ E7 k( y( X7 i" f2 p6 j) K
  57.     storage::authority_address = null_addr();1 h7 c4 Q! d" M' ?8 z. n
  58.     store_data();
    - f# @, k9 w: @# n% q  ^! p
  59.     return ();
    + Y) t+ M: K3 g/ Q
  60. }- w( f# g: J9 e
  61. if (op == op::revoke()) {
    5 F% L" H4 `" M3 ^
  62.     throw_unless(401, equal_slices(storage::authority_address, sender_address));
    5 R6 S: H7 Z8 N% w; }
  63.     throw_unless(403, storage::revoked_at == 0);$ i" H# v$ h0 P/ E
  64. 6 |3 t3 ?) @2 F5 j0 w
  65.     storage::revoked_at = now();
    $ T) G: J6 r' q+ Y# U
  66.     store_data();
      h" m& u' K4 ]! D* p9 O# X. D2 b
  67.     return ();7 [3 E  Q7 X: I6 S& i4 l
  68. }* Z9 u6 w6 r# m9 s- @3 U
  69. if (op == op::take_excess()) {
    - c. {8 l5 }6 z/ N: G+ }0 k
  70.     throw_unless(401, equal_slices(storage::owner_address, sender_address));
    9 W0 w$ N: t! C3 m. a

  71. 9 k) J1 B! ^! a8 D9 i; y
  72.     ;; reserve amount for storage
    & y; C" d/ l1 [' o- L0 W% p/ m5 M
  73.     raw_reserve(min_tons_for_storage(), 0);
    6 _% a* w, R7 V" _" H
  74. " @# I' c! M% _0 ~# P/ s# k
  75.     send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);
    " H. s& h$ K3 O* c) U
  76.     return ();
    2 N4 @0 \% \8 s" S( G) {( w
  77. }
    $ b  O6 N7 N% E. r$ n8 M8 w& s
  78. if (op == op::transfer()) {6 U9 ?5 Y5 n5 m$ v
  79.     throw(413);8 n" p. D! S0 T" _+ z
  80. }
复制代码
如你所见,这里有许多注释,但我们不会逐行深入研究。你可以看到,这个通证将被用于多种不同的用途,但与普通的 NFT 不同。这里是合约,你可以申请这个合约的所有者。一些操作代码可以请求所有权证明。您可以获取静态数据。你还可以销毁或撤销这个标记。你可以拿走多余的东西,比如,存放在这里的一些钱。如果你是这个合约的所有者,你就可以申请。但你显然不能转让它,就像你在这里看到的一样。这是一种完全不同类型的代币,如果你对 NFT 感兴趣,就应该在 GetGems 存储库里多花点时间。
我想向你展示的是,NFT collections 的所有代码可以保持不变,但collections合约部署的 item 可以不同。例如,你可以用 SBT item 替换 NFT item。这样您就会有一个 SBT item collection,然后您只需更改  code 来初始化你的 collection:
  1. (slice, int, cell, cell, cell) load_data() inline {
    8 s1 O' P( a" O' A
  2.   var ds = get_data().begin_parse();6 U8 o! d( E% T
  3.   return
    5 `$ b% S& l9 J/ Y1 ~" S
  4.     (ds~load_msg_addr(), ;; owner_address
    " I# J1 Y* \# J" u; }2 [7 P) M
  5.      ds~load_uint(64), ;; next_item_index* G" m# L& t: a6 p/ t3 u
  6.      ds~load_ref(), ;; content2 R. `2 \% d2 _
  7.      ds~load_ref(), ;; nft_item_code
    - ?- v: Y9 L: i. d: r! c4 {' }0 z  a
  8.      ds~load_ref()  ;; royalty_params
    1 _( x" h7 ~5 C, C) D, X4 f: G+ @9 b
  9.      );
    ! v/ o+ k2 E/ {* ~/ m
  10. }
复制代码
这样,您就可以实际部署 SBT item collection。此外,您还可以创建一个独立的没有collection SBT item。我们不会掌握这些item的编程,我只是想向大家展示它在更大范围内的工作原理。
; N% [4 B8 @+ m1 }0 l
结论! r2 A# S' n" X* G
从第 3 章中非常简单的逻辑开始,我们已经了解了 NFT 或 Jettons 等复杂合约中的许多内容。这就是一个很好的例子,你可以一步一步地学习某些语法和概念,然后就可以学习更复杂的逻辑。希望这几节课能帮助你理解 Jettons 和 NFTs 的真正含义,以及如何处理它们的代码库--弄清楚它们是如何工作的,并在使用 TON 构建时学习更多可用的语法和架构。
非常感谢您的关注。对你们来说,这可能是最难的一章,虽然我们没有编写任何代码;我们只是在阅读一些你们从未见过的东西,而且非常复杂。你们能坚持到最后一课,我真的很骄傲,我期待着在接下来的章节中看到你们。希望你们喜欢目前的课程。
! ^" N6 i7 D" w& o) N0 s
: Q0 ~( r/ d3 J7 s* R
4 B# h. `' w5 P6 Q( V7 s
; Z& G/ z5 e0 s8 o

/ Q, M+ X2 W! h/ \& i  D) |# ^- `  S) v( Z
$ v  Z( d- Y3 ~; X) Z: S
: I+ C! Y2 L$ ~5 K1 e
分享到:
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则