NSIS 使用 LogicLib.nsh 实现基本流程控制

2024-03-20 1324点热度 0人点赞 1条评论

程序的三种基本结构包括:顺序结构、分支结构、循环结构。顺序结构是最基本的结构,本文主要说明了如何在 NSIS 脚本中实现分支结构和循环结构。

最基本的分支结构和循环结构可以用 StrCmp 命令和 Goto 命令组成,但通过这种方式写出的代码可读性极差且难以调试。此时我们就需要用到头文件 LogicLib.nsh 中的功能了。

下面是一个示例程序,在 Debug 区段中插入我们要调试的代码:

Name "NSIS LogicLib Debug"
OutFile "LogicLib.exe"
ShowInstDetails Show
RequestExecutionLevel user

!include LogicLib.nsh

Section Debug
; --- 在这里插入要调试的代码 --- 
SectionEnd

一、If 分支语句

例1.1:If - ElseIf - Else - EndIf

标准的 If 语句分支结构

var /GLOBAL test1
StrCpy $test1 'a'
${If} $test1 == 'a'
  DetailPrint '$$test1 的值为 a'
${ElseIf} $test1 == 'b'
  DetailPrint '$$test1 的值为 b'
${Else}
  DetailPrint '$$test1 的值为 $test1'
${EndIf}

运行结果:$test1 的值为 a

例1.2:IfNot - Else - EndIf

IfNot 判断其后的表达式运算结果是否为假,与 If 的判断规则相反

var /GLOBAL test2
StrCpy $test2 'z'
${IfNot} $test2 == 'a'
  DetailPrint '$$test2 的值不为 a'
${Else}
  DetailPrint '$$test2 的值为 $test2'
${EndIf}

运行结果:$test2 的值不为 a

例1.3:If - ElseIf - ElseIfNot - EndIf

ElseIfNot 判断其后的表达式运算结果是否为假,与 ElseIf 的判断规则相反

var /GLOBAL test3
StrCpy $test3 'b'
${If} $test3 == 'a'
  DetailPrint '$$test3 的值为 a'
${ElseIfNot} $test3 == 'b'
  DetailPrint '$$test3 的值不为 b'
${Else}
  DetailPrint '$$test3 的值为 $test3'
${EndIf}

运行结果:$test3 的值为 b

例1.4:IfNot & Unless

Unless 与 IfNot 是等价的

var /GLOBAL test4
StrCpy $test4 'z'
${Unless} $test4 == 'a'
  DetailPrint '$$test4 的值不为 a'
${Else}
  DetailPrint '$$test4 的值为 $test4'
${EndUnless}

运行结果:$test4 的值不为 a

例1.5:ElseIfNot & ElseUnless

ElseUnless 与 ElseIfNot 是等价的

var /GLOBAL test5
StrCpy $test5 'b'
${If} $test5 == 'a'
  DetailPrint '$$test5 的值为 a'
${ElseUnless} $test5 == 'b'
  DetailPrint '$$test5 的值不为 b'
${Else}
  DetailPrint '$$test5 的值为 $test5'
${EndIf}

运行结果:$test5 的值为 b

二、If 语句与 And 和 Or 组合

例2.1:AndIf & AndIfNot & AndUnless

AndIf 的逻辑,是如果前面的 If 表达式结果为真,且当前的表达式结果也为真,才继续执行它下辖代码块中的语句。

AndIfNot 的逻辑,是如果前面的 If 表达式结果为真,且当前的表达式结果为假,才继续执行它下辖代码块中的语句。

AndUnless 功能同 AndIfNot。

var /GLOBAL test6
StrCpy $test6 'a'
${If} $test6 == 'a'
  DetailPrint '$$test6 的值为 a'
${AndIf} $test6 != 'b'
  DetailPrint '$$test6 的值不为 b'
${AndIfNot} $test6 == 'b'
  DetailPrint '$$test6 的值不为 b'
${AndUnless} $test6 == 'b'
  DetailPrint '$$test6 的值不为 b'
${EndIf}

运行结果:

$test6 的值为 a
$test6 的值不为 b
$test6 的值不为 b
$test6 的值不为 b

例2.2:OrIf & OrIfNot & OrUnless

AndIf 的逻辑,是如果前面的 If 表达式结果为假,且当前的表达式结果为真,才继续执行它下辖代码块中的语句。如果前面 If 表达式结果为真,也不会进入到它下辖的代码块中。

AndIfNot 的逻辑,是如果前面的 If 表达式结果为假,且当前的表达式结果也为假,才继续执行它下辖代码块中的语句。如果前面 If 表达式结果为真,也不会进入到它下辖的代码块中。

OrUnless 功能同 OrIfNot。

var /GLOBAL test7
StrCpy $test7 'z'
${If} $test7 == 'a'
  DetailPrint '$$test7 的值为 a'
