# TSL 变量与常量 文档类型:语法主线 是否可直接用于生成代码:是 是否含可直接照写示例:是 是否含不可照写反例:是 遇到不确定时:先按本页候选页继续判断;[03_values_and_literals.md](03_values_and_literals.md)、[06_expressions_and_operators.md](06_expressions_and_operators.md)、[05_functions_and_calls.md](05_functions_and_calls.md)、[16_lexical_structure_and_compile_options.md](16_lexical_structure_and_compile_options.md)、[11_pitfalls.md](11_pitfalls.md);仍不命中时回到语法路由中心 [index.md](index.md);如果问题已经超出语法层,回到 TSL 总入口 [../index.md](../index.md) 这一篇收拢变量与常量初始化规则。 ## 本篇职责 回答“普通变量怎样直接使用、`var` 在什么位置出现、常量必须怎样初始化、哪些名字一旦绑定就不能再赋值”。 ## 智能体变量/常量判断流程 1. 普通变量默认直接用 `:=` 首次赋值,不要先补一个没有需求证据的 `var` 段。 2. 只有用户要求显式声明或遇到 `{$explicit+}` 时才优先写 `var`。 3. 常量声明必须同时初始化;默认只生成 `const name = value;`,顶层脚本常量需要后续脚本语句时,仍按 `.tsl` 语句区规则组织。 4. 多参数赋值按 `[a, b] := array(...)` 写;单变量拆包必须写成 `[name, ] := array(...)`。 5. 如果问题已经变成表达式求值、函数参数、运行时类型 / 转换、或 `{$explicit+}` 之外的编译选项,不要留在变量页硬推断;分别跳到 [06_expressions_and_operators.md](06_expressions_and_operators.md)、[05_functions_and_calls.md](05_functions_and_calls.md)、[17_types_and_conversions.md](17_types_and_conversions.md) 或 [16_lexical_structure_and_compile_options.md](16_lexical_structure_and_compile_options.md)。 6. 没有对应代码块时不要发明变量/常量写法;尤其不要从 Pascal 的声明习惯反推 TSL 必须先声明变量。 ## 核心规则 - 默认变量模型是“直接赋值即得到变量”,不要求先写 `var`。 - 例如:`a := 1; b := array(1, 2, 3);` 这种写法可直接编译。 - `var name;` 只作为显式声明写法,不能当成默认必需步骤。 - `const name = value;` 是常量初始化的默认生成写法;`const` 不能只声明名字而不初始化。 - 文档规则: - `const name = value;` 适合顶层脚本常量、函数内部 `const` 段、`unit` 接口常量、类成员常量。 - 顶层脚本里的 `const name = value;` 可以放在后续脚本语句之前;单独只写一行 `const name = value;` 不作为可运行脚本骨架。 - 函数内部 `const = expr`、以及松散脚本顶层的 `const = expr`,都可以把右侧写成常量表达式。 - 用 `const =` 初始化的这些常量不能再次赋值。 - 多参数赋值文档明确写法是 `[a, b] := array(...)`。 - 左侧只有一个变量时,末尾逗号不能省略,必须写成 `[a, ] := array(...)`。 - 当左侧变量数大于右侧数组长度时,多出的变量会得到 `nil`。 - 右侧数组元素也可以是数组;拆出来的变量会直接得到对应子数组。 - 多参数赋值也可以出现在函数调用参数里。 - `{$explicit+}` 开启后,后续变量必须先用 `var` 声明;未声明变量会报 `variable not defined`。 ## 可直接照写示例 使用这些示例时遵守: - 默认生成普通变量时直接用 `:=`;只有用户明确要求或 `{$explicit+}` 场景才复制 `var`。 - 普通示例默认按 `.tsl` 脚本语句区书写;需要函数或类型时,放在后置声明区。 - 复制常量示例时只使用 `const name = value;`。 ### 普通变量与显式声明 默认变量模型: 代码块身份:可直接照写示例 ```tsl a := 1; b := array(1, 2, 3); writeLn(a); writeLn(b[1]); ``` 代码块身份:输出片段 ```text 1 2 ``` 显式 `var` 写法: 代码块身份:可直接照写示例 ```tsl var a; a := 1; ``` `{$explicit+}` 下的显式声明: 代码块身份:可直接照写示例 ```tsl {$explicit+} var a; a := 1; writeLn(a); ``` 代码块身份:输出片段 ```text 1 ``` ### 常量初始化 顶层最稳的常量写法: 代码块身份:可直接照写示例 ```tsl const value = 1; echo value; ``` 代码块身份:输出片段 ```text 1 ``` 顶层 `const =` 也可以写常量表达式: 代码块身份:可直接照写示例 ```tsl const value = 1 + 2 * 3; writeLn(value); ``` 代码块身份:输出片段 ```text 7 ``` 函数内部 `const` 段: 代码块身份:可直接照写示例 ```tsl function Demo(); const max_retries = 1 + 2 * 3; begin return max_retries; end; ``` `unit` 接口常量: 代码块身份:可直接照写示例 ```tsl unit DemoUnit; interface const value = 1; function GetValue(); implementation function GetValue(); begin return value; end; end. ``` 类成员常量: 代码块身份:可直接照写示例 ```tsl type DemoType = class public const value = 1; end; ``` 顶层 `const =` 只有在“后面继续接可执行语句”的脚本模型里才成立: 代码块身份:可直接照写示例 ```tsl const max_retries = 3 + 4; value := max_retries; ``` ### 多参数赋值 多参数赋值: 代码块身份:可直接照写示例 ```tsl [r1, r2] := array(1, 3, 5, 7, 9); writeLn(r1); writeLn(r2); ``` 代码块身份:输出片段 ```text 1 3 ``` 单变量拆包时,末尾逗号不能省略: 代码块身份:可直接照写示例 ```tsl [re, ] := array(1, 2, 3, 4); writeLn(re); ``` 代码块身份:输出片段 ```text 1 ``` 左侧变量比右侧数组更长时,多出的变量为 `nil`: 代码块身份:可直接照写示例 ```tsl [r1, r2] := array(1); writeLn(r1); writeLn(r2 = nil); ``` 代码块身份:输出片段 ```text 1 1 ``` 右侧元素也可以是数组: 代码块身份:可直接照写示例 ```tsl [r1, r2] := array((1, 2), (3, 4)); writeLn(r1[0]); writeLn(r1[1]); writeLn(r2[0]); writeLn(r2[1]); ``` 结果说明: - 依次输出 `1`、`2`、`3`、`4` 代码块身份:输出片段 ```text 1 2 3 4 ``` 函数返回数组后也可以直接拆包: 代码块身份:可直接照写示例 ```tsl [error, re] := PairAdd(3, 4); writeLn(error); writeLn(re); function PairAdd(a, b); begin return array(0, a + b); end; ``` 结果说明: - 依次输出 `0`、`7` 代码块身份:输出片段 ```text 0 7 ``` 函数参数里也可以使用多参数赋值: 代码块身份:可直接照写示例 ```tsl writeLn(Test(e := 3, [f, g] := array(1, 2), g)); function Test(a, b, c); begin return a + b + c; end; ``` 结果说明: - 输出 `6` 代码块身份:输出片段 ```text 6 ``` ## 默认生成模板 普通变量和顶层常量的默认骨架如下: 代码块身份:可直接照写示例 ```tsl const max_retries = 3; counter := max_retries; items := array(1, 2, 3); ``` ## 禁止项 - 以为普通变量必须先写 `var` 才能使用。 - 把常量初始化写成普通变量赋值风格。 - 以为 `const =` 顶层单独写一行就一定成立。 - 以为 `const` 可以只声明名字,不写初始化表达式。 - 以为单变量拆包可以写成 `[a] := array(...)`。 - 以为 `{$explicit+}` 开启后仍然可以继续直接写未声明变量。 代码块身份:反例 / 不可照写 ```text const value = 1; value := 2; ``` 上面这类写法会编译失败,报错点在重新赋值这一行。 代码块身份:反例 / 不可照写 ```text const value = 1; ``` 上面这类“顶层单独一行”的 `const =` 也会编译失败。 代码块身份:反例 / 不可照写 ```text const value; ``` 上面这类没有初始化表达式的 `const` 会编译失败;`const` 必须写初始化值。 错误原因是常量语句缺少初始化表达式。 代码块身份:反例 / 不可照写 ```text [re] := array(1, 2, 3); ``` 上面这种单变量拆包写法会报 `left side can not be assign to`。单变量时必须写成 `[re, ] := ...`。 代码块身份:反例 / 不可照写 ```text {$explicit+} a := 1; ``` 上面这类写法会编译失败,主因是 `variable not defined`。 代码块身份:输出片段 ```text variable not defined ```