# TSL 命名规范(Naming) 本仓库命名规则与 Google C++ Style Guide 对齐:通过名字的“形状”快速判断实体类型(类型/函数/变量/常量等),减少阅读成本。 ## 1. 选名原则 - **可读一致**:名字清晰可读,并随可见范围调整具体程度。 - 可见范围越大(越对外),名字越应具体、少省略。 - 本指南中“对外可见”指:`unit interface`、`class public`、顶层 `function`。 - **标识符语言**:标识符(类型/函数/property/变量/参数等)统一使用英文;禁止中文与拼音(注释可中英混合,见 `docs/tsl/code_style.md`)。 - **少用生僻缩写**:能写全称就写全称。 - 允许使用团队已约定、大家都懂的常见缩写;若缩写不够通用,优先写全称或在评审/文档中先达成约定。 - **驼峰/帕斯卡中的缩写规则**:缩写(首字母缩写/词组缩写)在 `PascalCase`/`camelCase` 中**按一个单词处理**,写成“首字母大写其余小写”,不要写一串全大写。 - 示例:`UserId`(不是 `UserID`)、`UrlTable`(不是 `URLTable`)、 `StartRpcServer`(不是 `StartRPCServer`)、`HttpClient`(不是 `HTTPClient`)。 - **避免无意义词**:如 `data`、`info`、`tmp`、`handle` 等。 - 可以作为限定词的一部分(例如 `user_data`),但不要单独用作名字(例如仅叫 `data`)。 ## 2. 命名风格总览 对于以下规则,“单词”指英文中不带空格的词。 - `snake_case`:全小写,下划线分隔单词,用于普通变量/参数等;私有类成员变量使用 `snake_case_`(见 5.2)。 - `PascalCase`(`UpperCamelCase`):每个单词首字母大写,无下划线,用于类型、顶层函数/“动作型”方法、property,以及(少量)公有成员字段(访问器/设置器方法见 7 的例外约定)。 - 自定义标识符只使用本指南约定的 `PascalCase`/`snake_case`;`lowerCamelCase` 仅用于沿用内置/标准库/第三方 API 的既有命名。 **大小写与关键字约定** - TSL 语言大小写无关,但本指南仍要求按约定使用大小写以提升可读性;不要用仅大小写不同的名字区分不同实体;同一标识符在仓库中应保持一致写法。 - 所有语法关键字统一使用全小写书写,例如 `if`、`for`、`class`、`function`、`unit`、`return` 等。 - 调用内置/标准库/第三方 API 时,推荐保持对方官方大小写形式(`aaBBCC`/lowerCamelCase),例如 `getSysParams("xxx")`;自定义 wrapper 仍按本指南使用 `PascalCase`。 ## 3. 类型命名(Type Names) TSL 的顶层声明只有三种:`class`、`unit`、`function`(仅适用于 `.tsf`)。因此 `.tsf` 文件基名必须与顶层声明同名(见“4. 文件命名与顶层声明”)。 - **类(class)与单元(unit)**使用 `PascalCase`,不带下划线;名称应为名词/名词短语(通常单数),避免动词开头。 - 不推荐 `*Unit` 作为 `unit` 的后缀(`unit` 本身已表达语义);需要表达用途时,可使用 `*Shared`/`*Common`/`*Enums` 等更具体后缀(按团队约定)。 - **顶层函数(function)**使用 `PascalCase`;名称优先动词/动词短语(例如 `Load`/`Parse`/`Build`),详见函数命名章节。 - 示例:`UserAccount`、`OrderShared`、`LoadMarketData()`。 ## 4. 文件命名与顶层声明(File Names) TSL 的语法要求(仅 `.tsf`):每个 `.tsf` 文件只能有一个顶层声明,且**文件基名必须与该顶层声明名字一致**。 - 顶层声明可能是 `class`、`unit` 或 `function`(见类型命名)。 - `.tsf` 代码文件:用于库/模块等“顶层声明”的承载文件;顶层声明可为 `class`/`unit`/`function`,文件基名需与之同名。 - `.tsl` 脚本文件:用于入口/编排层;允许直接写语句(如 `a := 1; echo a;`),不要求顶层声明,也不强制文件基名与函数名一致;可复用逻辑应下沉到 `.tsf`(见 `docs/tsl/code_style.md`)。 - 注:`.tsf` 也是 TSL 源文件,命名/风格与 `.tsl` 遵循同一套规则。 - **硬规则(仅 `.tsf`)**:重命名顶层声明时必须同步重命名文件基名,否则语法/加载规则无法识别;批量重命名可参考 `$bulk-refactor-workflow`。 命名建议: - 基名统一使用 `PascalCase`,与顶层声明的推荐写法一致。 - 示例: - `LoadMarketData.tsl` 中定义 `function LoadMarketData(...)`. - `UserAccount.tsf` 中定义 `type UserAccount = class ... end;`. - `DocxEnumerations.tsf` 中定义 `unit DocxEnumerations; ... end.` - `ParseConfig.tsf` 中定义 `function ParseConfig(...)`. 注:TSL 大小写无关,实际编译时按大小写比较不会出错,但仍应保持文件名与声明名的推荐写法一致以便检索与协作。 ## 5. 变量命名(Variable Names) ### 5.1 普通变量与参数 - **局部变量、函数参数、非成员变量**使用 `snake_case`。 - 若参数名与 TSL 关键字冲突导致编译失败,使用前导下划线的 `snake_case` 作为例外,例如 `_type`(`type` 是常见冲突关键字)。 - 前导下划线 `_` **仅用于上述关键字冲突的参数场景**,不要用于其他局部变量、成员变量、函数/类型/单元名称或全局变量。 - 建议把单位写进名字(尤其时间/金额/比例):例如 `timeout_ms`、`spread_bp`、`ratio_pct`。 - 示例:`table_name`、`max_retry_count`、`user_id`(短名例外见 5.5)。 ### 5.2 类成员(Class Data Members) - **私有成员变量**使用 `snake_case_`(尾随下划线)。 - 尾随下划线 `_` **仅用于私有成员变量**,不要用于局部变量、参数、公有成员字段、property 名称或顶层全局变量。 - **公有成员变量**若必须存在,使用 `PascalCase`;但**不推荐外部直接访问公有字段**。 - 对外暴露的成员优先使用 **property**: - property 名称使用 `PascalCase`(视为对外 API)。 - property 的 `read/write` 指向真实成员(通常为私有 `snake_case_`)。 - 布尔 property 使用 `Is`/`Has`/`Can`/`Should` 等前缀的 `PascalCase`(例如 `IsReady`),对应私有成员可用 `is_ready_` 等。 - 示例: ```tsl type User = class public property UserId read user_id_ write user_id_; property IsReady read is_ready_; // bool property example private user_id_; is_ready_; end; ``` ### 5.3 全局/静态变量 - 不推荐使用顶层全局/静态可变变量;优先封装到 `unit`/`class` 中,通过函数或 property 访问。 - 合理例外(仍需集中管理,避免到处读写): - 进程级只读配置缓存(启动后不再变)。 - 有上限/可清理的缓存(明确容量/TTL/清理点)。 - 指标/计数器(只增不减,或集中在少数写入点更新)。 - 若必须声明顶层全局/静态可变变量: - 命名仍使用 `snake_case`(不使用 `g_` 前缀)。 - 必须在声明处写注释说明:它是什么、用于什么、以及(如不明显)为什么需要是全局/静态。 - 建议补充写入点与生命周期:谁会写、何时写、何时清理/重置;如涉及并发,写明并发假设/保护方式。 - 不要在注释/日志中写入任何敏感信息(参考 `.agents/tsl/auth.md`)。 - 示例(注释模板,按需裁剪): ```tsl // : // Used for: // Global because: ``` - 全局/静态常量仍按常量规则使用 `kPascalCase`(建议同样写一句用途注释,便于检索与维护)。 ### 5.4 布尔变量 - 布尔变量建议区分两类语义: - **状态/谓词(predicate)**:描述“是否满足某条件/是否处于某状态”,使用 `is_ / has_ / can_ / should_` 等前缀表达语义。 - 示例:`is_ready`、`has_error`、`can_retry`。 - **选项/开关(flag/option)**:描述“是否启用某行为/模式”,允许使用不带 `is_` 的 `snake_case` 短语(更贴近配置项语义)。 - 示例:`dry_run`、`include_header`、`enable_cache`、`use_cache`。 - 尽量使用正向命名,避免双重否定:`is_valid` 优于 `is_not_valid`;`disable_cache` 这类命名需谨慎(容易在调用点读错)。 ### 5.5 短名例外 - 在极小作用域内可用习惯短名:仅限 `for/while` 的索引变量或约 5–10 行内的临时值。 - 允许短名清单(建议严格执行):`i/j/k`(索引)、`n`(计数);其他一律使用有含义的名字。 - 作用域一旦扩大(跨多个分支/循环、跨函数、跨文件),必须改为有含义的名字。 ### 5.6 集合与复数命名(Collections) - **数组/列表/可迭代集合**使用复数名词的 `snake_case`:`users`、`order_items`。 - 若复数形式不直观或为不可数名词,使用后缀明确类型:`news_list`、`price_items`。 - **映射/字典(key→value)**使用 `snake_case` 并加后缀 `_map`;必要时可用 `_by_` 表达键语义(仍需保留 `_map`):`user_map`、`price_by_symbol_map`。 - **集合/去重集合**使用后缀 `_set`:`user_id_set`、`symbol_set`。 ## 6. 常量命名(Constant Names) - **模块级/全局级固定常量**(写死、与入参无关、加载后不变)使用 `kPascalCase`,以 `k` 开头。 - 示例:`kDaysInAWeek`、`kAndroid8_0_0`。 - 常量名建议带单位(尤其时间/金额/比例):例如 `kTimeoutMs`、`kSpreadBp`、`kRatioPct`。 - 对于**局部 const 但值来自参数/运行时**的变量: - 可用普通变量名 `snake_case`; - 不要用 `k` 前缀误导读者认为其全局固定。 ### 6.1 单元枚举模拟(Unit Enumerations) TSL 没有内置 `enum`,推荐使用 `unit` + `const` 在 `interface` 区域模拟枚举集合。 - `unit` 名称使用 `PascalCase`,建议以 `Enumerations` 结尾表达用途(例如 `DocxEnumerations`)。 - 枚举值使用 `const` 定义并放在 `interface` 中: - 项目自定义枚举值:默认使用 `kPascalCase`(属于模块级固定常量)。 - 外部/互操作枚举值:允许沿用对方既有前缀与命名(例外场景)。 示例: ```tsl unit AlertEnumerations; interface const kAlertLevelAll = -1; const kAlertLevelNone = 0; end. ``` ```tsl unit DocxEnumerations; interface // WdAlertLevel const wdAlertsAll = -1; end. ``` ## 7. 函数与方法命名(Function Names) - 顶层函数与“动作型/业务型”方法使用 `PascalCase`。 - 访问器/设置器方法(仅当 property 无法表达语义时才使用)允许使用 `snake_case` 的小写形式,以贴近“字段/状态”的语义(按团队约定的例外)。 - **特殊函数/运算符重载为语法固定名,必须使用全小写**: - 构造/初始化函数:`create`。 - 析构/释放函数:`destroy`。 - 运算符重载:`operator+()` 等,按语法使用小写 `operator()` 形式。 - 示例:`AddTableEntry()`、`DeleteUrl()`、`OpenFileOrDie()`。 - **推荐使用 property 语法**对外暴露访问器:property 名 `PascalCase`,`read/write` 绑定成员变量(见类成员章节)。 - 不推荐新增显式 getter/setter;仅当 property 无法表达语义时,才使用 getter/setter: - getter:与字段同形的 `snake_case`(如 `count()`、`is_ready()`)。 - setter:使用 `set_` 前缀的 `snake_case`(如 `set_count(x)`、`set_ready(x)`)。 ## 8. 宏与编译期开关(Macro Names) - 能不用宏就不用。 - 若项目不支持宏/预处理,本节可忽略;若支持且必须使用,命名为全大写加下划线,并带项目/业务前缀: - `_ENABLE_FOO`、`_USE_BAR`、`_ROUND(x)`。 ## 9. 例外(Exceptions) - 当命名需要与外部既有 API/协议保持一致时,可沿用对方风格(例如对接 C/C++ 库、历史接口、跨语言互操作代码等)。 - 不要为了“命名隔离”引入不必要的 wrapper/嵌套;优先保证语义清晰、可检索,并在必要处用注释说明“该命名来自外部约束/协议”。