本帖最后由 riyad 于 2025-3-25 02:43 编辑
+ C; ~, l$ j3 e7 H4 n5 u. S2 j; B+ P- E
TON 智能合约中的算术运算始终使用整数,从不使用浮点数,因为浮点数是不可预测的。 因此,重点应放在整数及其处理上。 Int 是一个 257 位的有符号整数类型。 它能够存储 −2256−2256 和 2256−12256−1 之间的整数。 + W# ~) w+ J1 g$ c+ A
表示法 Tact 支持多种方式编写 Int 的原始值作为整数字面量。 3 C# M/ a1 V# t7 D& U
大多数符号允许在数字之间添加下划线 (_),但下列符号除外: : U; n C7 ^& t2 X
- 字符串表示法,如 NanoToncoins 案例所示。
- 带前导零的十进制数 00。一般不鼓励使用,参见 下文。
8 R8 O. G7 a' Y: K$ F% v+ L$ i# ?3 e- G& I- @. E
0 I0 O2 W+ K" I1 U
) ^ w2 Y" V& P( \$ a, d
/ X5 P9 W" x s. G此外,不允许在 4__24__2 中连续使用多个下划线,或在 42_42_ 中使用尾部下划线。
2 o, q _6 h% l C* t8 C9 `% B U& l6 \ a) r' h, ~
y) F& V3 w) w十进制最常见、最常用的数字表示方法,使用十进制数字系统: 123456789123456789。
6 N" b8 k' E' i您可以使用下划线(_)来提高可读性: 123_456_789123_456_789 等于 123456789123456789。 或者,您也可以在数字前加上一个 00,这样就禁止使用下划线,只允许使用十进制数字:0123=123.0123=123. 注意,强烈建议使用这种带前导零的符号,因为可能会与 TypeScript 中的八进制整数字面混淆,而 TypeScript 通常与 Tact 一起用于开发和测试合约。; m/ a y+ V; f1 [0 G6 ?
6 i) H& [7 s; I E1 E, n# h/ ?十六进制使用十六进制数字系统表示数字,用 0x0x(或 0X0X)前缀表示:0xFFFFFFFFF0xFFFFFFFFF。 使用下划线(_)提高可读性:0xFFF_FFFF_FFF0xFFF_FFFF_FFF 等于 0xFFFFFFFFF0xFFFFFFFFF。 八进制使用八进制数字系统表示数字,用 0o0o(或 0O0O)前缀表示:0o7777777770o777777777。 使用下划线(_)提高可读性:0o777_777_7770o777_777_777 等于 0o7777777770o777777777。 二进制使用二进制数字系统表示数字,用 0b0b(或 0B0B)前缀表示:0b1111111110b111111111。 使用下划线(_)提高可读性:0b111_111_1110b111_111_111 等于 0b1111111110b111111111。 NanoToncoins与美元的运算要求小数点后保留两位小数——这些用于表示美分(cents)的值。 但是,如果我们只能用整数来表示数字 $1.251.25 ,我们该如何表示呢? 解决的办法是直接使用 cents。 这样, $1.251.25 就变成了 125125 美分。 我们只需记住最右边的两位数代表小数点后的数字。 同样,在使用 TON 区块链的主要货币 Toncoin 时,需要九位小数,而不是两位小数。 可以说,nanoToncoin是Toncoin的 1109th1091th。 因此, 1.251.25 Toncoin 的数量,可以用 Tact 表示为 ton("1.25"),实际上就是数字 12500000001250000000。 我们称这样的数字为_nanoToncoin(s)(或_nano-ton(s))而不是_美分。 序列化将 Int 值编码为持久状态(contracts 和 traits 的字段)时,通常最好使用比 257257-bits 更小的表示形式,以降低存储成本。 这些表示法的使用也被称为 “序列化”,因为它们代表了 TON 区块链运行的本地TL-B类型。 持久状态大小在状态变量的每个声明中都会在 as关键字后指定: - contract SerializationExample { Z8 f) D3 L8 b
- // persistent state variables
1 i0 }- y( ]) k0 v - oneByte: Int as int8 = 0; // ranges from -128 to 127 (takes 8 bit = 1 byte)
* _% z4 s: Y1 [5 l - twoBytes: Int as int16; // ranges from -32,768 to 32,767 (takes 16 bit = 2 bytes)' q$ F8 r) U0 H; y1 a
- & Q* ]& m7 K7 ^! c) s Q8 J
- init() {" H! x$ m! w. z* k
- // needs to be initialized in the init() because it doesn't have the default value
8 u' t- b* m4 f* U R P, Z - self.twoBytes = 55*55;
; H; | v& p6 X6 O3 ]; U3 X - }! ~" O1 U6 A$ e- [( J4 @
- }
复制代码 . z. s. S* k- ?/ `0 q# z
整数序列化也适用于 Structs 和 Messages 的字段,以及 maps 的键/值类型: : n5 Z8 L5 E& t! H
- struct StSerialization {4 D, }1 T4 f( o: D/ h+ H
- martin: Int as int8;, v. w0 u+ C- T. V a4 T
- }2 h2 z! R% ?; _" K
2 R) ]# u; K. E1 n/ j- message MsgSerialization {, [" N+ u0 p& D: ^
- seamus: Int as int8;& \5 S& a t0 }; v2 i+ {# w
- mcFly: map<Int as int8, Int as int8>;1 |# ~( K: a: _/ S
- }
复制代码动机很简单: - 在状态中存储 10001000 个 257257 位的整数成本大约为每年 0.1840.184 TON。
- 相比之下,存储 10001000 3232-bit 整数每年只需花费 0.0230.023 ton 。
6 c; q. T0 h3 w. K% o& D9 k % T' x# r4 F, R0 a/ Y
$ F! b1 r( Y: t7 T+ H4 N/ K# Q2 U" e6 F" v% z
\" e9 i# M; M2 o0 C% Z |