${OrIf} $test7 == 'b'
  DetailPrint '$$test7 的值为 b'
${OrIfNot} $test7 != 'b'
  DetailPrint '$$test7 的值为 b'
${OrUnless} $test7 != 'b'
  DetailPrint '$$test7 的值为 b'
${Else}
  DetailPrint '$$test7 的值为 $test7'
${EndIf}

运行结果:$test7 的值为 z

三、IfThen 与 IfNotThen

例3.1:IfThen

IfThen 语句,判断其后的表达式结果是否为真,执行第三个参数中指定的命令

var /GLOBAL test8
StrCpy $test8 'a'
${IfThen} $test8 == 'a' ${|} Goto x1 ${|}
x1:
DetailPrint 'x1'
Goto endOfxy1
y1:
DetailPrint 'y1'
Goto endOfxy1
endOfxy1:
DetailPrint 'endOfxy1'

运行结果:

x1
endOfxy1

例3.2:IfNotThen

IfNotThen 的逻辑与 IfThen 相反,判断其后的表达式结果是否为假,执行第三个参数中指定的命令

var /GLOBAL test9
StrCpy $test9 'b'
${IfNotThen} $test9 == 'a' ${|} Goto y2 ${|}
x2:
DetailPrint 'x2'
Goto endOfxy2
y2:
DetailPrint 'y2'
Goto endOfxy2
endOfxy2:
DetailPrint 'endOfxy2'

运行结果:

y2
endOfxy2

四、IfCmd

例4.1:IfCmd

判断其后执行命令(如MessageBox)的结果为指定结果时,执行指定的命令。

如下面这段代码中,如果在弹出的 MessageBox 中点击了“Yes”,则会跳转到 y3

${IfCmd} MessageBox MB_YESNO "MY_YESNO" /SD IDYES IDYES ${||} Goto y3 ${|}
x3:
DetailPrint 'x3'
Goto endOfxy3
y3:
DetailPrint 'y3'
Goto endOfxy3
endOfxy3:
DetailPrint 'endOfxy3'

点击“是”时运行结果:

y3
endOfxy3

五、Select 多重分支语句

例5.1:Select - Case - CaseElse - EndSelect

Select 分支语句无需在每一个分支的最后添加 Break 标记,每个分支执行完毕后自动跳至 EndSelect

var /GLOBAL test10
StrCpy $test10 'b'
${Select} $test10
  ${Case} 'a'
    DetailPrint '$$test10 的值为 a'
  ${Case} 'b'
    DetailPrint '$$test10 的值为 b'
  ${Case} 'c'
    DetailPrint '$$test10 的值为 c'
  ${CaseElse}
    DetailPrint '$$test10 的值为 $test10'
${EndSelect}

运行结果:$test10 的值为 b

例5.2:Select - Case - Default - EndSelect

Default 的功能与 CaseElse 是一样的

var /GLOBAL test11
StrCpy $test11 'd'
${Select} $test11
  ${Case} 'a'
    DetailPrint '$$test11 的值为 a'
  ${Case} 'b'
    DetailPrint '$$test11 的值为 b'
  ${Case} 'c'
    DetailPrint '$$test11 的值为 c'
  ${Default}
    DetailPrint '$$test11 的值为 $test11'
${EndSelect}

运行结果:$test11 的值为 d

六、Switch 多重分支语句

例6.1:Switch - Case - CaseElse - EndSwitch

Switch 分支语句类似C语言的 switch 语句,如果不在一个 Case 的末尾添加 Break 标记,程序会一直向下执行其他 Case 中的部分。

var /GLOBAL test12
StrCpy $test12 'b'
${Switch} $test12
  ${Case} 'a'
    DetailPrint '$$test12 的值为 a'
    ${Break}
  ${Case} 'b'
    DetailPrint '$$test12 的值为 b'
    ${Break}
  ${Case} 'c'
    DetailPrint '$$test12 的值为 c'
    ${Break}
  ${CaseElse}
    DetailPrint '$$test12 的值为 $test12'
${EndSwitch}

运行结果:$test12 的值为 b

例6.2:Switch - Case - Default - EndSwitch

Default 的功能与 CaseElse 是一样的

var /GLOBAL test13
StrCpy $test13 'b'
${Switch} $test13
  ${Case} 'a'
    DetailPrint '$$test13 的值为 a'
    ${Break}
  ${Case} 'b'
    DetailPrint '$$test13 的值为 b'
    ${Break}
  ${Case} 'c'
    DetailPrint '$$test13 的值为 c'
    ${Break}
  ${Default}
    DetailPrint '$$test13 的值为 $test13'
${EndSwitch}

运行结果:$test13 的值为 b

例6.3:Switch - Case - CaseElse - EndSwitch WithOut Break

如果不在一个 Case 的末尾添加 Break 标记,程序会一直向下执行其他 Case 中的部分。

