playbook/docs/tsl/syntax/10_runtime_context_and_with.md

12 KiB

TSL 运行时上下文、服务与全局缓存

文档类型:语法主线 是否可直接用于生成代码:仅部分 是否含可直接照写示例:是 是否含不可照写反例:是 遇到不确定时:先按本页候选页继续判断;05_functions_and_calls.md06_expressions_and_operators.md14_ts_sql.md15_debug_and_profiler.md19_namespace_libpath_and_unit_runtime.md21_builtin_runtime_objects.md../reference/catalog/datawarehouse.md;仍不命中时回到语法路由中心 index.md;如果问题已经超出语法层,回到 TSL 总入口 ../index.md

这一篇只处理运行时环境参数、块环境 with 语句、with 后缀、# 网格调用、timeout 后缀、dupvalue(...) 和全局缓存,不处理任何金融业务语义。

本篇职责

回答“setSysParam / getSysParam 怎样用、sysParams[...] 是什么、块环境 with *, values do / with **, values do 怎样写、#Func() with array(...) 这种后缀环境调用怎样写,网格调用怎样取回结果,以及全局缓存最小读写规则是什么”。

智能体运行时上下文判断流程

  1. 先判断要操作系统参数、运行时上下文对象、块环境 with 语句、with 后缀调用、# 网格调用、timeout 后缀,还是全局缓存函数。
  2. 系统参数优先用本页明确的 setSysParam / getSysParam / sysParams[...] 形态。
  3. #Func() with array(...) 只作为运行时环境调用写法,不要套到普通本地函数。
  4. 网格调用返回的不是最终值;需要最终结果时继续写 dupvalue(...)
  5. 全局缓存读写要成对出现,并明确键和值的生命周期。
  6. 普通函数调用回到 05_functions_and_calls.md,不要把运行时服务写成普通语法糖;缓存值参与 select 时,查询语法仍回到 14_ts_sql.md
  7. 如果问题是内置运行时对象本身,转到 21_builtin_runtime_objects.md
  8. 本地函数后缀 with 属于反例时不要照写。
  9. 没有对应代码块时不要发明运行时上下文/运行时服务/全局缓存写法。

核心规则

  • TSL 有一组运行时系统参数;本页只写通用语法形态。
  • setSysParam(key, value)getSysParam(key) 可以直接用字符串键。
  • sysParams[key] 可以直接读写这些运行时参数。
  • 块环境语句可写成 with *, sys_param_values do begin ... endwith **, sys_param_values do begin ... end
  • with * 会把提供的系统参数合并进当前运行时上下文;不要依赖它在块结束后自动恢复外层值。
  • with ** 会用提供的系统参数建立隔离块环境;块结束后恢复外层系统参数。
  • 后缀 with 形式写在函数文件调用后面:#Func() with array(...)
  • with array(...) 只在该次调用里临时覆盖对应键,调用结束后会恢复外部原值。
  • 如果外部原值本来不存在,with array(...) 调用结束后,对应键会恢复成 nil
  • 不要把上面的后缀 with 直接泛化成“任何本地函数调用后面都能接 with array(...)”;本地函数后缀 with 属于反例。
  • 本页只收通用键的例子;像 pn_stock()pn_date() 这类金融上下文参数,统一到 ../reference/catalog/datawarehouse.md 查函数事实。
  • 网格调用的最小写法是 r := #Func(args);
  • 网格调用返回的不是最终值;用 dupvalue(r) 取回结果。
  • timeout N 后缀可直接接在网格调用后面。
  • 全局缓存函数的参数规格见 ../reference/catalog/system.md;本页只保留缓存引用的运行时行为示例。
  • 写入和读取全局缓存成功时返回 1
  • 从全局缓存取出的值,ifCache(v) 会返回 1
  • 一旦对取出的缓存值做本地写入,它会立刻实例化;写入后 ifCache(v) 返回 0
  • checkGlobalCacheExpired(v) 对同名缓存的最新引用返回 0;同名缓存被重置后,旧引用会变成过期状态并返回 1
  • 全局缓存取出的值可以直接参与 select

可直接照写示例

