playbook/docs/tsl/syntax/12_pitfalls.md

7.6 KiB
Raw Permalink Blame History

Pitfalls

文档类型:反例索引页 是否可直接用于生成代码:否 是否含已验证可执行示例:否 是否含已验证反例:是 遇到不确定时跳转到:03_core_model.md06_functions_and_calls.md09_objects_and_classes.md

手册位置:第 12 篇,共 32 篇。上一篇:11_runtime_context_and_with.md。下一篇:13_matrix_and_collections.md

这一篇不讲新知识,只做反例索引。

这一篇解决什么问题

回答“哪些写法最容易凭直觉写出来,但在 TSL 里会编译失败、运行出错,或语义并不可靠”。

这页怎么用

  • 先按主题扫一遍,再回到对应正文看正确写法。
  • 这里不重复讲完整规则,只保留“错法 -> 正确页”的索引。
  • 只有已经单独验证过的误写,才会列在这里。

已验证反例索引

文件模型与基础值

1. 把裸 class Name 当成顶层类声明

代码块身份:反例 / 不可照写

class Person
end;

这会编译失败。正确页:见 09_objects_and_classes.md

2. 把 = 当成赋值

代码块身份:反例 / 不可照写

a = 1;

这会编译失败。正确页:见 07_expressions_and_operators.md

3. 把字符串当成 0 基下标

代码块身份:反例 / 不可照写

s := "ABC";
WriteLn(s[0]);

这会在运行时报字符串下标越界。正确页:见 04_values_and_literals.md

4. 把普通字符串里的 "\uXXXX" 直接当成和 L"\uXXXX" 一样的单字符宽串

代码块身份:反例 / 不可照写

WriteLn(Length("\u0041"));
WriteLn(Length(L"\u0041"));

这不是同一件事。当前解释器里前者输出 2,后者输出 1。正确页:见 04_values_and_literals.md

5. 把 U"" 误当成和 L"" 一样的宽串

代码块身份:反例 / 不可照写

WriteLn(IfWString(U"\u5929\u8F6F"));
WriteLn(IfWString(L"\u5929\u8F6F"));

这不是同一回事。当前解释器里前者输出 0,后者输出 1。正确页:见 04_values_and_literals.md

6. 把带 #0 / \0 的字符串当成会自动截断的零结尾串

代码块身份:反例 / 不可照写

s := "A"#0"B";
WriteLn(Length(s));

这不要按 C 风格字符串去理解。当前解释器里这里输出 3,说明 #0 仍是字符串内容的一部分。正确页:见 04_values_and_literals.md

7. 把顶层函数定义和松散语句混写

代码块身份:反例 / 不可照写

function Add(a, b);
begin
    return a + b;
end;

value := Add(1, 2);

这会编译失败。正确页:见 03_core_model.md

8. 顶层单独写 const Name = value;

代码块身份:反例 / 不可照写

const value = 1;

这在当前解释器里会编译失败。正确页:见 05_variables_and_constants.md

函数与调用

9. 把 a = 1 当成命名参数

代码块身份:反例 / 不可照写

Pack(a = 1, b = 2)

这不要当成可靠的命名参数写法。它虽然可能编译通过,但实测返回结果不对。正确页:见 06_functions_and_calls.md

10. 把 like 当成 SQL % / _ 通配

代码块身份:反例 / 不可照写

WriteLn("abc" like "a%");

这在当前解释器里输出 0,不要按 SQL LIKE 去理解。当前手册里,like 的右侧按正则模式解释。正确页:见 07_expressions_and_operators.md

11. 把普通函数默认参数规则直接套到 unit interfaceconst 默认参数

代码块身份:反例 / 不可照写

unit UnitConst;
interface
const CS = 888;
function F(a, b = 100, c = CS);

这不要直接当成已经等价于普通函数默认参数规则的写法。在当前解释器里,对应的 F(1) 实测输出是 101,不是按 CS = 888 补值。正确页:见 06_functions_and_calls.md10_units_and_scope.md

12. 把 external 后面的 DLL 名直接写成字符串拼接表达式

代码块身份:反例 / 不可照写

function TickFromExpr(): int64; stdcall; external "kernel32"$"."$"dll" name "GetTickCount64";

这会编译失败,错误是 dll filename const string not found after external。正确页:见 06_functions_and_calls.md

Unit 与查找路径

13. 把“文件就在当前目录”当成当前命令行解释器已经能找到对应 unit / .tsf

代码块身份:反例 / 不可照写

program test;
uses DemoUnit;
begin
    WriteLn(Ping());
end.

如果 DemoUnit.tsf 所在目录没有进入 -LIBPATHtsl.confLibpath= 或运行时查找路径,上面这种写法虽然语法能过,但运行时仍然找不到 Ping()。正确页:见 10_units_and_scope.md

14. 把函数体或类定义体里的 uses 写在第一条语句之后,或在同一作用域里重复写第二个 uses

代码块身份:反例 / 不可照写

function Run();
begin
    a := 1;
    uses DemoUnit;
    return Ping();
end;

type Worker = class
    uses UnitA;
    uses UnitB;
end;

上面这两类写法都没有通过。函数里的错法会报 invalid statement,类里的错法会报 invalid class definition。正确页:见 10_units_and_scope.md

15. 把限定读取 DemoUnit.VarName 误当成也支持限定赋值

代码块身份:反例 / 不可照写

DemoUnit.UnitCounter := 13;

这在当前解释器里会编译失败,错误是左值不可赋值。正确页:见 10_units_and_scope.md

类与对象

16. 把裸类名当成 class function 或静态字段的直接访问入口

代码块身份:反例 / 不可照写

MathBox.Add(1, 2);

THuman.mCount := 7;

这两种写法在当前解释器里都没有通过。当前已验证可用的是先拿到类类型,再用 class(MathBox).Add(...) / FindClass("MathBox").Add(...) 调类方法,以及 class(THuman).mCount 访问静态字段。正确页:见 09_objects_and_classes.md

17. 把 private / protected create 当成一定会执行的构造函数

代码块身份:反例 / 不可照写

program test;
type A = class
public
    value;
private
    function create(v);
    begin
        self.value := v;
    end;
end;
begin
    a := new A(111);
    WriteLn(a.value);
end.

这不会报“不能创建对象”,但 create 不会执行;上面的 a.value 实测输出 <NIL>。正确页:见 09_objects_and_classes.md

18. 在子类里把父类 private 成员当成 protected 成员用

代码块身份:反例 / 不可照写

program test;
type A = class
private
    secret;
end;
type B = class(A)
public
    function SetSecret(v);
    begin
        self.secret := v;
    end;
end;

这会在执行时报对象成员访问错误。正确页:见 09_objects_and_classes.md

跳转指引