第二十七章 Caché 命令大全 TRY 命令
标识要在执行期间监视错误的代码块。
重点
- 不能使用带参数的
QUIT
退出TRY
块;尝试这样做会导致编译错误。要完全从TRY
块中退出例程,请发出RETURN
语句。 - 不能在
TRY
块中设置$ ZTRAP
或$ ETRAP
。 - 如果在进入
TRY
块之前设置了$ ZTRAP
,并且TRY
块内发生异常,则Caché将使用CATCH
块而不是$ ZTRAP
。
大纲
TRY {
. . .
}
描述
Try
命令不带任何参数。它用于标识用大括号括起来的CachéObjectScript代码语句块。此代码块是用于结构化异常处理的受保护代码。如果此代码块内发生异常,则Caché设置$ZERROR
和$ECODE
,然后将执行转移到由Catch
命令标识的异常处理程序。
异常可能由于运行时错误(如试图除以0)而发生,也可能通过发出Throw
命令显式传播。如果没有发生错误,则继续执行Catch
代码块之后的下一条ObjectScript语句。
TRY
块后面必须紧跟CATCH
块。不能在try
代码块的右花括号和catch
命令之间指定可执行代码语句或标签。但是,可以在TRY
和块及其CATCH
块之间指定注释。每个TRY
块只允许一个CATCH
块。但是,可以嵌套成对的TRY / CATCH
块,例如:
TRY {
/* TRY code */
TRY {
/* nested TRY code */
}
CATCH {
/* nested CATCH code */
}
}
CATCH {
/* CATCH code */
}
通常,一个CachéObjectScript程序由多个TRY
块组成,每个TRY
块紧随其后的是与其相关的CATCH
块。
QUIT 与 RETURN
可以使用QUIT
或RETURN
退出TRY
块。 QUIT
退出当前块结构,并使用该块结构之外的下一个命令继续执行。例如,如果在嵌套的TRY
块中,则发出QUIT
可使该TRY
块退出到封闭的块结构中。在TRY
块中发出QUIT
命令会将执行转移到相应CATCH
块之后的第一条代码行。不能使用带参数的QUIT
退出TRY
块;尝试这样做会导致编译错误。要完全从TRY
块中退出例程,请发出RETURN
语句。
在极少数情况下,TRY
块QUIT
或RETURN
命令可能会生成异常。如果TRY
创建了新上下文,然后删除了旧上下文的某些方面,则可能会发生这种情况;尝试还原到旧上下文将导致异常。TRY
块QUIT
或RETURN
异常不会调用CATCH
块异常处理程序。
ETRAP
TRY
和CATCH
命令在执行级别内执行错误处理。当TRY
块中发生异常时,Caché通常会执行紧随TRY
块之后的异常处理程序代码的CATCH
块。这是首选的错误处理行为。
不能在TRY
块中设置$ ZTRAP
或$ ETRAP
。但是,可以在TRY
块内的无参数DO
块内设置$ ZTRAP
或$ ETRAP
。这是因为DO
块的行处于其自己的执行级别,因此可以具有自己的错误处理程序。
如果在进入TRY
块之前设置了$ ZTRAP
,并且TRY
块内发生异常,则Caché将使用CATCH
块而不是$ ZTRAP
。
如果在进入TRY
块之前设置了$ ETRAP
,并且在TRY
块内发生了异常,则Caché可能会采用$ ETRAP
而不是CATCH
,除非阻止了这种可能性。如果发生异常时$ ETRAP
和CATCH
同时存在,则Caché将执行适用于当前执行级别的错误代码(CATCH
或$ ETRAP
)。因为$ ETRAP
本质上不与执行级别相关联,所以除非另外指定,否则Caché假定它与当前执行级别相关联。在设置$ ETRAP
之前,必须先新建$ ETRAP
才能为$ ETRAP
建立级别标记,以便Caché可以正确地将CATCH
用作当前级别的异常处理程序,而不是$ ETRAP
。否则,系统错误(包括THROW
命令引发的系统错误)可能会占用$ ETRAP
异常处理程序。
GOTO 与 DO
可以使用GOTO
或DO
命令在TRY
块内的标签处输入TRY
块。如果稍后在TRY
块中发生异常,则将采用CATCH
块异常处理程序,就像在TRY
关键字中输入TRY
块一样。但是,为使编码清晰起见,应避免使用GOTO
或DO
进入TRY
块。
当然,可以从TRY
块或CATCH
块中发出GOTO
。
强烈建议不要使用GOTO
或DO
输入CATCH
块。
在TRY区块内执行DO
当使用TRY
语句时,THROW
导致试图查找适当的CATCH
块的帧堆栈搜索。当帧堆栈指示在TRY
块内执行时,将在相应的CATCH
块处继续执行。但是,Caché必须在执行CATCH
块之前删除当前TRY
块内的所有“本地”调用。
如果TRY
块包含导致重新输入该TRY
块的DO
语句,则可能会发生以下两种情况之一:
“本地” DO
调用(保留在当前TRY
块内的DO
调用):如果先前的帧堆栈条目是位于同一TRY
块中的DO
调用,则该DO
被假定为当前内部的“本地”子例程调用尝试块。在这种情况下,不会立即输入CATCH
,而是弹出框架堆栈(可能会删除一些最近分配的NEW
变量),并在当前TRY
块中的DO
调用处继续搜索。如果新的先前帧堆栈条目不是当前TRY
块内部的DO
,则输入相应的CATCH
块。但是,如果先前的帧堆栈条目是同一TRY
中的另一个DO
,则再次弹出该帧堆栈(以及最近分配的NEW
变量)。继续此操作,直到上一个帧堆栈条目不是DO
,然后才进入CATCH
块。
“递归” DO
调用(在TRY
块内的DO
调用离开TRY
块,但稍后执行重新进入该TRY
块):搜索CATCH
块时,如果先前的帧堆栈条目是当前TRY
块内的DO
,如果该先前堆栈帧的目标标签不在当前TRY
块(包括任何嵌套的TRY
块)内,则不会弹出该帧堆栈(并且不会弹出最近分配的局部变量),并且立即输入CATCH
块。请注意,如果该CATCH
块又进行了一次THROW
,则可能会重新输入当前的CATCH
块,因为递归DO
帧仍在帧堆栈上。
示例
本节中的示例显示运行时错误(%Exception.SystemException
错误)。 DO
框架仍在框架堆栈上。
在以下示例中,将执行TRY
代码块。它尝试设置局部变量a
。在第一个示例中,代码成功完成,并且跳过了CATCH
。在第二个示例中,代码失败并显示错误,并且执行被传递给CATCH
异常处理程序。
TRY
成功。 CATCH
块被跳过。执行从第二个TRY
块继续:
/// d ##class(PHA.TEST.Command).TestTry()
ClassMethod TestTry()
{
TRY {
WRITE "第一个TRY块",!
SET x="fred"
WRITE "x是定义的变量",!
SET a=x
}
CATCH exp
{
WRITE !,"这是CATCH异常处理程序",!
IF 1=exp.%IsA("%Exception.SystemException") {
WRITE "系统异常",!
WRITE "Name: ",$ZCVT(exp.Name,"O","HTML"),!
WRITE "Location: ",exp.Location,!
WRITE "Code: ",exp.Code,!
WRITE "Data: ",exp.Data,!!
} ELSE {
WRITE "不是系统异常",!!
}
WRITE "$ZERROR: ",$ZERROR,!
WRITE "$ECODE: ",$ECODE
RETURN
}
TRY {
WRITE !,"第二个TRY块",!
WRITE "这是代码失败的地方",!
WRITE "$ZERROR: ",$ZERROR,!
WRITE "$ECODE: ",$ECODE
}
CATCH exp2 {
WRITE !,"这是第二个CATCH异常处理程序",!
}
}
DHC-APP>d ##class(PHA.TEST.Command).TestTry()
第一个TRY块
x是定义的变量
第二个TRY块
这是代码失败的地方
$ZERROR:
$ECODE:
TRY
失败。执行将继续执行CATCH
块。 CATCH
块以RETURN
结尾,因此不执行第二个TRY
块:
/// d ##class(PHA.TEST.Command).TestTry1()
ClassMethod TestTry1()
{
TRY {
WRITE "第一个TRY块",!
KILL x
WRITE "x是一个未定义的变量",!
SET a=x
}
CATCH exp
{
WRITE !,"这是CATCH异常处理程序",!
IF 1=exp.%IsA("%Exception.SystemException") {
WRITE "系统异常",!
WRITE "Name: ",$ZCVT(exp.Name,"O","HTML"),!
WRITE "Location: ",exp.Location,!
WRITE "Code: ",exp.Code,!
WRITE "Data: ",exp.Data,!!
} ELSE {
WRITE "不是系统异常",!!
}
WRITE "$ZERROR: ",$ZERROR,!
WRITE "$ECODE: ",$ECODE
RETURN
}
TRY {
WRITE !,"第二个TRY块",!
WRITE "这是代码失败的地方",!
WRITE "$ZERROR: ",$ZERROR,!
WRITE "$ECODE: ",$ECODE
}
CATCH exp2 {
WRITE !,"这是第二个CATCH异常处理程序",!
}
}
DHC-APP>d ##class(PHA.TEST.Command).TestTry1()
第一个TRY块
x是一个未定义的变量
这是CATCH异常处理程序
系统异常
Name: <UNDEFINED>
Location: zTestTry1+5^PHA.TEST.Command.1
Code: 9
Data: x
$ZERROR: zTestTry1+5^PHA.TEST.Command.1 *x
$ECODE: ,M7,M6,
TRY
退出。在以下示例中,由于TRY
块的执行以QUIT
或RETURN
(不是错误)结束,因此未执行CATCH
块。如果RETURN
,则程序执行停止。如果为QUIT
,程序将继续执行第二个TRY
块:
/// d ##class(PHA.TEST.Command).TestTry2()
ClassMethod TestTry2()
{
TRY {
WRITE "第一个TRY块",!
KILL x
WRITE "x是一个未定义的变量",!
SET decide=$RANDOM(2)
IF decide=0 {
WRITE "发出 QUIT",!
QUIT
}
IF decide=1 {
WRITE "发出 RETURN",!
RETURN
}
WRITE "这不应该显示",!
SET a=x
}
CATCH exp {
WRITE !,"这是CATCH异常处理程序",!
IF 1=exp.%IsA("%Exception.SystemException") {
WRITE "System exception",!
WRITE "Name: ",$ZCVT(exp.Name,"O","HTML"),!
WRITE "Location: ",exp.Location,!
WRITE "Code: ",exp.Code,!
WRITE "Data: ",exp.Data,!!
} ELSE {
WRITE "不是系统异常",!!
}
WRITE "$ZERROR: ",$ZERROR,!
WRITE "$ECODE: ",$ECODE
RETURN
}
TRY {
WRITE !,"第二个TRY块",!
WRITE "这是代码失败的地方",!
WRITE "$ZERROR: ",$ZERROR,!
WRITE "$ECODE: ",$ECODE
}
CATCH exp2 {
WRITE !,"这是第二个CATCH异常处理程序",!
}
}
DHC-APP>d ##class(PHA.TEST.Command).TestTry2()
第一个TRY块
x是一个未定义的变量
发出 QUIT
第二个TRY块
这是代码失败的地方
$ZERROR: zTestTry1+5^PHA.TEST.Command.1 *x
$ECODE: ,M7,M6,
DHC-APP>d ##class(PHA.TEST.Command).TestTry2()
第一个TRY块
x是一个未定义的变量
发出 RETURN