for: Difference between revisions

From Bohemia Interactive Community
Jump to navigation Jump to search
m (Text replacement - "\<dt class\=\"note\"\>\'\'\'\[\[(.*)\]\]\'\'\'" to "<dt class="note">$1")
m (Text replacement - "{{arma2}}" to "{{GameCategory|arma2|link= y}}")
 
(34 intermediate revisions by 4 users not shown)
Line 1: Line 1:
{{Command|Comments=
{{RV|type=command
____________________________________________________________________________________________


| arma |Game name=
|game1= arma1
|version1= 1.00


|1.00|Game version=
|game2= arma2
____________________________________________________________________________________________
|version2= 1.00


|game3= arma2oa
|version3= 1.50


| This operator creates a [[For Type]] which is used in the for-constructs.
|game4= tkoh
{{Important|Alternative syntax {{Inline code|[[for]] ''array'' [[do]] ''code''}} is [[Code_Optimisation#for|slower]] than the alternative because the condition is checked every iteration.}} |Description=
|version4= 1.00
____________________________________________________________________________________________


| [[for]] arguments |Syntax=
|game5= arma3
|version5= 0.50


|p1= arguments: [[String]] - declares the index variable name like "_myVar". [[from]] and [[to]] are required, [[step]] is optional. |Parameter 1=
|gr1= Program Flow


| [[For Type]] |Return value=
|descr= This operator creates a [[For Type]] which is used in the for-constructs.
____________________________________________________________________________________________
{{Feature|warning|
The alternative syntax <sqf inline>for _array do _code</sqf> is [[Code Optimisation#for|slower]] than the first syntax because the condition is checked in every iteration, and it has a few bugs.<br>
It is only kept for backward compatibility; always use the first syntax <sqf inline>for "_i" from 0 to _end do _code</sqf>.
}}
 
|pr= {{Feature|arma2|There is an {{GameCategory|arma2|link= y}} issue (not present in {{Name|arma2oa|short}} nor in {{arma3}}) where having a capital letter in the variable name '''will''' throw an error:
<sqf>
for "_i" from 0 to 2 do { hintSilent str _i; }; // works
for "_I" from 0 to 2 do { hintSilent str _I; }; // "variable _i is not defined" error
</sqf>
}}
 
|s1= [[for]] arguments


|s2= [[for]] array |Syntax 2=
|p1= arguments: [[String]] - declares the index variable name like "_myVar". [[from]] and [[to]] are required, [[step]] is optional.


|p21= array: [[Array]] - a 3-dimensional array of [[Code]] as follow: [init, condition, codeToExecute] |Parameter 21=
|r1= [[For Type]]


|r2= [[For Type]] |Return value 2=
|s2= [[for]] [init, condition, codeToExecute]
____________________________________________________________________________________________


|x1= <code>[[for]] "_i" [[from]] 1 [[to]] 10 [[do]] { [[systemChat]] [[str]] _i; };</code> Will output 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 (the to value is inclusive) |Example 1=
|p21= init: [[Code]] - loop variable definition
|p22= condition: [[Code]] - [[call]]ed code returns [[false]], leave the [[for]]-loop
|p23= codeToExecute: [[Code]] - code to be run on each loop


|x2= <code>[[for]] "_i" [[from]] 9 [[to]] 1 [[step]] -2 [[do]] { [[systemChat]] [[str]] _i; };</code> |Example 2=
|r2= [[For Type]]


|x3= <code>[[for]] [{ _i = 0 }, { _i < 10 }, { _i = _i + 1 }] [[do]] { [[systemChat]] [[str]] _i }; </code> |Example 3=
|x1= <sqf>
// will output 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 (the to value being inclusive)
for "_i" from 1 to 10 do { systemChat str _i; };
</sqf>


|x4 = <code>{{codecomment|// BAD CODE}}
|x2= <sqf>
_i = 100;
// will output 9, 7, 5, 3, 1
[[for]] [{_i = 0}, {_i < 5}, {_i = _i + 1}] [[do]] {};  
for "_i" from 9 to 1 step -2 do { systemChat str _i; };
[[hint]] [[str]] _i; {{codecomment|// 5}}
</sqf>


{{codecomment|// GOOD CODE ([[private]] keyword is recommended)}}
|x3= <sqf>
_i = 100;
// will output 0, 1, 2, 3, 4, 5, 6, 7, 8, 9
[[for]] [{[[private]] _i = 0}, {_i < 5}, {_i = _i + 1}] [[do]] {};  
for [{ _i = 0 }, { _i < 10 }, { _i = _i + 1 }] do { systemChat str _i };
[[hint]] [[str]] _i; {{codecomment|// 100}}</code> |Example 4=
</sqf>
____________________________________________________________________________________________


| [[Control Structures]], [[while]], [[do]] |See also=
|x4= <sqf>
}}
// BAD CODE
_i = 100;
for [{ _i = 0 }, { _i < 5 }, { _i = _i + 1 }] do { /* code */ };
hint str _i; // 5
 
// GOOD CODE (private keyword is recommended)
_i = 100;
for [{ private _i = 0 }, { _i < 5 }, { _i = _i + 1 }] do { /* code */ };
hint str _i; // 100


<h3 style="display:none">Notes</h3>
// BEST CODE (primary syntax, fastest)
<dl class="command_description">
_i = 100;
<!-- Note Section BEGIN -->
for "_i" from 0 to 4 do { /* code */ };
hint str _i; // 100
</sqf>


<dd class="notedate">Posted on 29 April 2010 (CEST)
|seealso= [[Control Structures]] [[while]] [[do]]
<dt class="note">[[User:Alef|alef]]
}}
<dd class="note"><code>[[for]] "_i" [[from]] 0 [[to]]  0 [[do]] {}; {{codecomment|// will do once, with _i {{=}} 0}}
[[for]] "_i" [[from]] 0 [[to]] -1 [[do]] {}; {{codecomment|// will not do}}
</code>


<dd class="notedate">Posted on Apr 15, 2014 - 12:54
{{Note
<dt class="note">[[User:ffur2007slx2_5|ffur2007slx2_5]]<dd class="note">
|user= Alef
|timestamp= 20100429160600
|text= <sqf>
for "_i" from 0 to  0 do {}; // will do once, with _i = 0
for "_i" from 0 to -1 do {}; // will not do
</sqf>
}}


*{{GVI|arma 3|1.16}} Please note the difference between the two syntaxed; {{Inline code|[[for]] ''arguments''}} detects Boolean in each scope while {{Inline code|[[for]] ''array''}} doesn’t. e.g.
{{Note
{| class="wikitable"
|user= ffur2007slx2_5
|timestamp= 20140415125400
|text= <nowiki/>
* {{GVI|arma 3|1.16}} Please note the difference between the two syntaxes; {{hl|[[for]] ''arguments''}} detects Boolean in each scope while {{hl|[[for]] ''array''}} doesn’t. e.g.
{{{!}} class="wikitable"
! command
! command
! Structure
! Structure
! Summary
! Summary
|-
{{!}}-
| [[for]]&nbsp;''array''
{{!}} [[for]]&nbsp;''array''
|
{{!}}
a = 0; b = [[true]];
<sqf>
[[for]] [{ _i = 0 }, { _i < 10 && b }, { _i = _i + 1 }] [[do]] {
a = 0; b = true;
a = a + 1;  
for [{ _i = 0 }, { _i < 10 && b }, { _i = _i + 1 }] do
[[if]] (a >= 7) [[then]] {b = [[false]]}
{
}
a = a + 1;
| loop can be exited via Boolean control, possible workaround can be like [[BIS_fnc_areEqual]]
if (a >= 7) then {b = false}
|-
};
|[[for]]&nbsp;''arguments''
</sqf>
|
{{!}} loop can be exited via [[Boolean]] control, possible workaround can be e.g [[BIS_fnc_areEqual]]
a = 0;
{{!}}-
[[for]] "_i" [[from]] 0 [[to]] 10 [[do]] {
{{!}} [[for]]&nbsp;''arguments''
a = a + 1;
{{!}}
[[if]] (a >= 7) [[exitwith]] {}
<sqf>
};
a = 0;
| has to be exited via [[exitWith]]
for "_i" from 0 to 10 do
{
a = a + 1;
if (a >= 7) exitwith {};
};
</sqf>
{{!}} has to be exited via [[exitWith]]


|}
{{!}}}
<br>
* Never try to tell a decimal number via binary number in a loop; otherwise the loop will be infinite:
*Never try to tell a decimal number via binary number in a loop; otherwise the loop will be infinite:
<sqf>for [{_a = 0},{_a != 1},{_a = _a + 0.1}] do {}; // an infinite loop; _a will never be 1 so the scope will always be true</sqf>
<code>[[for]] [{_a = 0},{_a != 1},{_a = _a + 0.1}] [[do]] {}; {{codecomment|// an infinite loop; _a will never be 1 so the scope will always be true}}</code>
Any binary number behind the decimal point is always the sum of 1/2, 1/4, 1/8, 1/16 etc. so decimal number with odd denominator like 1/3 or 1/10 cannot be exactly equal to each other.
Any binary number behind the decimal point is always the sum of 1/2, 1/4, 1/8, 1/16 etc. so decimal number with odd denominator like 1/3 or 1/10 cannot be exactly equal to each other.
* Avoid too large factorial multiply which may loose the leading indicator in result. And 12 is the biggest accessable factor in this example.
* Avoid too large factorial multiply which may loose the leading indicator in result. And 12 is the biggest accessable factor in this example.
<code>[[for]] [{_a = 2; _b = 1;},{_a < 100},{_a = _a + 1}] [[do]] {_b = _b * _a}; {{codecomment|// _b {{=}} 1.#INF}}</code>
<sqf>for [{_a = 2; _b = 1;},{_a < 100},{_a = _a + 1}] do {_b = _b * _a}; // _b = 1.#INF</sqf>
}}


<dd class="notedate">Posted on June 4, 2015 - 19:27 (UTC)</dd>
{{Note
<dt class="note">[[User:Killzone Kid|Killzone Kid]]</dt>
|user= Killzone_Kid
<dd class="note">
|timestamp= 20150604192700
Variable name doesn't have to start with _. could be:<code>[[for]] "LAlala" [[from]] 0 [[to]] 0 [[do]] {[[hint]] [[str]] LAlala}; {{codecomment|// 0}}</code>
|text= Variable name doesn't have to start with _. could be:
<sqf>for "LAlala" from 0 to 0 do { hint str LAlala }; // 0</sqf>
The variable ''LAlala'' will exist only inside [[do]] {} scope and will not overwrite any variable of the same name that existed before.
The variable ''LAlala'' will exist only inside [[do]] {} scope and will not overwrite any variable of the same name that existed before.
</dd>
}}


<dd class="notedate">Posted on June 4, 2015 - 19:45 (UTC)</dd>
{{Note
<dt class="note">[[User:Nickorr|Nickorr]]</dt>
|user= Nickorr
<dd class="note">
|timestamp= 20150604194500
Dont use this notation if you plan to change the cycle ranges dynamically. The range values are checked only before the cycle started. Use {{Inline code|[[for]] ''array''}} instead. <br>
|text= Dont use this notation if you plan to change the cycle ranges dynamically. The range values are checked only before the cycle started. Use {{hl|[[for]] ''array''}} instead. <br>


Example, that won't work correctly:
Example, that won't work correctly:
<code><nowiki>_xy = [1,2,3,4,5,6,7,8,9,10];
<sqf>
for "_i" from 0 to (count _xy - 1) do {
_xy = [1,2,3,4,5,6,7,8,9,10];
if ( _xy select _i == 3 ) then {
for "_i" from 0 to (count _xy - 1) do
{
if ( _xy select _i == 3 ) then
{
_xy deleteAt _i;
_xy deleteAt _i;
_i = _i - 1;
_i = _i - 1;
};
};
};</nowiki></code>
};
(Here the <tt><nowiki>_i = 9</nowiki></tt> step will still be checked by the cycle, which will lead to "out of the array range" error.) <br>
</sqf>
(Here the {{hl|c= <nowiki>_i = 9</nowiki>}} step will still be checked by the cycle, which will lead to "out of the array range" error.) <br>
This code will work correctly:
This code will work correctly:
<code><nowiki>_xy = [1,2,3,4,5,6,7,8,9,10];
<sqf>
for [{_i=1},{_i<=(count _xy - 1)},{_i=_i+1}] do {
_xy = [1,2,3,4,5,6,7,8,9,10];
if ( _xy select _i == 3 ) then {
for [{_i = 1},{_i<=(count _xy - 1)},{_i=_i+1}] do
{
if ( _xy select _i == 3 ) then
{
_xy deleteAt _i;
_xy deleteAt _i;
_i = _i - 1;
_i = _i - 1;
};
};
};</nowiki></code>
};
(The last step here will be <tt><nowiki>_i = 8</nowiki></tt> with array looking like this: <tt><nowiki>[1,2,4,5,6,7,8,9,10]</nowiki></tt>)
</sqf>
</dd>
(The last step here will be {{hl|_i {{=}} 8}} with array looking like this: {{hl|[1,2,4,5,6,7,8,9,10]}})
}}


<dd class="notedate">Posted on January 29, 2016 - 05:18 (UTC)</dd>
{{Note
<dt class="note">[[User:DreadedEntity|DreadedEntity]]</dt>
|user= DreadedEntity
<dd class="note">
|timestamp= 20160129051800
For loops can be safely nested. This means that there should not be any problems with recursion.
|text= For loops can be safely nested. This means that there should not be any problems with recursion.
<code>_array = [];
<sqf>
[[for]] "_i" from 0 to 3 do
_array = [];
for "_i" from 0 to 3 do
{
{
[[for]] "_i" from 0 to 3 do
for "_i" from 0 to 3 do
{
{
_array [[pushBack]] _i;
_array pushBack _i;
};
};
_array [[pushBack]] _i;
_array pushBack _i;
};
};
[[hint]] [[str]] _array;</code>
hint str _array;
</dd>
</sqf>
 
}}
<!-- Note Section END -->
</dl>
 
[[Category:Scripting Commands|{{uc:{{PAGENAME}}}}]]
[[Category:Scripting Commands ArmA|{{uc:{{PAGENAME}}}}]]
[[Category:Scripting Commands Arma 2|{{uc:{{PAGENAME}}}}]]
[[Category:Scripting Commands Take On Helicopters|{{uc:{{PAGENAME}}}}]]
[[Category:Scripting Commands Arma 3|{{uc:{{PAGENAME}}}}]]
[[Category:Command Group: Program Flow|{{uc:{{PAGENAME}}}}]]

Latest revision as of 13:01, 19 March 2024

Hover & click on the images for description

Description

Description:
This operator creates a For Type which is used in the for-constructs.
The alternative syntax for _array do _code is slower than the first syntax because the condition is checked in every iteration, and it has a few bugs.
It is only kept for backward compatibility; always use the first syntax for "_i" from 0 to _end do _code.
Problems:
Arma 2
There is an Arma 2 issue (not present in Arma 2:OA nor in Arma 3) where having a capital letter in the variable name will throw an error:
for "_i" from 0 to 2 do { hintSilent str _i; }; // works for "_I" from 0 to 2 do { hintSilent str _I; }; // "variable _i is not defined" error
Groups:
Program Flow

Syntax

Syntax:
for arguments
Parameters:
arguments: String - declares the index variable name like "_myVar". from and to are required, step is optional.
Return Value:
For Type

Alternative Syntax

Syntax:
for [init, condition, codeToExecute]
Parameters:
init: Code - loop variable definition
condition: Code - called code returns false, leave the for-loop
codeToExecute: Code - code to be run on each loop
Return Value:
For Type

Examples

Example 1:
// will output 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 (the to value being inclusive) for "_i" from 1 to 10 do { systemChat str _i; };
Example 2:
// will output 9, 7, 5, 3, 1 for "_i" from 9 to 1 step -2 do { systemChat str _i; };
Example 3:
// will output 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 for [{ _i = 0 }, { _i < 10 }, { _i = _i + 1 }] do { systemChat str _i };
Example 4:
// BAD CODE _i = 100; for [{ _i = 0 }, { _i < 5 }, { _i = _i + 1 }] do { /* code */ }; hint str _i; // 5 // GOOD CODE (private keyword is recommended) _i = 100; for [{ private _i = 0 }, { _i < 5 }, { _i = _i + 1 }] do { /* code */ }; hint str _i; // 100 // BEST CODE (primary syntax, fastest) _i = 100; for "_i" from 0 to 4 do { /* code */ }; hint str _i; // 100

Additional Information

See also:
Control Structures while do

Notes

Report bugs on the Feedback Tracker and/or discuss them on the Arma Discord or on the Forums.
Only post proven facts here! Add Note
Alef - c
Posted on Apr 29, 2010 - 16:06 (UTC)
for "_i" from 0 to 0 do {}; // will do once, with _i = 0 for "_i" from 0 to -1 do {}; // will not do
ffur2007slx2_5 - c
Posted on Apr 15, 2014 - 12:54 (UTC)
  • Arma 3 logo black.png1.16 Please note the difference between the two syntaxes; for arguments detects Boolean in each scope while for array doesn’t. e.g.
command Structure Summary
for array

a = 0; b = true; for [{ _i = 0 }, { _i < 10 && b }, { _i = _i + 1 }] do { a = a + 1; if (a >= 7) then {b = false} };

loop can be exited via Boolean control, possible workaround can be e.g BIS_fnc_areEqual
for arguments

a = 0; for "_i" from 0 to 10 do { a = a + 1; if (a >= 7) exitwith {}; };

has to be exited via exitWith
  • Never try to tell a decimal number via binary number in a loop; otherwise the loop will be infinite:

for [{_a = 0},{_a != 1},{_a = _a + 0.1}] do {}; // an infinite loop; _a will never be 1 so the scope will always be true
Any binary number behind the decimal point is always the sum of 1/2, 1/4, 1/8, 1/16 etc. so decimal number with odd denominator like 1/3 or 1/10 cannot be exactly equal to each other.

  • Avoid too large factorial multiply which may loose the leading indicator in result. And 12 is the biggest accessable factor in this example.

for [{_a = 2; _b = 1;},{_a < 100},{_a = _a + 1}] do {_b = _b * _a}; // _b = 1.#INF

Killzone_Kid - c
Posted on Jun 04, 2015 - 19:27 (UTC)
Variable name doesn't have to start with _. could be:
for "LAlala" from 0 to 0 do { hint str LAlala }; // 0
The variable LAlala will exist only inside do {} scope and will not overwrite any variable of the same name that existed before.
Nickorr - c
Posted on Jun 04, 2015 - 19:45 (UTC)
Dont use this notation if you plan to change the cycle ranges dynamically. The range values are checked only before the cycle started. Use for array instead.
Example, that won't work correctly:
_xy = [1,2,3,4,5,6,7,8,9,10]; for "_i" from 0 to (count _xy - 1) do { if ( _xy select _i == 3 ) then { _xy deleteAt _i; _i = _i - 1; }; };
(Here the _i = 9 step will still be checked by the cycle, which will lead to "out of the array range" error.)
This code will work correctly:
_xy = [1,2,3,4,5,6,7,8,9,10]; for [{_i = 1},{_i<=(count _xy - 1)},{_i=_i+1}] do { if ( _xy select _i == 3 ) then { _xy deleteAt _i; _i = _i - 1; }; };
(The last step here will be _i = 8 with array looking like this: [1,2,4,5,6,7,8,9,10])
DreadedEntity - c
Posted on Jan 29, 2016 - 05:18 (UTC)
For loops can be safely nested. This means that there should not be any problems with recursion.
_array = []; for "_i" from 0 to 3 do { for "_i" from 0 to 3 do { _array pushBack _i; }; _array pushBack _i; }; hint str _array;