1、在FB块中使用结构编写FB块的准则,就是其使用的内部变量尽量与外部隔离,除了像PLC的新启动/重启动标志,以及一些方波/脉冲波等全局变量可以在FB块中使用外,其他的任何全局变量都不应该在FB内部使用,即使是自定义结构也应该在FB中单独定义,在FB块中使用结构应该在静态类型变量中定义,
如下: VAR // Static Variables IM:STRUCT //Data structure of Internal Flags H1_AFCountImp:BOOL:=False; // Aux Flag Counter Impulse H1_CountImp:BOOL:=False; // Counter Impulse H1_ELCountMV:BOOL:=False; // Endless Counter Maximum Value END_STRUCT; //other data structure … END_VAR 在使用这些结构时,可以按照如下方式: IM. H1_CountImp:=Imp;
2、在SCL中替代Set/Reset指令的方法 SCL中不存在Set/Reset指令,或者说也没有必要使用。在SCL中,不使用排他条件Else的条件语句就是一个Set/Reset指令。
如下编程: IF THEN Variable name:=1; END_IF; 其等同于: (S)
若加上Else条件,如下: IF THEN Variable name:=1; ELSE Variable name:=0; END_IF; 则等同于: ( )
一条完整的包含置位和复位的语句可以使用如下方式编程:
IF THEN Variable name:=1; END_IF; IF THEN Variable name:=0; END_IF; 其等效于SR指令,若将上面的两个条件语句的先后次序颠倒一下,则等效于RS指令。
3、简化程序指令 <1>、尽量使用赋值语句替代那些不用于SR/RS指令的BOOL型赋值条件语句,如下: IF fnAdd &(button=false) THEN pus1:=true; ELSE pus1:=false; END_IF; 其等效于pus1:= fnAdd &(NOT button),这样使程序看起来更加简洁和容易阅读。
<2>、对于非BOOL型赋值语句则不能这如上简化,而是可以通过SEL函数实现: IF fnAdd &(button=false) THEN pus1:=value1; ELSE pus1:= value2; END_IF; 其等效于pus1:= SEL (G:= fnAdd &(button=false), IN0:= value2,IN1:= value1); 使用该函数时注意两点:<1>、参数名不能省略;<2>、当选择条件G为TRUE时,选择后一个参数值IN1,为FALSE时,选择前一个参数值IN0,这点与计算机C语言等正好相反。 <3>、XOR指令有着比AND 和OR更为复杂的表达,能使用XOR的地方应该尽量使用 IF (condition1 AND (NOT condition2)) OR (condition2 AND ( NOT condition1)) THEN Result:=true; ELSE Result:=false; END_IF; 其等效于Result:= condition1 XOR condition2; XOR功能就是两条件不同输出TRUE,相同输出FALSE
4、脉冲沿检测功能使用以下两条语句替代脉冲上升沿检测函数,譬如检测button_Input上升沿的代码如下: Puls:=button_Input & (NOT button_Last); button_Last:= button_Input; 同样的下降沿脉冲检测如下: Puls:= ( NOT button_Input) & button_Last; button_Last:= button_Input;
5、编写脉冲发生器波峰持续时间仅为一个PLC扫描周期的波形称为脉冲波,而波峰持续时间大于或等于两个PLC扫描周期的波形称为方波,脉冲波可用于计数、定时,方波可用于控制信号灯的闪烁输出,可以在西门子PLC的硬件配置中配置一个字节的各种时间的方波(波峰时间和波谷时间为1:1),假设"FP_1Sec" 为这个字节中1秒的方波,则: <1>、间隔1秒的脉冲波“Impls_1Sec” 如下编程: “Impls_1Sec” := "FP_1Sec" AND (NOT “Impls_1Sec_Aux”); “Impls_1Sec_Aux”:= "FP_1Sec" ; <2>、间隔10秒的脉冲波“Impls_10Sec” 如下编程: IF (“Impls_10Sec” ) THEN Count_ Actual:=0; “Impls_10Sec”:=0; ELSE IF (“Impls_1Sec” ) THEN Count_ Actual:= Count _ Actual +1; END_IF; “Impls_10Sec”:= Count_ Actual>=10; END_IF; Count_ Actual的初始值为0,同时当系统新启动时,也需将其设为零。间隔更长时间的脉冲波编程都可以按照上面的方式编程。
6、尽量使用编程计数功能来替代定时器功能,这样使程序更可靠和易于阅读假设Input_Condition为输入,Output_Delay为通过定时处理后的输出,Timer_Setpoint为时间设定点,Timer_Actual为当前时间计数的实际值,“Impls_1Sec” 为系统编程产生的1秒脉冲。 <1>、在输入条件满足的情况下,延时输出的定时器: IF (NOT Input_Condition) THEN Timer_Actual:= 0; Output_Delay:= 0; ELSE IF (“Impls_1Sec” AND NOT Output_Delay) THEN Timer_ Actual:= Timer_ Actual +1; END_IF; Output_Delay:= Timer_Actual >= Timer_Setpoint; END_IF;
<2>、有记忆的延时输出定时器,即在延时过程中,若输入条件终止,不影响延时,这种定时器必须使用其它的信号复位。 IF Input_Condition THEN Output_Aux:=1; END_IF; IF (NOT Output_Aux) THEN Timer_Actual:= 0; Output_Delay:=0; ELSE IF (“Impls_1Sec” AND NOT Output_Delay) THEN Timer_ Actual:= Timer_ Actual +1; END_IF; Output_Delay:= Timer_Actual >= Timer_Setpoint; END_IF; 若想终止Output_Delay的输出,必须在后面追加一条条件语句,用于复位Output_Aux
<3>、立即输出,延时断开的定时器 IF Input_Condition THEN Timer_Actual:= 0; Output_Aux:= 0; Output_Delay:=1; //立即输出 ELSE IF (“Impls_1Sec” AND NOT Output_Aux) THEN Timer_ Actual:= Timer_ Actual +1; END_IF; Output_Aux:= Timer_Actual >= Timer_Setpoint; END_IF; IF Output_Aux THEN Output_Delay:=0; //延时断开 END_IF;
<4>、在检测到一个上升沿脉冲后,立即输出,并开始计时,在时间到达后断开。 IF Input_Condition THEN Output_Aux:=1; END_IF; IF (NOT Output_Aux) THEN Timer_Actual:= 0; Timer_Arrived := 0; ELSE IF (NOT Timer_ Arrived AND “Impls_1Sec” ) THEN Timer_Actual:= Timer_Actual +1; END_IF; Timer_ Arrived := Timer_Actual >= Timer_Setpoint; END_IF; IF Timer_ Arrived THEN Output_Aux:=0; END_IF; Output_Delay:= Output_Aux;
通过以上的编程方式可以实现任何定时器功能,而代码却可以为不同的PLC系统所使用。
7、使用编程计数功能来替代计数器在SCL语言中使用计数功能是最为简单的,其关键是必须首先对输入进行脉冲检测假设Input_Imp为输入脉冲,CountImp为输入脉冲检测,Counter为计数值,Factor为计数因子(更详细点就是每来一次脉冲,计数值增加多少)。 (*----- Create Impulse (Impulse Evaluation) -----------------------------------------------------*) CountImp:= Input_Imp AND (NOT CountImp_Old); CountImp_Old:= Input_Imp; (*----- Counter ---------------------------------------------------------------------------------*) IF CountImp THEN Counter:= Counter+Factor; END_IF; 一个完整的计数程序应该还有计数器复位功能以及计数值上限检测条件(以防止计数值溢出)。
8、新故障/新警告的检测一个完整的FB块应该能够检测故障/警告,以及新故障/新警告,假设Input1, Input2… Inputn对应故障的输入(有信号表示OK),Fault1, Fault2… Faultn对应故障位,NFault1, NFault2…NFaultn对应新故障位,Flt和NFlt分别对应综合的故障和新故障,Ackn对应故障应答输入,为常开点,Mute对应新故障消除输入(或者称为蜂鸣器沉寂),为常开点: Fault1:= NOT Input1 OR (Fault1 AND NOT Ackn); NFault1:= Fault1 AND (Mute OR NFault1); Fault2:= NOT Input2 OR (Fault2 AND NOT Ackn); NFault2:= Fault2 AND (Mute OR NFault2); … Flt := Fault1 OR Fault2 OR Faultn NFlt :=(Fault1 AND NOT NFault1) OR (Fault2 AND NOT NFault2) OR (Faultn AND NOT NFaultn) NFlt就是最终的新故障输出指示,新警告的检测与之类似。
9、字中取位字中取位有两种方式,一种是通过西门子所特有的字取位方式实现,一种是通过计算机编程的标准方式实现,假设Input_Word为输入参数,Word类型,W0,W1,…W15为位变量。
<1>、通过西门子的M变量实现: Temp_Aux:=MW[10]; MW[10]:=Input_Word; W0:=M[11,0]; W1:=M[11,1]; W2:=M[11,2]; W3:=M[11,3]; W4:=M[11,4]; W5:=M[11,5]; W6:=M[11,6]; W7:=M[11,7]; W8:=M[10,0]; W9:=M[10,1]; W10:=M[10,2]; W11:=M[10,3]; W12:=M[10,4]; W13:=M[10,5]; W14:=M[10,6]; W15:=M[10,7]; MW[10]:=Temp_Aux;
<2>、通过标准编程实现 w0:=(Input_Word & 16#1)=16#1; w1:=(Input_Word & 16#2)=16#2; w2:=(Input_Word & 16#4)=16#4; w3:=(Input_Word & 16#8)=16#8; w4:=(Input_Word & 16#10)=16#10; w5:=(Input_Word & 16#20)=16#20; w6:=(Input_Word & 16#40)=16#40; w7:=(Input_Word & 16#80)=16#80; w8:=(Input_Word & 16#100)=16#100; w9:=(Input_Word & 16#200)=16#200; w10:=(Input_Word & 16#400)=16#400; w11:=(Input_Word & 16#800)=16#800; w12:=(Input_Word & 16#1000)=16#1000; w13:=(Input_Word & 16#2000)=16#2000; w14:=(Input_Word & 16#4000)=16#4000; w15:=(Input_Word & 16#8000)=16#8000;
使用方式1会更加简单和容易理解一些,但方式2具有更加宽广的应用场合,更加标准化,即使是当今的计算机编程在取位操作时也类似于上面的编程。字取位的场合,一般用于总线数据(譬如变频器的状态数据),则可能是字/整数,此时就需要用到上面的编程。 10、将位组合成字相当于“字中取位”的反向操作,这也有两种方法,一种方法是使用M变量,类似于“字中取位”的方式<1>,另一种也是标准编程,假设Output_Word为输出参数,Word类型,W0,W1,…W15为位变量。 <1>、通过西门子的M变量实现: Temp_Aux:=MW[10]; M[11,0] := W0; M[11,1] := W1; M[11,2] := W2; M[11,3] := W3; M[11,4] := W4; M[11,5] := W5; M[11,6] := W6; M[11,7] := W7; M[10,0] := W8; M[10,1] := W9; M[10,2] := W10; M[10,3] := W11; M[10,4] := W12; M[10,5] := W13; M[10,6] := W14; M[10,7] := W15; Output_Word:=MW[10]; MW[10]:=Temp_Aux; <2>、通过标准编程实现 IF W0 THEN Output_Word:=Output_Word OR 16#1; ELSE Output_Word:=Output_Word AND (NOT 16#1); END_IF; IF W1 THEN Output_Word:=Output_Word OR 16#2; ELSE Output_Word:=Output_Word AND (NOT 16#2); END_IF; IF W2 THEN Output_Word:=Output_Word OR 16#4; ELSE Output_Word:=Output_Word AND (NOT 16#4); END_IF; IF W3 THEN Output_Word:=Output_Word OR 16#8; ELSE Output_Word:=Output_Word AND (NOT 16#8); END_IF; IF W4 THEN Output_Word:=Output_Word OR 16#10; ELSE Output_Word:=Output_Word AND (NOT 16#10); END_IF; IF W5 THEN Output_Word:=Output_Word OR 16#20; ELSE Output_Word:=Output_Word AND (NOT 16#20); END_IF; IF W6 THEN Output_Word:=Output_Word OR 16#40; ELSE Output_Word:=Output_Word AND (NOT 16#40); END_IF; IF W7 THEN Output_Word:=Output_Word OR 16#80; ELSE Output_Word:=Output_Word AND (NOT 16#80); END_IF; IF W8 THEN Output_Word:=Output_Word OR 16#100; ELSE Output_Word:=Output_Word AND (NOT 16#100); END_IF; IF W9 THEN Output_Word:=Output_Word OR 16#200; ELSE Output_Word:=Output_Word AND (NOT 16#200); END_IF; IF W10 THEN Output_Word:=Output_Word OR 16#400; ELSE Output_Word:=Output_Word AND (NOT 16#400); END_IF; IF W11 THEN Output_Word:=Output_Word OR 16#800; ELSE Output_Word:=Output_Word AND (NOT 16#800); END_IF; IF W12 THEN Output_Word:=Output_Word OR 16#1000; ELSE Output_Word:=Output_Word AND (NOT 16#1000); END_IF; IF W13 THEN Output_Word:=Output_Word OR 16#2000; ELSE Output_Word:=Output_Word AND (NOT 16#2000); END_IF; IF W14 THEN Output_Word:=Output_Word OR 16#4000; ELSE Output_Word:=Output_Word AND (NOT 16#4000); END_IF; IF W15 THEN Output_Word:=Output_Word OR 16#8000; ELSE Output_Word:=Output_Word AND (NOT 16#8000); END_IF; 同样的,使用标准化编程会繁琐一些,但有着很强的通用性,在总线通讯控制中,很多控制字(如变频器)都是以字的形式传递,所以需要把一些BOOL数据合并到一个字中,可以采用上面的对字中的位进行置位/复位操作的方式,但事实上使用时,控制命令可能只有启动/停止和方向控制等,所以这是可以直接对输出赋值,譬如当我们知道16#0F对应启动命令和正传时,可以直接使用如下赋值语句即可控制变频器正向运转:Output_Word:= 16#0F,如需反向运转,则再赋另一个值即可,而不需要像上面那样对字的每一位操作。以上是我总结的一些使用技巧,其编程可能有更好的实现方式,欢迎来信探讨。