Difference between revisions of "If"
Added section on the use of Elseif and Else
imported>Qazaaq (→Combining Comparisons: clarifying OR and AND a bit, changed external into internal link) |
imported>Saebel (Added section on the use of Elseif and Else) |
||
(4 intermediate revisions by 3 users not shown) | |||
Line 73: | Line 73: | ||
Note that "||" is evaluated before "&&": '''"||" has precedence over "&&"''', just like "*" is evaluated before "+" in normal algebra. Which also shows an opposite behaviour from the standard operator notations for scripting conditional expressions (IF statements), and needs to be clarified, as it impacts the design of '''boolean expressions''': in arithmetic and algebra, from the earliest use of mathematical notation, multiplication took precedence over addition, and the standard order of operators is: 1-exponents and roots; 2-multiplication and division; 3-addition and subtraction; | Note that "||" is evaluated before "&&": '''"||" has precedence over "&&"''', just like "*" is evaluated before "+" in normal algebra. Which also shows an opposite behaviour from the standard operator notations for scripting conditional expressions (IF statements), and needs to be clarified, as it impacts the design of '''boolean expressions''': in arithmetic and algebra, from the earliest use of mathematical notation, multiplication took precedence over addition, and the standard order of operators is: 1-exponents and roots; 2-multiplication and division; 3-addition and subtraction; | ||
* In terms of computing, we're talking about a '''precedence number order''', and operator precedence is usually ordered with the corresponding number order. For expressions where two operators of different precedences compete for the same operand, the operator with the ''higher precedence'' wins. | * In terms of computing, we're talking about a '''precedence number order''', and operator precedence is usually ordered with the corresponding number order. For expressions where two operators of different precedences compete for the same operand, the operator with the ''higher precedence'' wins. | ||
* '''functions have precedence over | * '''functions have precedence over comparison and arithmetic operators, which themselves always have precedence over logical operators ("||" and "&&").''' | ||
* In ''Common operator notation'' involving "normal" algebra or boolean algebra, "*" ("&&") is always evaluated before "+" ("||"), it has a higher precedence number than the "+" operator. For example, 3×4+5 = ((3×4)+5), not (3×(4+5)). | * In ''Common operator notation'' involving "normal" algebra or boolean algebra, "*" ("&&") is always evaluated before "+" ("||"), it has a higher precedence number than the "+" operator. For example, 3×4+5 = ((3×4)+5), not (3×(4+5)). Oblivion's scripting language keeps the normal precedence for arithmetic operations, '''but OR ("||") has a higher precedence than AND ("&&")''':<pre>if myVar1 == 1 && myVar2 == 1 || myVar2 == 5</pre>is equivalent to<pre>if myVar1 == 1 && (myVar2 == 1 || myVar2 == 5)</pre>This is true when MyVar1 = 1 AND myVar2 is either 1 or 5.<BR><BR>If you need the "&&" comparison operator to be evaluated before the "||" (OR) operator, you must include its part in-between parentheses. In this case: <pre>if (myVar1 == 1 && myVar2 == 1) || myVar2 == 5</pre>is true when either myVar2 is 5 OR both, myVar1 and myVar2 are 1. | ||
The later also explains why one has to be very careful in positioning conditions in a [[ | The later also explains why one has to be very careful in positioning conditions in a [[Logical Conditions#The_Condition_List|condition list]] of an editor item: for the CS/OB's engine, OR has order preference, ''has precedence'' over AND. For example, the condition items (A AND B OR C AND D) are evaluated as (A AND (B OR C) AND D), and not (( A AND B) OR (C AND D)), as opposed to common operator notation for most languages. In general, we call this an ''inversed'' or ''negative'' notation. '''Always keep the later in mind when [http://en.wikipedia.org/wiki/Boolean_algebra_%28logic%29#Basic_operations applying boolean algebra] for evaluating some given expression when scripting''', as standard operator notations will lead you to errors. | ||
=== Oblivion evaluates entire If statement === | === Oblivion evaluates entire If statement === | ||
Line 152: | Line 152: | ||
IF IsActor != 0 && Flag != 0 | IF IsActor != 0 && Flag != 0 | ||
== Comparisons and Reference Variables == | |||
In general it is better to use GetIsReference than using the logical operators with references (see the talk page). | |||
So instead of: | |||
<pre>if ( refVar == player ) ; especially in the case of the player this is unreliable</pre> | |||
use | |||
<pre>if ( refVar.getIsReference player )</pre> | |||
Here are some equivalent ways to check if a refVar is a Welkynd stone : | |||
<pre> | |||
if refVar.GetIsID WelkyndStone | |||
if refVar.GetIsID "00000191" ; valid but avoid it - makes code less readable | |||
set refVarBase to WelkyndStone | |||
if refVar.GetIsID refVarBase | |||
set refVarBase to "00000191" ; valid but avoid it - makes code less readable | |||
if refVar.GetIsID refVarBase | |||
</pre> | |||
== Assignment == | == Assignment == | ||
Line 161: | Line 184: | ||
set bValue to ( bValue == 0 ) | set bValue to ( bValue == 0 ) | ||
If bValue is intially 1 (True), then it will be reset to 0 (False). Or if initially 0 (False), it will be reset to 1 (True). | If bValue is intially 1 (True), then it will be reset to 0 (False). Or if initially 0 (False), it will be reset to 1 (True). | ||
== Use of Else and Elseif == | |||
Although Elseif and Else are not required for an If/Endif block, they can be very useful. Here are a few basics to remember: | |||
=== Else === | |||
- There can only be one Else in an If/Endif block and it must follow after any and all other Elseifs | |||
- An Else should only be used when, in the absence of all previous If/Elseif criteria, something should definitely happen. | |||
=== Elseif === | |||
- Elseifs are useful if you want to make sure that a certain thing only happens once, even though there are potentially multiple sets of criteria that could lead to the same result. | |||
For example, with the following code: | |||
if a && b | |||
call FunctionScriptA | |||
endif | |||
if c && d | |||
call FunctionScriptA | |||
endif | |||
If by chance a && b is true, as well as c && d, then FunctionScriptA will get called twice in a row. If you only want it to be called once, then you would use elseif as follows: | |||
if a && b | |||
call FunctionScriptA | |||
elseif c && d | |||
call FunctionScriptA | |||
endif | |||
- Elseifs are also very helpful when there are a variety of different events that can happen under different conditions, but only one of the events should occur: | |||
if aInt == 23 | |||
set myVariable to 1 | |||
elseif bRef == PlayerRef | |||
set myVariable to 2 | |||
elseif cString == "Lich" | |||
set myVariable to 3 | |||
endif | |||
- You can also potentially use Elseifs to break up a complex If statement that uses lots of logical ORs. Instead of: | |||
if (a && b) || (h && k) || (x && y) | |||
do stuff | |||
else | |||
do other stuff | |||
endif | |||
you could do: | |||
if a && b | |||
do stuff | |||
elseif h && k | |||
do same stuff as above | |||
elseif x && y | |||
do same stuff as above | |||
else | |||
do other stuff | |||
endif | |||
As mentioned earlier in this page, Oblivion will evaluate an entire If line, even when it technically would not need to. This technique avoids that issue by breaking the OR criteria into separate chunks. So even though there are more lines of code, performance-wise it may be more efficient, depending on what the criteria is. | |||
'''Tip:''' When deciding which order to do your Ifs/Elseifs, go in ascending order of simplicity/complexity. The first If statement should be very easy for the Oblivion engine to calculate. The next should be the second easiest, and so on. | |||
'''Disclaimer:''' You may want to ignore the priority of simplicity/complexity depending on the importance of the If/Elseif criteria. If there are two sets of criteria of equal value, then put the simpler to evaluate criteria first. If one set of criteria has to be checked first before processing the rest, then that becomes the priority, regardless of complexity. | |||
== Notes == | == Notes == |