# TSL 矩阵深水专题 文档类型:语法深水专题 是否可直接用于生成代码:是 是否含可直接照写示例:是 是否含不可照写反例:是 遇到不确定时:先按本页候选页继续判断;[12_matrix_and_collections.md](12_matrix_and_collections.md)、[23_fmarray.md](23_fmarray.md);仍不命中时回到语法路由中心 [index.md](index.md);如果问题已经超出语法层,回到 TSL 总入口 [../index.md](../index.md) 这一篇只讲矩阵专用语法主干:矩阵初始化、数列构造、矩阵逆/广义逆、矩阵尺寸与索引、矩阵遍历、子矩阵和 `mfind` 查找。它和 [12_matrix_and_collections.md](12_matrix_and_collections.md) 的分工是:`12` 讲普通数组与集合关系,这一篇讲矩阵专用构造、遍历、子矩阵和矩阵查找接口。 ## 本篇职责 回答“怎样直接构造全零矩阵、全一矩阵、随机矩阵、单位矩阵、空矩阵和数列数组,怎样写矩阵逆/广义逆,怎样拿到矩阵的行数、列数、行索引和列索引,怎样遍历矩阵、取/改子矩阵,以及怎样用 `mfind` 找到或替换符合条件的单元格”。 ## 智能体矩阵深水判断流程 1. 先判断要写矩阵初始化、数列构造、矩阵逆/广义逆、矩阵尺寸与索引读取、矩阵遍历、子矩阵,还是 `mfind` 查找/替换。 2. 基础数组和矩阵样比较先回看 `12_matrix_and_collections.md`。 3. `mrows` / `mcols` / `msize` 等函数只照文档返回形态写。 4. 需要逐单元执行语句块时用 `matrix::begin ... end`;需要把表达式结果写回每个单元时用 `matrix ::= expression`。 5. 需要遍历到嵌套数组最深层时用 `matrix:.begin ... end` 或 `matrix:.= expression`。 6. 子矩阵范围使用 `row_start:row_end`、`:`、下标数组和列范围组合;不要把字符串键当作子矩阵范围序号。 7. `mfind(...)` 用于把符合条件的单元格转换成下标列表,或同时返回原值/替换原值。 8. 不要把列索引数组误当成单个数字。 9. 没有对应代码块时不要发明矩阵深水写法。 ## 核心规则 - 矩阵初始化函数的参数规格见 [../reference/catalog/math.md](../reference/catalog/math.md);本页只保留矩阵行为示例和返回形态边界。 - `zeros(...)`、`ones(...)`、`rand(...)`、`nils(...)`、`eye(...)` 都可以直接用于矩阵初始化。 - `zeros(3)`、`ones(3)`、`nils(2)` 这类单参数写法可以直接生成一维结果。 - `zeros(2, 3)`、`rand(2, 3)` 这类双参数写法可以直接生成二维矩阵。 - `zeros(2, array("A", "B"))` 这种写法可以直接生成带列名的二维结果。 - `eye(3)` 生成的是 `3 x 3` 单位矩阵,不是一维数组。 - `->` 用来生成数列;默认步长是 `1`,也可以显式传入步长和索引数组。 - 在矩阵语境里,`!A` 是一元倒数运算符作用于矩阵的形态,用于矩阵逆/广义逆;非方阵输入可以返回行列数互换后的广义逆结果。 - `msize(...)`、`mrows(...)`、`mcols(...)` 的参数规格见 [../reference/catalog/system.md](../reference/catalog/system.md)。 - `msize(matrix_value)` 返回 `array(行数, 列数)`。 - `msize(matrix_value, 1)` 返回行索引数组和列索引数组。 - `mrows(matrix_value)` / `mcols(matrix_value)` 默认返回数量;第二个参数写成 `1` 时返回索引数组。 - `mrows(matrix_value, 1)` / `mcols(matrix_value, 1)` 的返回值可用于索引匹配;不要把它们当成数量。 - `matrix::begin ... end` 是矩阵遍历语句块,最多按二维遍历;语句块里可用 `mcell`、`mrow`、`mcol`。 - `matrix ::= expression` 是矩阵遍历赋值,把表达式结果逐单元写回原矩阵;它不是语句块。 - `matrix:.begin ... end` 是深度遍历语句块;对嵌套数组会遍历到最深节点。 - `matrix:.= expression` 是深度遍历赋值;需要对不规则嵌套数组逐叶子节点写回时使用。 - 遍历时需要维度信息可用 `mIndexCount`;需要第 `n` 维下标可用 `mIndex(n)`。 - 子矩阵提取可以写 `matrix[row_start:row_end, col_start:col_end]`、`matrix[:, col_range]`、`matrix[row_index_array, col_index_array]`。 - 子矩阵赋值可以写成单个标量,也可以写成同结构矩阵;赋值矩阵应和目标子矩阵形状匹配。 - 子矩阵也可以接 `::=` 做逐单元赋值。 - `mfind(matrix)` 返回真值单元格下标;一维数组的无条件 `mfind` 返回一维下标数组。 - `mfind(matrix, condition)` 返回符合条件的下标行;二维矩阵下每行形如 `array(row_index, col_index)`。 - `mfind(matrix, condition, 1)` 在每个下标行末尾追加原单元值。 - `mfind(matrix, condition, ret_value, replacement)` 会把符合条件的单元格替换成 `replacement`,并返回替换前的匹配信息。 - `mfindSparse(matrix, condition)` 用于嵌套数组或稀疏结构,返回包含完整深层路径的下标行。 ## 可直接照写示例 ### 矩阵初始化 代码块身份:可直接照写示例 ```tsl zeros_1d := zeros(3); zeros_2d := zeros(2, 3); ones_1d := ones(3); nils_1d := nils(2); eye_2d := eye(3); rand_2d := rand(2, 3); named_zeros := zeros(2, array("A", "B")); writeLn(length(zeros_1d)); writeLn(mrows(zeros_2d)); writeLn(mcols(zeros_2d)); ``` 结果说明: - `zeros(3)` 的长度是 `3`,前三个元素依次是 `0`、`0`、`0` - `zeros(2, 3)` 的行数是 `2`、列数是 `3`,第一行前三个元素是 `0`、`0`、`0` - `ones(3)` 的前三个元素依次是 `1`、`1`、`1` - `nils(2)` 可直接生成长度为 `2` 的结果 - `eye(3)` 的行数是 `3`、列数是 `3`,并且 `(0,0)`、`(1,1)`、`(2,2)` 为 `1`,`(0,1)`、`(1,0)` 为 `0` - `rand(2, 3)` 的行数是 `2`、列数是 `3` - `zeros(2, array("A", "B"))` 的行数是 `2`、列数是 `2`,并且 `named_zeros[0]["A"]`、`named_zeros[0]["B"]`、`named_zeros[1]["A"]`、`named_zeros[1]["B"]` 都是 `0` 代码块身份:输出片段 ```text 3 2 3 ``` ### `->` 数列数组初始化 默认步长为 `1`: 代码块身份:可直接照写示例 ```tsl seq_default := 1 -> 5; ``` 结果说明: - `seq_default` 是 `array(1, 2, 3, 4, 5)` 显式指定步长: 代码块身份:可直接照写示例 ```tsl seq_step := array(2.5, 0.5) -> 5; ``` 结果说明: - `seq_step` 的长度是 `6` - 六个元素依次是 `2.5`、`3`、`3.5`、`4`、`4.5`、`5` 显式指定索引数组: 代码块身份:可直接照写示例 ```tsl seq_indexed := array(0, 1, array("A", "B", "C", "D", "E", "F")) -> 5; ``` 结果说明: - `seq_indexed` 的长度是 `6` - `seq_indexed["A"]` 到 `seq_indexed["F"]` 依次是 `0`、`1`、`2`、`3`、`4`、`5` ### 矩阵一元倒数 / 逆 / 广义逆:`!A` 方阵输入返回普通矩阵逆: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2), (3, 4)); inverse_value := !matrix_value; writeLn(mrows(inverse_value)); writeLn(mcols(inverse_value)); writeLn(inverse_value[0][0]); writeLn(inverse_value[0][1]); writeLn(inverse_value[1][0]); writeLn(inverse_value[1][1]); ``` 代码块身份:输出片段 ```text 2 2 -2 1 1.5 -0.5 ``` 非方阵输入返回广义逆: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2, 3), (4, 5, 6)); inverse_value := !matrix_value; writeLn(mrows(inverse_value)); writeLn(mcols(inverse_value)); writeLn(inverse_value[0][0]); writeLn(inverse_value[0][1]); writeLn(inverse_value[1][0]); writeLn(inverse_value[1][1]); writeLn(inverse_value[2][0]); writeLn(inverse_value[2][1]); ``` 代码块身份:输出片段 ```text 3 2 -0.944444444444444 0.444444444444444 -0.111111111111111 0.111111111111111 0.722222222222222 -0.222222222222222 ``` 说明: - `array((1, 2, 3), (4, 5, 6))` 是 `2 x 3` 矩阵样数组。 - `!matrix_value` 返回的是 `3 x 2` 广义逆结果。 - 生成矩阵逆/广义逆时写 `!matrix_value`;不要把它改写成 `1 / matrix_value`。 - `!` 不表示逻辑非;逻辑非回 [06_expressions_and_operators.md](06_expressions_and_operators.md) 使用 `not`。 ### `msize`、`mrows`、`mcols` 代码块身份:可直接照写示例 ```tsl matrix_rows := array( ("A": 1, "B": 2), ("A": 11, "B": 22), ("A": 21, "B": 32) ); size_info := msize(matrix_rows); size_index := msize(matrix_rows, 1); row_count := mrows(matrix_rows); row_index := mrows(matrix_rows, 1); col_count := mcols(matrix_rows); col_index := mcols(matrix_rows, 1); ``` 结果说明: - `msize(matrix_rows)` 返回 `array(3, 2)` - `msize(matrix_rows, 1)` 的第一项是 `array(0, 1, 2)`,第二项是 `array("A", "B")` - `mrows(matrix_rows)` 返回 `3` - `mrows(matrix_rows, 1)` 返回 `array(0, 1, 2)` - `mcols(matrix_rows)` 返回 `2` - `mcols(matrix_rows, 1)` 返回 `array("A", "B")` ### 矩阵遍历:`::` 与 `::=` `::` 执行语句块: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2, 3), (4, 5, 6)); sum_value := 0; matrix_value::begin sum_value += mcell; end writeLn(sum_value); ``` 代码块身份:输出片段 ```text 21 ``` `::=` 把表达式结果逐单元写回: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2), (3, 4)); matrix_value ::= mrow * 10 + mcol; writeLn(matrix_value[0][0]); writeLn(matrix_value[0][1]); writeLn(matrix_value[1][0]); writeLn(matrix_value[1][1]); ``` 代码块身份:输出片段 ```text 0 1 10 11 ``` 说明: - `mcell` 是当前单元格值。 - `mrow` 是当前行下标。 - `mcol` 是当前列下标。 - `::=` 右侧写表达式,不写 `begin ... end` 语句块。 ### 深度遍历:`:.` 与 `:.=` `:.` 会遍历到嵌套数组的最深节点: 代码块身份:可直接照写示例 ```tsl nested_value := array(10, 12, ("A": array(30), "B": 22), ("A": array(31, 32), "B": 23)); deep_count := 0; nested_value:.begin deep_count += 1; end writeLn(deep_count); ``` 代码块身份:输出片段 ```text 7 ``` `:.=` 会把表达式结果写回最深节点: 代码块身份:可直接照写示例 ```tsl values := array(-1, 2, -3); values:.= abs(mcell); writeLn(values[0]); writeLn(values[1]); writeLn(values[2]); ``` 代码块身份:输出片段 ```text 1 2 3 ``` ### 子矩阵 按行列范围提取: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2, 3), (4, 5, 6), (7, 8, 9)); sub_value := matrix_value[1:2, 0:1]; writeLn(mrows(sub_value)); writeLn(mcols(sub_value)); writeLn(sub_value[0][0]); writeLn(sub_value[0][1]); writeLn(sub_value[1][0]); writeLn(sub_value[1][1]); ``` 代码块身份:输出片段 ```text 2 2 4 5 7 8 ``` 按下标数组提取: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2, 3), (4, 5, 6), (7, 8, 9)); sub_value := matrix_value[array(0, 2), array(1, 2)]; writeLn(sub_value[0][0]); writeLn(sub_value[0][1]); writeLn(sub_value[1][0]); writeLn(sub_value[1][1]); ``` 代码块身份:输出片段 ```text 2 3 8 9 ``` 子矩阵赋值: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2, 3), (4, 5, 6), (7, 8, 9)); matrix_value[0:1, 1:2] := array((10, 11), (12, 13)); writeLn(matrix_value[0][1]); writeLn(matrix_value[0][2]); writeLn(matrix_value[1][1]); writeLn(matrix_value[1][2]); ``` 代码块身份:输出片段 ```text 10 11 12 13 ``` 子矩阵逐单元赋值: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 2), (3, 4)); matrix_value[0:1, 0:1] ::= mrow * 10 + mcol; writeLn(matrix_value[0][0]); writeLn(matrix_value[0][1]); writeLn(matrix_value[1][0]); writeLn(matrix_value[1][1]); ``` 代码块身份:输出片段 ```text 0 1 10 11 ``` ### `mfind` 与 `mfindSparse` 一维数组无条件查找会返回一维下标数组: 代码块身份:可直接照写示例 ```tsl values := array(1, 0, 2, 0, 3); indexes := mfind(values); writeLn(length(indexes)); writeLn(indexes[0]); writeLn(indexes[1]); writeLn(indexes[2]); ``` 代码块身份:输出片段 ```text 3 0 2 4 ``` 二维矩阵按条件查找: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 0), (2, 3)); indexes := mfind(matrix_value, mcell >= 2); writeLn(length(indexes)); writeLn(indexes[0][0]); writeLn(indexes[0][1]); writeLn(indexes[1][0]); writeLn(indexes[1][1]); ``` 代码块身份:输出片段 ```text 2 1 0 1 1 ``` 第三个参数写成 `1` 时,每行末尾追加原单元值: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 0), (2, 3)); matches := mfind(matrix_value, mcell >= 2, 1); writeLn(matches[0][0]); writeLn(matches[0][1]); writeLn(matches[0][2]); writeLn(matches[1][0]); writeLn(matches[1][1]); writeLn(matches[1][2]); ``` 代码块身份:输出片段 ```text 1 0 2 1 1 3 ``` 第四个参数用于替换符合条件的单元格: 代码块身份:可直接照写示例 ```tsl matrix_value := array((1, 0), (2, 3)); matches := mfind(matrix_value, mcell >= 2, 1, 100); writeLn(matches[0][2]); writeLn(matches[1][2]); writeLn(matrix_value[1][0]); writeLn(matrix_value[1][1]); ``` 代码块身份:输出片段 ```text 2 3 100 100 ``` `mfindSparse` 返回深层路径: 代码块身份:可直接照写示例 ```tsl nested_value := array(10, ("A": 0, "B": array(2)), 0); indexes := mfindSparse(nested_value, mcell = 2); writeLn(length(indexes)); writeLn(length(indexes[0])); writeLn(indexes[0][0]); writeLn(indexes[0][1]); writeLn(indexes[0][2]); ``` 代码块身份:输出片段 ```text 1 3 1 B 0 ``` ## 默认生成模板 需要矩阵构造时,优先从这个最短模板开始: 代码块身份:可直接照写示例 ```tsl matrix_value := zeros(2, 3); ``` ## 决策边界和禁止项 - 把 `eye(3)` 当成一维数组。 - 把 `!A` 当成逻辑非表达式。 - 以为 `mrows(matrix_value, 1)` 和 `mcols(matrix_value, 1)` 返回的还是数量。 - 写带步长的 `->` 时,漏掉外层 `array(...)`。 - 还在普通数组页里硬塞矩阵专用大小接口。 - 把 `::=` 写成带 `begin ... end` 的语句块。 - 用 `::` 期待遍历到任意深度;深度遍历使用 `:.`。 - 子矩阵赋值时用形状不匹配的矩阵硬塞。 代码块身份:反例 / 不可照写 ```text seq_value := 2.5, 0.5 -> 5; ``` 上面这种写法不对。显式步长模式需要写成 `array(2.5, 0.5) -> 5`。