直接设置和读取系统参数:

代码块身份:可直接照写示例

setSysParam("a", 123);
setSysParam("b", "XYZ");
writeLn(getSysParam("a"));
writeLn(getSysParam("b"));

结果说明:

  • 依次输出 123XYZ

代码块身份:输出片段

123
XYZ

sysParams[...] 直接读写:

代码块身份:可直接照写示例

sysParams["a"] := 321;
sysParams["b"] := "QQ";
writeLn(sysParams["a"]);
writeLn(sysParams["b"]);
writeLn(getSysParam("a"));

结果说明:

  • 依次输出 321QQ321
  • 说明 sysParams[...]getSysParam(...) / setSysParam(...) 指向的是同一组运行时环境参数

块环境 with *

代码块身份:可直接照写示例

setSysParam("a", 1);
sys_param_values := array("a": 2, "b": 3);

with *, sys_param_values do
begin
    writeLn(getSysParam("a"));
    writeLn(getSysParam("b"));
end

writeLn(getSysParam("a"));
writeLn(getSysParam("b"));

结果说明:

  • 块内输出 23
  • 块后输出 23
  • 说明 with * 会把传入键合并进当前系统参数上下文;不要把它当成自动恢复外层值的隔离块

块环境 with **

代码块身份:可直接照写示例

setSysParam("a", 1);
setSysParam("b", 9);
sys_param_values := array("b": 4);

with **, sys_param_values do
begin
    writeLn(getSysParam("a") = nil);
    writeLn(getSysParam("b"));
end

writeLn(getSysParam("a"));
writeLn(getSysParam("b"));

结果说明:

  • 块内输出 14
  • 块后输出 19
  • 说明 with ** 不继承未传入的外层系统参数,并且块结束后恢复外层系统参数

沿用同一个 TestDo.tsf,看 with array(...) 的覆盖边界:

代码块身份:配置片段 / 概念骨架 代码块说明:多文件结构骨架;依赖函数文件查找路径,不是可直接复制的单文件最小示例。

// TestDo.tsf
function TestDo();
begin
    return array(getSysParam("a"), getSysParam("b"));
end;

// main.tsl
setSysParam("a", 7);
setSysParam("b", 8);
r := #TestDo() with array("a": 101, "b": 202);
writeLn(r[0]);
writeLn(r[1]);
writeLn(getSysParam("a"));
writeLn(getSysParam("b"));

结果说明:

  • r[0] 输出 101
  • r[1] 输出 202
  • 调用结束后,外层 getSysParam("a") 输出 7
  • 调用结束后,外层 getSysParam("b") 输出 8
  • 说明 with array(...) 是“只在该次调用里临时覆盖,再恢复外部原值”

沿用上一个 TestDo.tsf,只把 main.tsl 改成下面这样:

代码块身份:配置片段 / 概念骨架 代码块说明:多文件结构骨架;依赖函数文件查找路径,不是可直接复制的单文件最小示例。

// main.tsl

r := #TestDo() with array("a": 101, "b": 202);
writeLn(r[0]);
writeLn(r[1]);
writeLn(getSysParam("a") = nil);
writeLn(getSysParam("b") = nil);

结果说明:

  • 依次输出 10120211
  • 说明外层原值不存在时,调用后不是“泄露”成 101 / 202,而是恢复成 nil

# 网格调用与 dupvalue

代码块身份:可直接照写示例