var /GLOBAL test14
StrCpy $test14 'a'
${Switch} $test14
  ${Case} 'a'
    DetailPrint '$$test14 的值为 a'
  ${Case} 'b'
    DetailPrint '$$test14 的值为 b'
  ${Case} 'c'
    DetailPrint '$$test14 的值为 c'
  ${Default}
    DetailPrint '$$test14 的值为 $test14'
${EndSwitch}

运行结果:

$test14 的值为 a
$test14 的值为 b
$test14 的值为 c
$test14 的值为 a

七、While、Do、DoWhile、DoUntil 循环

例7.1:While - EndWhile 循环

While 循环,只要后面的表达式为真,就一直循环下去

StrCpy $R1 0
${While} $R1 < 5
  IntOp $R1 $R1 + 1
  DetailPrint $R1
${EndWhile}

运行结果:

1
2
3
4
5

例7.2:DoWhile - Loop 循环

DoWhile 循环,用法同While

StrCpy $R1 0
${DoWhile} $R1 < 5
  IntOp $R1 $R1 + 1
  DetailPrint $R1
${Loop}

运行结果:

1
2
3
4
5

例7.3:DoUntil - Loop 循环

DoUntil 循环,只要后面的表达式值为假,就一直循环下去

StrCpy $R1 0
${DoUntil} $R1 >= 5
  IntOp $R1 $R1 + 1
  DetailPrint $R1
${Loop}

运行结果:

1
2
3
4
5

例7.4:Do - LoopWhile 循环

Do 循环,先执行指定代码,再判断如果 LoopWhile 后面的表达式为真,就一直循环该段代码

StrCpy $R1 0
${Do} 
  IntOp $R1 $R1 + 1
  DetailPrint $R1
${LoopWhile} $R1 < 5

运行结果:

1
2
3
4
5

例7.5:Do - LoopUntil 循环

Do 循环,先执行指定代码,再判断如果 LoopUntil 后面的表达式为假,就一直循环该段代码

StrCpy $R1 0
${Do}
  IntOp $R1 $R1 + 1
  DetailPrint $R1
${LoopUntil} $R1 >= 5

运行结果:

1
2
3
4
5

例7.6:Break & Continue

Break 和 Continue 可用于退出所有类型的循环

StrCpy $R1 0
${While} $R1 < 5
  IntOp $R1 $R1 + 1
  ${If} $R1 == 2
    ${Continue}
  ${ElseIf} $R1 == 4
    ${Break}
  ${EndIf}
  DetailPrint $R1
${EndWhile}

运行结果:

1
3

例7.7:ExitDo

ExitDo 可用于退出 Do - LoopWhile、Do - LoopUntil、DoWhile、DoUntil 四类循环

StrCpy $R1 0
${Do}
  IntOp $R1 $R1 + 1
  ${If} $R1 == 4
    ${ExitDo}
  ${EndIf}
  DetailPrint $R1
${LoopWhile} $R1 < 5

运行结果:

1
2
3

例7.8:ExitWhile

ExitWhile 只能用于退出 While 循环

StrCpy $R1 0
${While} $R1 < 5
  IntOp $R1 $R1 + 1
  ${If} $R1 == 4
    ${ExitWhile}
  ${EndIf}
  DetailPrint $R1
${EndWhile}

运行结果:

1
2
3

八、For 与 ForEach 循环

例8.1:For - Next 循环

For 循环的第一个参数为循环变量,第二个参数为该遍历进入循环的初始值,当该变量的值在执行循环的过程中与第三个参数相等时,循环退出。

${For} $R1 1 5
  DetailPrint $R1
${Next}

运行结果:

1
2
3
4
5

例8.2:ForEach 循环

ForEach 循环与 For 循环的不同之处在于,ForEach 循环还有第四、五个参数,用于指定循环的步长,第四个参数用于指定步长的正负(+、-),第五个参数用于指定步长的绝对值。

${ForEach} $R1 1 5 + 1
  DetailPrint $R1
${Next}
${ForEach} $R1 10 2 - 2
  DetailPrint $R1
${Next}

运行结果:

1
2
3
4
5
10
8
6
4
2

例8.3:Break & Continue

Break 和 Continue 可用于退出所有类型的循环

${For} $R1 1 5
  ${If} $R1 == 2
    ${Continue}
  ${ElseIf} $R1 == 4
    ${Break}
  ${EndIf}
  DetailPrint $R1
${Next}

运行结果:

1
3

例8.4:ExitFor

ExitFor 可用于退出 For 循环与 ForEach 循环

${For} $R1 1 5
  ${If} $R1 == 4
    ${ExitFor}
  ${EndIf}
  DetailPrint $R1
${Next}

运行结果:

1
2
3
10
8
6

南陇居士

NLJS.SITE - 分享 · 记录

文章评论

  • mefcl

    厉害,讲的清楚。

    2024-03-27