writeLn(dupvalue(#AddOne(5)));

function AddOne(v);
begin
    return v + 1;
end;

结果说明:

  • #AddOne(5) 可以执行
  • dupvalue(r) 返回最终结果 6

代码块身份:输出片段

6

网格调用的 timeout

代码块身份:可直接照写示例

writeLn(dupvalue(#AddOne(5) timeout 3000));

function AddOne(v);
begin
    return v + 1;
end;

结果说明:

  • timeout 3000 这种后缀写法可以通过并正常执行
  • 上例输出 6

setGlobalCachegetGlobalCacheifCache

代码块身份:可直接照写示例

v1 := array(1, 2, 3);
writeLn(setGlobalCache("PB_TEST_GC_BASIC", v1));
writeLn(getGlobalCache("PB_TEST_GC_BASIC", v2));
writeLn(ifCache(v2));
writeLn(length(v2));
writeLn(v2[0], ',', v2[1], ',', v2[2]);

结果说明:

  • setGlobalCache("PB_TEST_GC_BASIC", v1) 返回 1
  • getGlobalCache("PB_TEST_GC_BASIC", v2) 返回 1
  • 取出的 v2ifCache(v2) 返回 1
  • v2 长度是 3,内容是 1,2,3

checkGlobalCacheExpired

代码块身份:可直接照写示例

setGlobalCache("PB_TEST_GC_EXPIRE", array(1, 2, 3));
getGlobalCache("PB_TEST_GC_EXPIRE", v);
writeLn(checkGlobalCacheExpired(v));
setGlobalCache("PB_TEST_GC_EXPIRE", array(1, 2, 3, 4));
writeLn(checkGlobalCacheExpired(v));

结果说明:

  • 刚取出的缓存引用上,checkGlobalCacheExpired(v) 返回 0
  • 同名缓存被重新设置后,旧引用上的 checkGlobalCacheExpired(v) 返回 1

写入后会实例化

代码块身份:可直接照写示例

setGlobalCache("PB_TEST_GC_DETACH", array(1, 2, 3));
getGlobalCache("PB_TEST_GC_DETACH", v);
writeLn(ifCache(v));
v[0] := 100;
writeLn(ifCache(v));
writeLn(v[0], ',', v[1], ',', v[2]);

结果说明:

  • 刚取出时 ifCache(v) 返回 1
  • v[0] 赋值后,ifCache(v) 立即返回 0
  • 写入后的本地值内容是 100,2,3

全局缓存参与 select

代码块身份:可直接照写示例

src := array((1, 2), (3, 4), (2, 1));
setGlobalCache("PB_TEST_GC_SELECT", src);
getGlobalCache("PB_TEST_GC_SELECT", v);
q := select * from v order by [0] desc end;
writeLn(dataType(q));
writeLn(mrows(q));
writeLn(q[0][0], ',', q[0][1], ';', q[1][0], ',', q[1][1], ';', q[2][0], ',', q[2][1]);

结果说明:

  • 对缓存值做 select 可以正常执行
  • 返回结果的 dataType5
  • 行数是 3
  • 排序后内容依次是 (3,4)(2,1)(1,2)

本页不生成的范围

  • 网格超时触发错误时的完整边界
  • getGlobalCacheInfolistGlobalCachelistGlobalCacheRemoved
  • 初始化 TSL、监控线程、回收策略与兼容旧系统方案
  • with [S => ..., T => ...] do 块语句
  • {$include ...} 包含文件指令

这些名称只作为边界提示,不作为本页可生成模板。

默认生成模板

最小运行时环境入口的默认骨架如下:

代码块身份:可直接照写示例

setSysParam("a", 1);
writeLn(getSysParam("a"));

需要网格调用结果时,优先从下面这个模板起步:

代码块身份:可直接照写示例

writeLn(dupvalue(#AddOne(5)));

function AddOne(v);
begin
    return v + 1;
end;

需要缓存时,从 setGlobalCache / getGlobalCache 那一节选择模板。

禁止项

  • 把系统参数页直接写成金融函数页。
  • #Func() with array(...) 误判成也能直接套在本地函数 Demo() 后面。
  • with * 误判成会自动恢复外层系统参数。
  • 以为 with array(...) 改的是全局永久值,不会恢复外层原环境。
  • 把网格句柄直接当最终值用,而不做 dupvalue(...)
  • 以为从全局缓存取出的值,本地写入后仍然保持缓存身份。
  • 以为旧缓存引用在同名缓存被重置后还会继续视为“未过期”。
  • r := #Func(args); 保存的是网格调用句柄;需要最终结果时继续写 dupvalue(r)

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

r := Demo() with array("a": 11);

function Demo();
begin
    return getSysParam("a");
end;

上面这种“本地函数后缀 with”写法不作为可写事实;常见报错是 Statement missing terminator,随后出现 function compile error