Programmer Guide/STx Guru: Difference between revisions
No edit summary |
No edit summary |
||
(176 intermediate revisions by 4 users not shown) | |||
Line 1: | Line 1: | ||
{{DISPLAYTITLE:Becoming | {{DISPLAYTITLE:Becoming an {{STx}} Guru}} | ||
The following text is based on an {{STX}} workshop of 2005, which took place in the [http://www.kfs.oeaw.ac.at Acoustics Research Institute (ARI)], and which was described as "our weekly time-wasting meeting" in a colleague's blog. It explains the internal structure and the features of the {{STX}} 3.7 release, and has been carefully adapted to the current {{Stx}} version (4.0) in 2014. | |||
__TOC__ | __TOC__ | ||
Line 7: | Line 6: | ||
==Whom this is for== | ==Whom this is for== | ||
This manual is a general introduction to the programming language that is part of the {{ | This manual is a general introduction to the programming language that is part of the {{Stx}} environment. Reading this manual will allow you to implement procedural functions and then to proceed to object-oriented classes for a wide range of tasks, with a natural focus on numerical applications, sound processing, and visualization. As a reader of this manual, you should be slightly familiar with programming in general, and it would be great if you were an avid user of {{STx}} (on the other hand, being faintly familiar with starting up {{STX}}, loading sound files, and maybe even starting the spectrogram function should suffice). | ||
This manual is intended for reading from start to end (not necessarily without interruptions). It is not a reference manual (there is such a thing, too: the [[Programmer_Guide/Quick_Reference|{{STx}} Quick Reference]]), meaning that it will abstract from, you might even bluntly say: omit, many a detail in order not to depress the reader with a seemingly abundant amount of material. Instead, a careful selection has been taken on what is, and what is not, necessary for achieving common goals. Of course this selectivity (or, if you prefer, these omissions) try hard not to give any false impressions of what can, and what can't, be done with {{Stx}} (and how). We are well aware that writing such an introductory programmer's (or, as we hope, programmers') manual is a slippery slope, and we hope for the reader's (or, as we hope, the readers') pardon if the depth covering each topic is too shallow, or to deep, or both (we are not quite sure whether the latter is logically possible, but one never knows). We appreciate any comments or criticism, both on this manual and on {{STx}}. | |||
If you want to dig deeper into any specific topic, or if you are looking for information on a specific issue, we recommend having a look either at the online help of {{STX}} (just start {{Stx}} and select the "Help" menu, or press the "F1" key), or, even better, at the [[Programmer_Guide|{{STx}} Programmer Guide]]. Both are a vast and ever-growing collection of deep, sometimes even exhausting wisdom. They are, in fact, the official reference manual. | |||
When new to {{STX}} programming, we recommend first reading this manual (you might skip paragraphs you find particularly uninteresting), probably trying out all the examples by yourself, preferably even modifying and improving them. Afterwards, you will have a basis firm enough for using the [[Programmer_Guide|{{Stx}} Programmer Guide]], or even the [[Programmer_Guide/Quick_Reference|{{STx}} Quick Reference]], as a reference manual. And you will surely have a sound basis for solving all your future programming tasks with {{STx}}, disposing of the need for any other programming languages or environments. | |||
== How to write an {{STX}} Script: The Easy Part == | |||
This chapter deals with quite a basic fact, answering the first question that may come to your mind: ''How'' can I create and edit a script, and ''what'' need I do to submit it to {{Stx}} for execution? | |||
Well, the first part is easy: Just use a text editor of your choosing. Since Windows does not come equipped with the editor of your choosing, you should probably get it from [http://www.vim.org/download.php www.vim.org], unless, of course, you already did so. | |||
There is a faint convention to use the filename extension ".STS" for {{STx}} scripts, but everything else will really work equally well. So: just start editing. In its simplest case, your script will contain only one macro. The start of the macro is indicated by macro header, and it should end with an <code>EXIT</code> statement: | |||
[macro TowersOfHanoi] | |||
[[Programmer_Guide/Macro_Library/StdLib#UM_and_EM|um]] 'Rapid prototyping of the towers of Hanoi problem. Program logic not yet present.' | |||
exit | |||
There is, of course, [[Programmer_Guide/Source_code|a more formal definition of what an {{Stx}} script is built up from]], but at this stage you should probably not bother with all these other thingies. | |||
After crafting your {{STX}} script, you still have {{Stx}} to notice its presence, and to execute it. The simplest way for working on a script is pushing the large button labelled "Script file" (on the top of the main {{STx}} window, labelled (and called) "Workspace". After pressing this button, a file dialog titled "Select script file" will open, allowing you to, well, select your shiny new script file. | |||
[[File:ws_sc.png]] | |||
To the right of the "Script file" area, there is another area labelled "Macro". If your script file contains more than one macro function, this drop-down thingy will allow you to select which macro to execute. | |||
There is a third field, labelled "Arguments", where you can supply arguments for your macro. And there are two awesome checkboxes labelled "Debug" and "Console", respectively. When checking "Debug", {{STX}} will start your amazing macro program in the [[User_Guide/Debugger|gorgeous {{Stx}} macro debugger]]. When checking "Console", your macro will run in the interactive {{STx}} console (an interactive version of the {{STX}} command interpreter). When checking both, either may happen, or both. | |||
The awesome buttons, fields and checkboxes mentioned here build up what we call the gorgeous Script Controller. You will not find much more information in [[User_Guide/Workspace/Script_Controller|Script Controller]]. | |||
' | ''Note'': If you do not see any of the fields mentioned here (bluntly put: if you do not see the Script Controller at all), it is likely that it is not there. In this case, try using the Workspace menu item "Scripts > Show Scripts" to unhide it. | ||
==Constants== | |||
Our {{Stx}} does not strictly discern between string constants and numerical constants. Normally each constant is considered a string, there only being exceptions dependent on the context where the constant occurs. If, for example, a constant occurs as part of a numerical expression, {{STx}} tries to get its numerical value. | |||
Generally, you may, but you need not, put single quotes around constants. With a few exceptions depending on context, this will not change the way {{STX}} handles the constant. So each of the following strings is a valid {{Stx}} constant: | |||
string1 | |||
'string1' | |||
-12.34 | |||
'-12.34' | |||
Regardless of the presence or absence of single quotes, both the first and the second argument will be considered string constants. Also regardless of the presence or absence of single quotes, both the third and the fourth constant will be considered numerical constants when occurring in a numerical context, or string constants when occurring in a string context. | |||
If a string constant contains whitespace characters, it depends on the context whether {{STX}} considers it ''one'' string constant or ''more than one'' string constant. If you want to make sure that a string constant is considered one constant, you should always put single quotes around the whole affair: | |||
'Hello World' | |||
Hello World | |||
While the first string is always considered one string constant, the second one may, depending on where it is occurring, be considered one string constant denoting the string "Hello World", or two string constants, denoting the strings "Hello" and "World", respectively. | |||
These issues will be dealt with in more detail below. For the moment, the curious reader may have a look at the following assignment statements: | |||
#a := [[Programmer_Guide/Command_Reference/NUM|num]] '5' * '3' // value of #a will be 15 | |||
#b := [[Programmer_Guide/Command_Reference/SET|set]] 5 * 3 // value of #a will be "5 * 3" | |||
The first statement is a numerical assignment (denoted by the keyword "<code>[[Programmer_Guide/Command_Reference/NUM|num]]</code>"). Both constants, 5 and 3, will be considered numerical constants, even if surrounded by quote characters. On the other hand, the second statement is a string assignment (denoted by the keyword "<code>[[Programmer_Guide/Command_Reference/SET|set]]</code>"). So the argument will, logically, be interpreted as one string constant, "5 * 3", even though it contains whitespace and lacks any quote character. What happens physically is that all separate words will be concatenated to the one string constant expected, inserting exactly one blank between each pair of words. The following statements show the consequences of this procedure: | |||
#b := [[Programmer_Guide/Command_Reference/SET|set]] 5 * 3 // value of #b will be "5 * 3" | |||
#b := [[Programmer_Guide/Command_Reference/SET|set]] 5 * 3 // value of #b will be "5 * 3", too | |||
#b := [[Programmer_Guide/Command_Reference/SET|set]] '5 * 3' // value of #b will also be "5 * 3" | |||
In the second statement, though the words "5", "*", and "3" are separated by more than one whitespace character, the one string that will be built up from them is "5 * 3" – when concatenating them, they get separated by exactly one space character. | |||
You may influence the way concatenation works by quoting some, or all, of the words to concatenate. Quoting a word will prevent {{STx}} from automatically inserting a blank before and after that word. So, compare the above statements with the statements below: | |||
#b := [[Programmer_Guide/Command_Reference/SET|set]] 5 '*' 3 // #b is "5*3" (no space) | |||
#b := [[Programmer_Guide/Command_Reference/SET|set]] '5' * 3 // #b is "5* 3" (one space) | |||
#b := [[Programmer_Guide/Command_Reference/SET|set]] 5 ' * ' 3 // #b is "5 * 3" (three spaces) | |||
With the | With the first statement, the word in the middle, "*", is quoted. This indicates {{STX}} on concatenation not to insert a space character either before or after this word, resulting in <var>#b</var> being set to "5*3" (no intervening whitespace). | ||
With the | With the second statement, the first word is quoted and will, hence, be concatenated to its right successor (there is no left predecessor) without inserting space. The second and the third word are not quoted and will be concatenated with an additional space in between them. This results in <var>#b</var> being assigned "5* 3" (no whitespace between "5" and "*", one blank between "*" and "3"). | ||
With the third statement, concatenation will not add any additional blanks either before or after the word in the middle. The whitespace that is part of the word, i.e. part of the quotation (three blanks before and after the asterisk, each), will be unaltered, though. So what results it <var>#b</var> being assigned the string "5 * 3" (exactly three blanks both before and after the asterisk). | |||
Within a constant, you may alter the meaning of special characters by using the {{Stx}} escape character "`", the backwards single quote, sometimes called back-tick. At the current stage, we can only use this feature for defining a string constant that contains single quote characters themselves: | |||
#a := set 'Rome is a city but `'Rome`' is a four-letter word' | |||
#a := set Rome is a city but `'Rome`' is a four-letter word | |||
Both statements will assign the string "Rome is a city but 'Rome' is a four-letter word" to a variable called #a (although it may not always be easy later to retrieve the value of this variable). We have to leave these issues open for later discussion. | Both statements will assign the string "Rome is a city but 'Rome' is a four-letter word" to a variable called <var>#a</var> (although it may not always be easy later to retrieve the value of this variable). We have to leave these issues open for later discussion. | ||
==Variables== | ==Variables== | ||
The names of {{ | The names of {{STx}} variables start with an optional one-character prefix indicating the scope of this variable (the lack of such a prefix indicating shell-global scope, see below). Besides this prefix, they may consist of letters and digits, although their first actual character must be a letter. Names are not case-sensitive, meaning that e.g. „freq", „Freq", and „FREQ", are names of the same variable. | ||
{{STX}} discerns four kinds of scope and, hence, four kinds of variables: | Our {{STX}} discerns four kinds of scope and, hence, four kinds of variables: | ||
{| | {| | ||
Line 89: | Line 105: | ||
|- | |- | ||
|Global | |Global | ||
|@ | |<code>@</code> | ||
|Global variables are known to, and may be changed by, every shell instance, both running or yet to start. Global variables are the only kind of variables guaranteed to be persistent over interactive macro calls during one | |Global variables are known to, and may be changed by, every shell instance, both running or yet to start. Global variables are the only kind of variables guaranteed to be persistent over interactive macro calls during one {{Stx}} run. | ||
|- | |- | ||
|Shell | |Shell | ||
|no prefix | |no prefix | ||
|"Shell | |"Shell-global" variables are global to the running shell. This implies that they are known to, any may be changed by, any macro invoked by a normal macro call. The variable will not, though, be known to other running shells or to shells yet to start. So, in general, shell-global variables will not be persistent through interactive calls to a user-defined macro.N.B.: Do not mix up shell-global variables with item handles that will be dealt with in chapter [[Programmer Guide/STx Guru/Shell Items: An Overview|5]] (page 1). For the time being, suffice it to say that item handles look like shell-global variables, but that they reside in different namespaces and that they behave differently. | ||
|- | |- | ||
|Local | |Local | ||
|# | |<code>#</code> | ||
|Local variables are valid only during the | |Local variables are valid only during the runtime of one invocation of a shell macro. On each invocation of a shell macro, a separate namespace containing all its local variables is being created. This namespace is destroyed as soon as the macro finishes. Note that this implies that on invocating a macro recursively, it will find itself starting with a fresh, empty copy of all its local variables, while the calling instance will find its namespace, i.e. its local variables, untouched. | ||
|- | |- | ||
|Member | |Member | ||
|& | |<code>&</code> | ||
|When adhering to the object | |When adhering to the object-oriented programming paradigm, you will define classes and instantiate them. Each instance of a class will have its own set of variables called member variables. The introductory chapters will stick to the clean ole' procedural way of programming. | ||
|} | |} | ||
As a rule of thumb, most of the time you will be using local variables whose names start with "<code>#</code>", e.g. variables like <var>#i</var>, <var>#depth</var> or <var>#title</var>. | |||
Variables need neither be declared nor to be explicitly initialized. If you want to introduce a variable, just start using it. Note that a variable is empty, i.e. contains the empty string, when being referred to without having been explicitly set to a value different from the void. | Variables need neither be declared nor to be explicitly initialized. If you want to introduce a variable, just start using it. Note that a variable is empty, i.e. contains the empty string, when being referred to without having been explicitly set to a value different from the void. | ||
===Typing=== | |||
There is really not much about typing in {{ | There is really not much about typing in {{STx}}. All variables store strings, like it is the case with most scripting languages. Of course these strings are at liberty only to consist of digits, a comma, and an optional sign character, making them look like numbers, smell like numbers, taste like numbers, and being treated like numbers by numerical {{STX}} functions like addition, multiplication, or even the controversial subtraction. | ||
===Setting Variables=== | |||
Setting a variable, i.e. assigning a value to the variable, is done with the "<code>:=</code>" operator. Its general format is | Setting a variable, i.e. assigning a value to the variable, is done with the "<code>:=</code>" operator. Its general format is | ||
variable := expression | |||
You should ''never'' use this kind of assignment with {{Stx}}, though. Instead, ''always'' use one of the following assignments: | |||
variable := [[Programmer_Guide/Command_Reference/SET|SET]] string_expression // typed string assignment | |||
variable := [[Programmer_Guide/Command_Reference/INT|INT]] numerical_expression // typed integer assignment | |||
variable := [[Programmer_Guide/Command_Reference/NUM|NUM]] numerical_expression // typed numerical assignment | |||
variable := [[Programmer_Guide/Command_Reference/EVAL|EVAL]] amazing_expression // awesome assignment using [[Programmer_Guide/Command_Reference/EVAL|EVAL]] | |||
See the following chapters for the reasons of this recommendation. | |||
====Simple string assignment==== | |||
In its simplest form, this "expression" is a simple string constant, just like in the following example: | In its simplest form, this "expression" is a simple string constant, just like in the following example: | ||
#adress := 'Reichsratsstrasse 17' // discouraged, see further below - always use [[Programmer_Guide/Command_Reference/SET|SET]] | |||
Or even: | Or even: | ||
#adress := Reichsratsstrasse 17 // discouraged, see further below - always use [[Programmer_Guide/Command_Reference/SET|SET]] | |||
Although generally valid, either usage is strongly discouraged, because it may lead to often surprising, seldom desired results if or when the string to be assigned starts with the name of a built-in function or a user-defined macro (note that words so far not reserved may become reserved words any time now, and that, when in a large software-building project, you never know how your colleagues call their helper macros today). | Although generally valid, either usage is strongly discouraged, because it may lead to often surprising, seldom desired results if or when the string to be assigned starts with the name of a built-in function or a user-defined macro (note that words so far not reserved may become reserved words any time now, and that, when in a large software-building project, you never know how your colleagues call their helper macros today). | ||
====Typed string assignment==== | |||
You may indicate your expressed desire for the expression to be a string constant by prefixing it with the type-selector statement "set": | You may indicate your expressed desire for the expression to be a string constant by prefixing it with the type-selector statement "<code>[[Programmer_Guide/Command_Reference/SET|set]]</code>": | ||
#address := [[Programmer_Guide/Command_Reference/SET|set]] Reichsratsstrasse 17 // this is better | |||
or, even better: | or, even better: | ||
#address := [[Programmer_Guide/Command_Reference/SET|set]] 'Reichsratsstrasse 17' // and this is the way to go | |||
In this case, the assignment will even work if one day the {{ | In this case, the assignment will even work if one day the {{Stx}} macro language should be added a built-in function "Reichsratsstrasse" (which is, we dare to admit, unlikely) – or if one of your colleagues' macros happens to be likely called. | ||
====Typed numerical assignment==== | |||
Although the value assigned is invariably a string, this string may be the result of a numerical computation. You may indicate your desire for it to be so by using one out of the following type-selector statements: "<code>int</code>", "<code>num</code>", and "<code>eval</code>". | Although the value assigned is invariably a string, this string may be the result of a numerical computation. You may indicate your desire for it to be so by using one out of the following type-selector statements: "<code>[[Programmer_Guide/Command_Reference/INT|int]]</code>", "<code>[[Programmer_Guide/Command_Reference/NUM|num]]</code>", and "<code>[[Programmer_Guide/Command_Reference/EVAL|eval]]</code>". | ||
The "<code>int</code>" type-selector will cause your expression being evaluated as an integer expression. More precisely (more precisely less wrongly), the expression will be evaluated numerically, and the result will be converted to an integer whose textual representation will be the string to be assigned to the destination variable. The calculation itself will be done with the point floating, though (see the below examples for what that means). | The "<code>[[Programmer_Guide/Command_Reference/INT|int]]</code>" type-selector will cause your expression being evaluated as an integer expression. More precisely (more precisely less wrongly), the expression will be evaluated numerically, and the result will be converted to an integer whose textual representation will be the string to be assigned to the destination variable. The calculation itself will be done with the point floating, though (see the below examples for what that means). | ||
The "<code>num</code>" type-selector will cause your expression being evaluated as a numerical expression, provided it is such. The textual representation of the numerical result will be the string to assign to the destination variable. | The "<code>[[Programmer_Guide/Command_Reference/NUM|num]]</code>" type-selector will cause your expression being evaluated as a numerical expression, provided it is such. The textual representation of the numerical result will be the string to assign to the destination variable. | ||
The "<code>eval</code>" type-selector is the most powerful of them all. Firstly, it does everything the "<code>num</code>" type-selector does. So, when evaluating a plain numerical expression, it is your free choice whether to use "<code>num</code>" or "<code>eval</code>" (we might one day choose spontaneously to fade out the "<code>num</code>" type-selector, but do not allow this to bias your choice). But, secondly: The "<code>eval</code>" type-selector is capable of much, much more: It does vector and matrix operations of the most sophisticated kind, calculates averages, converts units, dances the Fourier transform, and so on. The actual number of functions available to "<code>eval</code>" expressions is more than 70 and counting (see the interactive {{STX}} help topics "EVAL | The "<code>[[Programmer_Guide/Command_Reference/EVAL|eval]]</code>" type-selector is the most powerful of them all. Firstly, it does everything the "<code>[[Programmer_Guide/Command_Reference/NUM|num]]</code>" type-selector does. So, when evaluating a plain numerical expression, it is your free choice whether to use "<code>[[Programmer_Guide/Command_Reference/NUM|num]]</code>" or "<code>[[Programmer_Guide/Command_Reference/EVAL|eval]]</code>" (we might one day choose spontaneously to fade out the "<code>[[Programmer_Guide/Command_Reference/NUM|num]]</code>" type-selector, but do not allow this to bias your choice). But, secondly: The "<code>[[Programmer_Guide/Command_Reference/EVAL|eval]]</code>" type-selector is capable of much, much more: It does vector and matrix operations of the most sophisticated kind, calculates averages, converts units, dances the Fourier transform, and so on. The actual number of functions available to "<code>eval</code>" expressions is more than 70 and counting (see the interactive {{STX}} help topics "[[Programmer_Guide/Command_Reference/EVAL|eval]]"). | ||
For an example, compare the following statements: | For an example, compare the following statements: | ||
#a := [[Programmer_Guide/Command_Reference/NUM|num]] 3*3.4 // result is 10.19999999999 | |||
#a := [[Programmer_Guide/Command_Reference/INT|int]] 3*3.4 // result is 10 (!) | |||
#a := [[Programmer_Guide/Command_Reference/NUM|num]] 3*int(3.4) // result is 9 | |||
#a := [[Programmer_Guide/Command_Reference/INT|int]] 3*int(3.4) // result is 9 | |||
#a := [[Programmer_Guide/Command_Reference/NUM|num]] int(3*3.4) // result is 10 (!) | |||
#a := [[Programmer_Guide/Command_Reference/NUM|num]] int(3*int(3.4)) // result is 9 | |||
Here the first statement will cause <code>#a</code> to be assigned the result of the floating point multiplication of 3 and 3.4, i.e. about 10.2. In the second statement, the same multiplication will be calculated, and only the result of this calculation, i.e. 10.2 (roughly...), will get truncated to an integer – this integer in turn being 10. Only the third and the fourth statement (both!) will cause the second factor, 3.4, to be broken down to an integer before multiplication. This usage, though, is strictly not a feature of the type selector-based assignment, but of the numerical function (we will come to these later) "int". (Note that since the product of two integers is an integer itself, in the third and fourth statement it does not make any difference whether we use the "num" or the "int" type selector.) | Here the first statement will cause <code>#a</code> to be assigned the result of the floating point multiplication of 3 and 3.4, i.e. about 10.2. In the second statement, the same multiplication will be calculated, and only the result of this calculation, i.e. 10.2 (roughly...), will get truncated to an integer – this integer in turn being 10. Only the third and the fourth statement (both!) will cause the second factor, 3.4, to be broken down to an integer before multiplication. This usage, though, is strictly not a feature of the type selector-based assignment, but of the numerical function (we will come to these later) "int". (Note that since the product of two integers is an integer itself, in the third and fourth statement it does not make any difference whether we use the "num" or the "int" type selector.) | ||
Line 171: | Line 193: | ||
So in the third to sixth example, the type selectors are "<code>num</code>", "<code>int</code>", "<code>num</code>", and again "<code>num</code>", whereas the strings "<code>int(3.4)</code>", "<code>int(3*3.4)</code>", and "<code>int(3*int(3.4))</code>" are calls to the built-in type conversion function, "<code>int</code>". (If this sounds confusing for the moment, you should not worry: In practice, things are much easier, and it is not normally necessary to think these things over). | So in the third to sixth example, the type selectors are "<code>num</code>", "<code>int</code>", "<code>num</code>", and again "<code>num</code>", whereas the strings "<code>int(3.4)</code>", "<code>int(3*3.4)</code>", and "<code>int(3*int(3.4))</code>" are calls to the built-in type conversion function, "<code>int</code>". (If this sounds confusing for the moment, you should not worry: In practice, things are much easier, and it is not normally necessary to think these things over). | ||
====Function calls==== | |||
Syntactically, every built-in {{ | Syntactically, every built-in {{STx}} function, every macro, and every method of a user-defined class may be the source of an assignment. If this happens to be the case, the respective function or macro is executed, and its result is assigned to the destination variable. See it for yourself: | ||
#i := [[Programmer_Guide/Command_Reference/WORD|word]] 2 one two three four // #i will be "three" | |||
The built-in "<code>word</code>" function takes an integer index and a list of strings as its arguments. From this list, it selects and returns the string with the respective index. Hence, the expression "<code>word 2 one two three four</code>" will select, and return, the third word, „three". When part of an assignment as in the above example, this very string "three" will be assigned to the respective destination variable, in this case: to the local variable <code>#i</code>. | The built-in "<code>[[Programmer_Guide/Command_Reference/WORD|word]]</code>" function takes an integer index and a list of strings as its arguments. From this list, it selects and returns the string with the respective index. Hence, the expression "<code>word 2 one two three four</code>" will select, and return, the third word, „three". When part of an assignment as in the above example, this very string "three" will be assigned to the respective destination variable, in this case: to the local variable <code>#i</code>. | ||
The same holds true if the source of the assignment is the name of a class method. In this case, the respective method is being called, and its result (which may as well be the empty string) gets assigned to the destination variable. | The same holds true if the source of the assignment is the name of a class method. In this case, the respective method is being called, and its result (which may as well be the empty string) gets assigned to the destination variable. | ||
The issue of function calls will be dealt with in full detail in chapter | The issue of function calls will be dealt with in full detail in chapter [[#Calls_to_Macros_and_Built-In_Functions|Calls to {{Stx}} Macros and Built-in Functions]]. | ||
===Accessing Variables=== | |||
If and when you want to access the value of a variable (bluntly put: to read out its content), you need to put a dollar sign in front of the variable name. What happens internally is that, before actually executing a line from the macro file, {{STX}} replaces all occurrences of variable names that are prefixed by a dollar sign by the content of the respective variables. See for yourself: | If and when you want to access the value of a variable (bluntly put: to read out its content), you need to put a dollar sign in front of the variable name. What happens internally is that, before actually executing a line from the macro file, {{STX}} replaces all occurrences of variable names that are prefixed by a dollar sign by the content of the respective variables. See for yourself: | ||
#i := [[Programmer_Guide/Command_Reference/INT|int]] 7 | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'The current value of variable #i is $#i' | |||
#i := [[Programmer_Guide/Command_Reference/INT|int]] $#i + 1 // #i will be set to 8 | |||
#heading := [[Programmer_Guide/Command_Reference/SET|set]] 'This is page $#curpage out of $#totpage' | |||
< | The first line in this example will assign the value 7 to a local variable called <var>#i</var>. This is nothing new; note that we are using the type-selector "<code>[[Programmer_Guide/Command_Reference/INT|int]]</code>" to make sure the assigned value is interpreted as an integer, though in this case this is strictly redundant because 7 cannot help being an integer anyway. | ||
The second line will print out the text "The current value of variable #i is 7". This should not be surprising, for the name of a variable is only replaced by its content if preceded by a dollar sign. Hence, the first occurrence of "#i" – although we know that there is a variable called <var>#i</var> – does not get replaced by that variable's value. The second, though, does, because it is preceded by the dollar sign. {{Stx}} is one reliable piece of software, strictly and indiscriminately following the instructions laid out by its master. | |||
What the third line does is increase the value of the local variable <var>#i</var> by 1. What's more interesting is how it does so. First, {{STx}} replaces all dollar-prefixed variable names by the contents of the respective variables. Hence, {{STX}} will replace the string "$#i" by the current value of <var>#i</var> which, in our example, happens to be 7. This replacement will change the current statement from "#<code>i := [[Programmer_Guide/Command_Reference/INT|int]] $#i + 1</code>" to "<code>#i := [[Programmer_Guide/Command_Reference/INT|int]] 7 + 1</code>". Now this is one fine integer expression that in turn gets evaluated to 8, thereby causing 8 to be the string that is finally assigned to the destination variable. | |||
The | The fourth and last line demonstrates that substitution also works within string constants, and that it does so even if they are put under quotation marks (in our case, apostrophes). Some macro languages, e.g. the well-known UNIX shells, let certain kinds of quotation marks prevent substitution. Users familiar with such shells should bear in mind that {{Stx}} is a kind of its own. (Note that if you really want to suppress the special meaning of a character like the dollar sign, you may precede it with the {{STx}} escape character "<code>`</code>", the so-called back-tick.) | ||
The aspiring {{STX}} guru may find it instructive to consider the following example (anyone else will find no harm in completely skipping it): | |||
#var := [[Programmer_Guide/Command_Reference/SET|set]] 'one' | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '#var now containing "$#var"' | |||
#var := [[Programmer_Guide/Command_Reference/SET|set]] 'two' | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '#var now containing "$#var"' | |||
$#var := [[Programmer_Guide/Command_Reference/SET|set]] 'three' // N.B.: substitution will make this | |||
// "two := set 'three'" | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '#var still containing "$#var"' | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '...but there suddenly is a variable called two' | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '...and its value is "$two"' | |||
The first line is well familiar. It assigns the string "one" to a local variable called "<code>#var</code>". Consequently, the second line will print exactly the following string: | The first line is well familiar. It assigns the string "one" to a local variable called "<code>#var</code>". Consequently, the second line will print exactly the following string: | ||
#var now containing "one" | |||
The third line changes the value of <code>#var</code> to "two", hence the third line will print out the following string: | The third line changes the value of <code>#var</code> to "two", hence the third line will print out the following string: | ||
#var now containing "two" | |||
No surprises yet. But what will the fourth line do? Well, not to be surprised about the answer to this semi-rhetorical question, we must analyze the statement carefully. It reads "<code>$#var := set 'three'</code>" – did you notice that the assignment target is preceded by a dollar sign? Alas, this instructs {{Stx}} to replace the variable name by its content; but the content of variable "<code>#var</code>" is "two." Hence the statement gets, by substitution, altered to "<code>two := set 'three'</code>". This is a perfectly valid assignment statement, only that the target of the assignment is a shell-global variable called "<code>two</code>". So we assign the string "three" to a variable called "<code>two</code>". The next statements only illustrate this fact by printing out the respective values. | |||
===Read Functions=== | |||
<code>/ | The {{STx}} [[Programmer_Guide/Command_Reference/READ|<code>read</code> family of functions]] supply a means for parsing the contents of a string or a variable, that is for splitting them into several pieces, and for storing some or all of these pieces into one or more other variables (or even in the same variable). That being said, it should be noticed that everything is much easier than this description implicates. See for yourself: | ||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one two three four five' #a #b #c /Delete | |||
// #a is now "one", #b is now "two", #c is "three four five" | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'now a="$#a", b="$#b", and c="$#c".' | |||
The <code>readstr</code> command parses its first argument into (at most) as many blank-separated strings as there are variables. If the number of variables is higher than the number of available words, the remaining variables will either be cleared (if supplying the <code>/Delete</code> option), or they will be left untouched (if omitting the <code>/Delete</code> option). If, on the other hand, the number of variables is lower than the number of available words, the last variable gets all the remaining words. Note that if there is more than one whitespace character between two words, this will not do any harm. | The <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> command parses its first argument into (at most) as many blank-separated strings as there are variables. If the number of variables is higher than the number of available words, the remaining variables will either be cleared (if supplying the <code>/Delete</code> option), or they will be left untouched (if omitting the <code>/Delete</code> option). If, on the other hand, the number of variables is lower than the number of available words, the last variable gets all the remaining words. Note that if there is more than one whitespace character between two words, this will not do any harm. | ||
So what the above example does is parse the string "one two three four five" into three substrings (there are three variables supplied, < | So what the above example does is parse the string "<code>one two three four five</code>" into three substrings (there are three variables supplied, <var>#a</var>, <var>#b</var>, and <var>#c</var>). The first substring will be the first word, "one". The second substring will be the second word, "two". Since there are more words than variables, the third substring will catch all the rest, that is, the string "three four five". It's really simple, isn't it? | ||
There is one additional feature you may, or may not, find convenient. You may as well parse strings that are separated by exactly one character of your choice. If, for some reason, you prefer semicolons over blanks, you might have done the above example as follows: | There is one additional feature you may, or may not, find convenient. You may as well parse strings that are separated by exactly one character of your choice. If, for some reason, you prefer semicolons over blanks, you might have done the above example as follows: | ||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one;two;three;four;five' #a ';' #b ';' #c /Delete | |||
// #a is now "one", #b is now "two", #c is "three;four;five" | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'now a="$#a", b="$#b", and c="$#c".' | |||
The syntactical difference between those two examples is that the latter explicitly names the separator character between each pair of variables. The difference in semantics is that now there must be exactly one separator character between two words. So this variant of the <code>readstr</code> command empowers you to read empty words, two. Consider the following statement (we may abbreviate the <code>/Delete</code> option to <code>/D</code>, if we do not care for the reduced legibility): | The syntactical difference between those two examples is that the latter explicitly names the separator character between each pair of variables. The difference in semantics is that now there must be exactly one separator character between two words. So this variant of the <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> command empowers you to read empty words, two. Consider the following statement (we may abbreviate the <code>/Delete</code> option to <code>/D</code>, if we do not care for the reduced legibility): | ||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one;;two;three;four;five' #a ';' #b ';' #c /D | |||
// #a is "one", #b is empty, #c is " two;three;four;five" | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'now a="$#a", b="$#b", and c="$#c".' | |||
< | Looking awfully identical, doesn't it? Well, instead of one semicolon, there are now two semicolons between the first two words, "one" and "two". Believe it or not, this is making all the difference in the world: {{STx}} will consider these successive semicolons three separate words, the first being "one", the second one being empty, and the third one being, in general, "two" (in our case where there are only three variables supplied, the third word will get the rest of the string). So, with this example, variable <var>#a</var> will get the string "one", variable <var>#b</var> will be cleared, and variable <var>#c</var> will get the rest, that is the string, "two;three;four;five". Cool, isn't it? | ||
<code> | Note that if you omit the <code>/Delete</code> option, target variables corresponding to empty words will not be cleared, that is, they will keep whatever value that had before calling <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code>. That being said, you can easily foretell the results of the following statements: | ||
b := set 'old value before calling readstr' | |||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one;;two;three;four;five' #a ';' #b ';' #c | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'now a="$#a", b="$#b", and c="$#c".' | |||
As you rightly foretold, the <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> command in this example will not change the value of the second variable, <var>#b</var>, since the second word in this string is empty. | |||
<code> | Although technically a consequence of the above, it may not immediately be clear that the space character, too, may explicitly named as separating arguments – and that this does cause a difference to the default <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> behaviour. Look for yourself (and notice that there are two space characters between the "a" and the "b" in the first string constant): | ||
<code>readstr ' | [[Programmer_Guide/Command_Reference/READ|readstr]] 'a b c' #a #b #c /Delete</code> | ||
// #a is now "a", #b is "b", #c is "c" | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'a="$#a", b="$#b", c="$#c"' | |||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'a b c' #a ' ' #b ' ' #c /Delete | |||
// #a is now "a", #b is empty, #c is "b c" | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'a="$#a", b="$#b", c="$#c"' | |||
<code> | As you already know, there is one important difference between both <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> variants: When explicitly naming the separation character, {{STX}} considers consecutive occurrences of the separation character to separate empty strings. When not naming a separation, consecutive occurrences of whitespace are considered one single separator character, thereby causing no empty word to be read. So, in the above example, the first <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> command will read "a" into <var>#a</var>, "b" into <var>#b</var>, and "c" into <var>#c</var>, whereas the second <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> command will read "a" into <var>#a</var>, the empty word into <var>#b</var>, and the remaining string, "b c", into the last variable, <var>#c</var>. | ||
Of course the string argument supplied to <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> may even be the result of variable substitution. Less prosaically put, you might as well supply code like the following: | |||
#three := set 'THREE' | |||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one two $#three four' #a #b #c #d | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'now a="$#a", b="$#b", c="$#c", d="$#d"' | |||
Before executing the command, {{Stx}} will, as usual, look for any variable name prefixed with a dollar sign. If there happens to be any, they will be replaced by the contents of the respective variables. So in the above example, the text "<code>$#three</code>" will get replaced by the contents of variable <var>#three</var>, thereby causing the [[Programmer_Guide/Command_Reference/READ|{{Stx}} <code>readstr</code> command]] to actually be processed to be the following: | |||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one two THREE four' #a #b #c #d | |||
Only the strong survive the following example (anyone else will find no harm in skipping it). | Only the strong survive the following example (anyone else will find no harm in skipping it). | ||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'x y z' #a #b #c | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'now a="$#a", b="$#b", c="$#c"' | |||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one two three' $#a $#b $#c | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'gee, now a="$#a", b="$#b", c="$#c"' | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'but mysteriously, x="$x", y="$y", z="$z"' | |||
<code> | When listening very carefully, you might hear the above example speak for itself. If not, you will find the key in the third line that contains the second <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code> command. Did you notice the variables being prefixed by a dollar sign each? Now the dollar sign indicates {{STx}} that variable substitution is desired (and required) before the command is to be processed. So all {{STX}} does is replace the two strings "<code>$#a</code>", "<code>$#b</code>", and "<code>$#c</code>" by the contents of the respective variables. Since, at this stage, their contents are "x", "y", and "z", the command actually to get executed will look as follows: | ||
[[Programmer_Guide/Command_Reference/READ|readstr]] 'one two three' x y z | |||
Now this clearly is a request to parse the string "one two three" into the three shell-global variables "<var>x</var>", "<var>y</var>", and "<var>z</var>". | |||
When reading directly from one variable, you might prefer a variant of <code>[[Programmer_Guide/Command_Reference/READ|readstr]]</code>, the <code>[[Programmer_Guide/Command_Reference/READ|readvar]]</code> command. <code>[[Programmer_Guide/Command_Reference/READ|readvar]]</code> works similar to <code>readstr</code>, as the following example shows: | |||
#var := set 'one two three' | |||
[[Programmer_Guide/Command_Reference/READ|readvar]] #var #a #b #c /Delete | |||
<code> | The first argument to <code>[[Programmer_Guide/Command_Reference/READ|readvar]]</code> is one variable to read from. The remaining arguments are the variables where to store the words read. Otherwise that <code>[[Programmer_Guide/Command_Reference/READ|readvar]]</code> reads from a variable as opposed to reading from a literal string, there is no difference between <code>[[Programmer_Guide/Command_Reference/READ|readvar]]</code> and <code>readstr</code>. | ||
<code>readvar | You will undoubtedly ask why there is such a thing as a <code>[[Programmer_Guide/Command_Reference/READ|readvar]]</code> command. After all, all that | ||
[[Programmer_Guide/Command_Reference/READ|readvar]] #var #a #b #c | |||
does may as well be done using the following command: | does may as well be done using the following command: | ||
[[Programmer_Guide/Command_Reference/READ|readstr]] '$#var' #a #b #c | |||
You are, of course, right in principle. The difference between the two commands is that the latter depends upon variable substitution which introduces a second step when evaluating and executing the command. Hence, the former is simply faster. | You are, of course, right in principle. The difference between the two commands is that the latter depends upon variable substitution which introduces a second step when evaluating and executing the command. Hence, the former is simply faster. | ||
Note that there is a <code>readtable</code> function, too. We will come to that much later when dealing with the versatile (both are) {{ | Note that there is a <code>[[Programmer_Guide/Command_Reference/READ|readtable]]</code> function, too. We will come to that much later when dealing with the versatile (both are) {{Stx}} table feature. | ||
===Special Variables=== | |||
There are a number of reserved variables that serve special purposes. At this stage, it suffices to give a short overview of the most important special variables. | There are a number of reserved variables that serve special purposes. At this stage, it suffices to give a short overview of the most important special variables. | ||
Line 350: | Line 344: | ||
|Description | |Description | ||
|- | |- | ||
|< | |<var>rc</var> | ||
|After executing an | |After executing an {{STx}} statement or built-in function (not a user-defined macro!), this variable contains its numerical return code, 0 indicating success, and values different from 0 indicating different kinds of failure. "<var>rc</var>" gets set after each invocation of an {{STX}} statement or built-in function, meaning that an error code will get reset when executing the next statement. If you later need to map a numerical error code to a textual error message (e.g. for presenting it to the user), you may always use the [[Programmer_Guide/Command_Reference/EMSG|EMSG]] command. | ||
|- | |- | ||
|< | |<var>emsg</var> | ||
|This variable contains a textual description of the value of the "< | |This variable contains a textual description of the value of the "<var>rc</var>" variable. It, too, gets reset with each new {{Stx}} statement. If you have saved the value of <var>rc</var> e.g. in a local variable and you later need to map it to a textual error message (e.g. for presenting it to the user), you may always use the [[Programmer_Guide/Command_Reference/EMSG|EMSG]] command. (And don't confuse the <var>emsg</var> variable and the [[Programmer_Guide/Command_Reference/EMSG|EMSG]] command - they are not the same). | ||
|- | |- | ||
|< | |<var>#argc</var>, <var>#argv</var> | ||
|On macro invocation, "< | |On macro invocation, "<var>argc</var>" contains the number of arguments supplied to the respective macro, while "<var>argv</var>" contains the actual arguments (all of them). See below. | ||
|- | |- | ||
|< | |<var>result</var> | ||
|After returning from a user | |After returning from a user-defined macro call, the variable "<var>result</var>" contains the value returned by the respective macro, i.e. the value the macro supplied as an argument to the "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]]</code>" call. If the macro was left without returning an argument, "<var>result</var>" is empty (this is not considered an error – honestly, there is not much that is ever considered an error as far as {{STx}} is concerned). | ||
|- | |- | ||
|< | |<var>#read</var> | ||
|After using one of the | |After using one of the {{STX}} [[Programmer_Guide/Command_Reference/READ|READ]] commands (<code>read</code>, <code>readstr</code>, <code>readvar</code>, <code>readtab</code>), this variable contains the number of arguments actually read. | ||
|- | |- | ||
|< | |<var>#new</var> | ||
|When allocating an | |When allocating an {{Stx}} shell item, its item name will be stored in a variable called "<var>#new</var>". On the issue of shell items, please be patient until chapter [[Programmer_Guide/STx_Guru#Shell_Items:_An_Overview|Shell Items: An Overview]], or divert to the [[Programmer_Guide/Shell_Items|Shell Items]] chapter in the Reference Manual. | ||
|} | |} | ||
In general, even the reserved variable may be target of an assignment (this is sometimes used with the "< | In general, even the reserved variable may be target of an assignment (this is sometimes used with the "<var>#argv</var>" variable for implementing default macro arguments). You should not be surprised, though, that assigning a value to either "<var>rc</var>" or "<var>emsg</var>" will not have the desired outcome: Since the assignment statement is built-in {{STx}} statement itself, executing it will reset "<var>rc</var>" to 0 and "<var>emsg</var>" to the empty string, thereby indicating that the assignment statement itself was successful (which it was). | ||
==Control Structures== | ==Control Structures== | ||
Line 375: | Line 369: | ||
===Calls to Macros and Built-In Functions=== | ===Calls to Macros and Built-In Functions=== | ||
You might argue that we have been through that already, but, unfortunately, this is only part true: There is much more to passing an argument to a function than we have explored when calling the built-in "<code>word</code>" function, or the built-in "<code>writelog</code>" command. This chapter tells why and what. | You might argue that we have been through that already, but, unfortunately, this is only part true: There is much more to passing an argument to a function than we have explored when calling the built-in "<code>[[Programmer_Guide/Command_Reference/WORD|word]]</code>" function, or the built-in "<code>[[Programmer_Guide/Command_Reference/WRITELOG|writelog]]</code>" command. This chapter tells why and what. | ||
===Macros=== | ===Macros=== | ||
Line 381: | Line 375: | ||
Semantically, an {{STX}} macro is like a procedure or function of any procedural programming language of your choice. Syntactically, a macro starts with a macro definition surrounded by square brackets, e.g. the following line: | Semantically, an {{STX}} macro is like a procedure or function of any procedural programming language of your choice. Syntactically, a macro starts with a macro definition surrounded by square brackets, e.g. the following line: | ||
[Macro mymacro] | |||
There is no clear end to a macro. Execution continues until the control flow reaches an "<code>exit</code>" statement, until a new macro starts, or until the file comes to an end (whichever happens first) | There is no clear end to a macro. Execution continues until the control flow reaches an "<code>exit</code>" statement, until a new macro starts, or until the file comes to an end (whichever happens first). | ||
==== | ==== Calling a macro ==== | ||
For calling a macro, you simply type the macro name. In the above case, if {{Stx}} encounters a line starting whose first word is the string "<code>mymacro</code>", it will execute the like-named macro. {{STx}} supports recursive macro calls. | |||
There are a few commands providing different, not always cleaner ways of calling a macro, namely the [[Programmer_Guide/Command_Reference/MACRO|MACRO]], the [[Programmer_Guide/Command_Reference/MACROX|MACROX]] and the [[Programmer_Guide/Command_Reference/SHELL|SHELL]] command. Please do not use these commands unless there is very good reason to (there normally isn't). | |||
====Returning From a Macro==== | |||
Macro execution will stop when {{STX}} encounters the "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]]</code>" statement. Normally, control flow resumes at the line immediately following the statement that caused the macro to execute. You may supply the following optional arguments to an exit statement: | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] <var>level</var> <var>result</var> | |||
<code>exit 1 | When calling "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]]</code>" without any arguments or when calling "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]] 1</code>", control flow will, as said above, resume with the next statement after the macro call. | ||
<code>exit | In general, the first argument to "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]]</code>" indicates how many call levels to skip. When calling "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]] 2</code>", control flow will not resume at the line after the statement calling the macro, but at the line after that statement calling whichever macro was in turn calling our macro. When calling "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]] 3</code>", control flow will resume at the line after the statement calling the macro calling the macro calling our macro. When calling "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]] 4</code>", control flow will resume at the line after the statement calling the macro calling the macro calling the macro calling our macro, while on "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]] 5</code>" execution will resume with the line after the statement calling the macro calling the macro calling the macro calling the macro calling our macro, and so on. There is limited use to this feature, and you are at the safe side when, at least in the beginning, always using "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]] 1</code>". | ||
There are two special cases, one when supplying 0 for the exit level, the other when supplying a negative number. | |||
When supplying 0 for the exit level, i.e. when executing <code>[[Programmer_Guide/Command_Reference/EXIT|exit]] 0</code>, the executing shell will be terminated, effectively ending execution of the whole user script (both the running macro and all calling macros). | |||
When supplying a ''negative'' level argument (e.g. <code>[[Programmer_Guide/Command_Reference/EXIT|exit]] -2</code>), {{Stx}} will return from as many macro levels as needed to find a macro level where there is a non-empty local variable <var>#onexitall</var> defined. Since this is a bit complicated and not very clean, you probably should try not to use this feature. | |||
<code> | Of more interest than the first is the second argument to "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]]</code>". It is the result of the macro, that is the number or string the macro is to return. Like with assignment operations, this result may, and should, start with a type selector, one out of "<code>[[Programmer_Guide/Command_Reference/SET|set]]</code>", "<code>[[Programmer_Guide/Command_Reference/INT|int]]</code>", "<code>[[Programmer_Guide/Command_Reference/NUM|num]]</code>", and "<code>[[Programmer_Guide/Command_Reference/EVAL|eval]]</code>". Consider the following "<code>exit</code>" statements: | ||
[[Programmer_Guide/Command_Reference/EXIT|exit]] 1 set 'Hallo Welt' | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] 1 int 5/3 | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] 1 num 3/5 | |||
The first statement will exit the running macro and return the friendly string "Hello World". The second statement will exit the running macro and return whichever integer results on dividing 5 by 3. Finally, the third statement will exit the running macro and return the quotient of 3 and 5. | |||
There are several interchangeable ways for the caller to retrieve whatever value the macro has returned. First of all, the result is stored in the reserved shell variable "<code>result</code>", rendering the following code snipped a working example: | |||
greetings // call the "greetings" macro | |||
writelog '$result' // ...and print its results | |||
exit // ...and rest after a long day's work | |||
[Macro greetings] | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] 1 set 'Hello World' | |||
An even more elegant feature is using command substitution, a concept both known and feared from the UNIX shells: | An even more elegant feature is using command substitution, a concept both known and feared from the UNIX shells: | ||
#var := set '$(greetings)' | |||
writelog '$(greetings)' | |||
The first statement executes the macro "greetings" and assigns its result to a variable called "<code>var</code>". The second statement, too, calls the "greetings" macro, but directly pastes its result into a "<code>writelog</code>" statement. | The first statement executes the macro "greetings" and assigns its result to a variable called "<code>var</code>". The second statement, too, calls the "greetings" macro, but directly pastes its result into a "<code>writelog</code>" statement. | ||
Though command substitution is the most general way, a macro may be directly called, too – just like any built-in {{ | Though command substitution is the most general way, a macro may be directly called, too – just like any built-in {{STx}} function: | ||
#var := greetings | |||
The latter format greatly improves legibility and is therefore he recommended way of calling a user-defined macro. | The latter format greatly improves legibility and is therefore he recommended way of calling a user-defined macro. | ||
Line 437: | Line 436: | ||
There are, they say, many ways leading to Rome, and this surely holds true for retrieving and parsing the arguments a user-defined macro has been supplied with. The easiest way is disposing of the problem in the macro definition, i.e. by writing something like this: | There are, they say, many ways leading to Rome, and this surely holds true for retrieving and parsing the arguments a user-defined macro has been supplied with. The easiest way is disposing of the problem in the macro definition, i.e. by writing something like this: | ||
[Macro multiply #mand #mor] | |||
#prod := [[Programmer_Guide/Command_Reference/EVAL|eval]] $#mand * $#mor | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'multiplying $#mand by $#mor results in $#prod' | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] | |||
As you may have noticed, the macro name may be followed by one or more variable names. In this case the arguments supplied to the macro are being automatically parsed and stored to the like-named variables. We call this the implicit argument parsing feature. Hence, when calling "<code>multiply 7 9</code>", the starting macro will find its local variable <code>#mand</code> set to 7 and its local variable <code>#mor</code> set to 9. This will cause the above macro to print the message "multiplying 7 by 9 results in 63". If there are fewer arguments than variables, the remaining variables will be empty. If there are more arguments than variables, the whole remaining part of the arguments will be stored to the last variable. When you think it over, this is completely analogous to the <code>readstr</code> and <code>readvar</code> functions already known. | As you may have noticed, the macro name may be followed by one or more variable names. In this case the arguments supplied to the macro are being automatically parsed and stored to the like-named variables. We call this the implicit argument parsing feature. Hence, when calling "<code>multiply 7 9</code>", the starting macro will find its local variable <code>#mand</code> set to 7 and its local variable <code>#mor</code> set to 9. This will cause the above macro to print the message "multiplying 7 by 9 results in 63". If there are fewer arguments than variables, the remaining variables will be empty. If there are more arguments than variables, the whole remaining part of the arguments will be stored to the last variable. When you think it over, this is completely analogous to the <code>readstr</code> and <code>readvar</code> functions already known. | ||
Line 449: | Line 445: | ||
What's also analogous to <code>readstr</code> and <code>readvar</code> is the possibility explicitly to name a separator character of one's choice. See for yourself: | What's also analogous to <code>readstr</code> and <code>readvar</code> is the possibility explicitly to name a separator character of one's choice. See for yourself: | ||
[Macro multiply #mand';'#mor] | |||
#prod := [[Programmer_Guide/Command_Reference/EVAL|eval]] $#mand * $#mor | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'multiplying $#mand by $#mor results in $#prod' | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] | |||
While the first variant of the user-defined macro "multiply" will expect its arguments to be separated by an arbitrary number of whitespace characters, the second version will expect them to be separated by exactly one semicolon. This comes in handy when passing arguments that may contain, or contain, or must contain, or should contain, or would contain were it not for the fact that this is impossible with the default way of argument parsing, whitespace characters. | While the first variant of the user-defined macro "multiply" will expect its arguments to be separated by an arbitrary number of whitespace characters, the second version will expect them to be separated by exactly one semicolon. This comes in handy when passing arguments that may contain, or contain, or must contain, or should contain, or would contain were it not for the fact that this is impossible with the default way of argument parsing, whitespace characters. | ||
Line 461: | Line 454: | ||
A further and generally slightly more flexible, though a little more laborious way of parsing one's arguments is using the special variables "<code>#argc</code>" and "<code>#argv</code>". On start-up of a macro, these variables get set to the number of arguments and to the actual argument list, respectively. So the above example may be re-written for explicit argument parsing as follows: | A further and generally slightly more flexible, though a little more laborious way of parsing one's arguments is using the special variables "<code>#argc</code>" and "<code>#argv</code>". On start-up of a macro, these variables get set to the number of arguments and to the actual argument list, respectively. So the above example may be re-written for explicit argument parsing as follows: | ||
[Macro multiply] | |||
readvar #argv #mand #mor | |||
#prod := [[Programmer_Guide/Command_Reference/EVAL|eval]] $#mand * $#mor | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'multiplying $#mand by $#mor results in $#prod' | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] | |||
Why is this more flexible in general? – Because of the greater versatility of the <code>readvar</code> and <code>readstr</code> commands that even allow parsing argument lists built up from a variable number of arguments. We will come to that with the macro <code>countSTXProgrammers</code> of chapter 4.1.1.3 on page 1. | Why is this more flexible in general? – Because of the greater versatility of the <code>readvar</code> and <code>readstr</code> commands that even allow parsing argument lists built up from a variable number of arguments. We will come to that with the macro <code>countSTXProgrammers</code> of chapter 4.1.1.3 on page 1. | ||
Line 475: | Line 464: | ||
That being said, we may well continue with a real-world example. Consider e.g. the following macro: | That being said, we may well continue with a real-world example. Consider e.g. the following macro: | ||
[Macro declare #person #attribute] | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'I hereby declare $#person an $#attribute' | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] | |||
Now, if you would like to declare, say, Toni an {{STX}} hero, you might call the macro as follows: | Now, if you would like to declare, say, Toni an {{STX}} hero, you might call the macro as follows: | ||
declare Toni {{Stx}} hero | |||
This will work fine because the first word, "Toni", gets stored to the first variable, <code>#person</code>, and the rest of the arguments, "STX hero", gets stored to the second, and last, variable, <code>#attribute</code>. But what if you want to declare Christian Gottschall an {{ | This will work fine because the first word, "Toni", gets stored to the first variable, <code>#person</code>, and the rest of the arguments, "STX hero", gets stored to the second, and last, variable, <code>#attribute</code>. But what if you want to declare Christian Gottschall an {{STx}} beginner? See for yourself: | ||
declare Christian Gottschall {{STX}} beginner | |||
This will invariably lead to the first word, "Christian", being stored to <code>#person</code>, and the whole rest, "Gottschall {{ | This will invariably lead to the first word, "Christian", being stored to <code>#person</code>, and the whole rest, "Gottschall {{STx}} beginner", being stored to <code>#attribute</code>, hence causing Christian being declared a (sic!) "Gottschall {{STx}} beginner", which he surely isn't. | ||
You might be tempted to attacking this problem by using quotes, but in fact this is utterly impossible. You will find any conceivable combination (and even most unconceivable combinations) of quotes and escape characters to have a meaning very different from that intended. So the only general solution to this problem is using a different separator character: | You might be tempted to attacking this problem by using quotes, but in fact this is utterly impossible. You will find any conceivable combination (and even most unconceivable combinations) of quotes and escape characters to have a meaning very different from that intended. So the only general solution to this problem is using a different separator character: | ||
[Macro declare #person';'#attribute] | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'I hereby declare $#person an $#attribute' | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] | |||
Or: | Or: | ||
[Macro declare] | |||
readvar #argv #person ';' #attribute | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'I hereby declare $#person an $#attribute' | |||
[[Programmer_Guide/Command_Reference/EXIT|exit]] | |||
Now you may declare whomever you want whatever you please: | Now you may declare whomever you want whatever you please: | ||
declare Toni ; {{STX}} hero | |||
declare Christian Gottschall ; {{Stx}} beginner | |||
Though looking unusual from the standpoint of several other programming languages, it is good {{STx}} practice to quote the macro arguments, either as a whole, or argument-wise, or even (don't read this loudly!) on a per-word basis: | |||
declare 'Jonnie White ; {{STX}} guru' | |||
declare 'Jonnie White' ; 'STX guru' | |||
declare 'Jonnie' ' White' ; 'STX ' 'guru' | |||
In general, neither form of quotation will do any harm. Remember, though, that, with {{ | In general, neither form of quotation will do any harm. Remember, though, that, with {{Stx}}, writing several quoted strings will cause them to be implicitly concatenated with any intervening whitespace removed. Hence, although the above examples work as desired (note the quotations of the fourth statement containing space characters), the following won't: | ||
declare 'Jonnie' 'White' ; 'STX' 'guru' | |||
The latter statement will print out the text "I hereby declare JonnieWhite an STXguru": Although there is a whitespace character between each pair of quoted words, the quotations themselves do not contain any blank character, thereby causing the respective words to be concatenated without any intervening space. Compare this with the third statement in the previous example where there is a space character at the beginning of the second, and at the end of the third quoted word. (We've already had a few words on concatenation in chapter on page 1.) | The latter statement will print out the text "I hereby declare JonnieWhite an STXguru": Although there is a whitespace character between each pair of quoted words, the quotations themselves do not contain any blank character, thereby causing the respective words to be concatenated without any intervening space. Compare this with the third statement in the previous example where there is a space character at the beginning of the second, and at the end of the third quoted word. (We've already had a few words on concatenation in chapter on page 1.) | ||
Line 531: | Line 511: | ||
Note that when using the implicit argument parsing feature (or when using <code>readvar</code> with the default separation characters (whitespace), you need to be careful not to let the concatenation feature come in your way. Consider the following macro call: | Note that when using the implicit argument parsing feature (or when using <code>readvar</code> with the default separation characters (whitespace), you need to be careful not to let the concatenation feature come in your way. Consider the following macro call: | ||
usermacro 'one' 'two' 'three' | |||
Here, by string concatenation, the two strings "one", "two," and "three" will get concatenated to one single string, "onetwothree". Consequently, the macro will, whatever way of parsing it uses, get only one argument – the string "onetwothree." If you want the macro actually to be called with three arguments, you will need to use one of the following statements: | Here, by string concatenation, the two strings "one", "two," and "three" will get concatenated to one single string, "onetwothree". Consequently, the macro will, whatever way of parsing it uses, get only one argument – the string "onetwothree." If you want the macro actually to be called with three arguments, you will need to use one of the following statements: | ||
usermacro one two three | |||
usermacro 'one two three' | |||
usermacro 'one ' 'two ' ' three' | |||
It will work either way, additional whitespace characters never doing any harm. | It will work either way, additional whitespace characters never doing any harm. | ||
Line 549: | Line 527: | ||
The following example builds up several things we are at this stage familiar with (and several others we are not). You need not fully understand the macro at this stage, but you might notice several familiar issues. | The following example builds up several things we are at this stage familiar with (and several others we are not). You need not fully understand the macro at this stage, but you might notice several familiar issues. | ||
[Macro countSTXProgrammers] | |||
// read 0 both into #count and #totcount | |||
[[Programmer_Guide/Command_Reference/READ|readstr]] '0 0' #count #totcount // [[#ckremark1|(1)]] | |||
// the "forever" loop will never terminate by itself. | |||
// it will only stop at a "break" statement | |||
[[Programmer_Guide/Command_Reference/FOREVER|forever]] // [[#ckremark2|(2)]] | |||
// read the first person into #person, and the | |||
// remaining persons into #argv | |||
[[Programmer_Guide/Command_Reference/READ|readvar]] #argv #person ';' #argv /Delete // [[#ckremark3|(3)]] | |||
// return both the total number of persons counted and | |||
// the number of {{STx}} persons | |||
[[Programmer_Guide/Command_Reference/IF|if]] '$#read' == 0 then // [[#ckremark4|(4)]] | |||
[[{Programmer_Guide/Command_Reference/EXIT|exit]] 1 set '$#totcount;$#count' // [[#ckremark5|(5)]] | |||
end | |||
#totcount := int $#totcount+1 | |||
#index := keyword '$#person' Toni Jonnie Christian // [[#ckremark6|(6)]] | |||
[[Programmer_Guide/Command_Reference/IF|if]] '$#index' >= 0 then | |||
#count := int $#count+1 | |||
end | |||
end | |||
<code> | The <code>countSTXProgrammers</code> macro gets an arbitrary number of arguments separated by semicolons. Each argument is considered the name of a person. What the macro does, slightly arbitrarily, is count both the total number of persons supplied, and the number of {{STX}} programmers amongst them ({{Stx}} programmers being considered Toni, Jonnie, and Christian only). The most important statements are the following (see the respective numbers commented in the macro source): | ||
;{{anchor|ckremark1|(1)}}: This shows an idiomatic use of the <code>readstr</code> command for initializing multiple variables. Here the string "0 0" is parsed into two variables, <code>#count</code> and <code>#totcount</code>, effectively setting them both to zero. Of course you might as well use the two separate assignment statements "<code>#count := int 0</code>", and "<code>#totcount := int 0</code>". | |||
;{{anchor|ckremark2|(2)}}: The <code>forever</code> keyword starts kind of an eternal loop running until a <code>break</code> statement will be met. Both issues are dealt with in separate chapters ([[#Forever_and_Ever|Forever and Ever]] and [[#Ending_a_Loop_Prematurely:_Break_and_Continue|Ending a Loop Prematurely: Break and Continue]]). | |||
;{{anchor|ckremark3|(3)}}: The <code>readvar</code> statement will be utterly familiar to you. Note, though, that one of the destination variables is the same as the source variable. This is not a problem at all. Parsing will proceed normally, and the assignment will take place only after parsing has finished. So this <code>readvar</code> statement will result in the first word of <var>#argv</var> being parsed into <var>#person</var>. The remaining contents of <var>#argv</var> will, in turn, be parsed to%… <var>#argv</var> itself(!), effectively removing the first word from <var>#argv</var>. So with each pass of the loop, the next "first" name from the list will both be stored in <var>#person</var> and be removed from <var>#argv</var>. Isn't ''that'' cool? | |||
;{{anchor|ckremark4|(4)}}: This line introduces the <code>if</code> command. You need not worry about this command; it will be dealt with properly in chapterc 4.1.3. | |||
;{{anchor|ckremark5|(5)}}: The <code>exit</code> statement will return from the macro, and it will do so – due to the <code>if</code> statement – on the condition that <var>#read</var> is zero which is the case as soon as there are no more names to read. The first argument to the <code>exit</code> statement is 1, indicating that only the current macro is to end (and that execution shall continue with the calling macro). The second argument to the <code>exit</code> statement is the result of the macro, in this case: the string to return (indicated by the string assignment type selector, <code>set</code>). This string is built up from <var>#totcount</var>, a semicolon, and <var>#count</var>, and will hence contain the total number of persons and the number of {{STx}} programmers, separated by a semicolon. | |||
;{{anchor|ckremark6|(6)}}: This statement uses the built-in <code>keyword</code> function to investigate whether the current person is on the list of {{STX}} programmers. | |||
This shows an idiomatic use of the | |||
The | |||
The | |||
===Statements and Built-in Functions=== | ===Statements and Built-in Functions=== | ||
Arguments to {{ | Arguments to {{Stx}} statements and to built-in functions differ in several respects from macro calls: | ||
If (and only if) the statement or the function expects a numerical argument, you may supply a numerical expression instead of a number. This expression will be evaluated before executing the statement, or calling the built-in function, respectively. | # One quoted string is always considered one argument, even if it contains whitespace characters. | ||
# Hence there is no automatic string concatenation either. | |||
# If (and only if) the statement or the function expects a numerical argument, you may supply a numerical expression instead of a number. This expression will be evaluated before executing the statement, or calling the built-in function, respectively. | |||
The first two issues are easily demonstrated by the following example: | The first two issues are easily demonstrated by the following example: | ||
#a := [[Programmer_Guide/Command_Reference/WORD|word]] 2 'a b c' d 'e' f | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '#a="$#a"' // value of #a is "e" | |||
Both quoted strings will be considered one argument each, and no concatenation will take place. Hence, the built-in <code>[[Programmer_Guide/Command_Reference/WORD|word]]</code> function will return the string "e", "e" being the third argument (index 2). | |||
Both quoted strings will be considered one argument each, and no concatenation will take place. Hence, the built-in | |||
More surprising is the third issue, but it, too, can be demonstrated by a simple example: | More surprising is the third issue, but it, too, can be demonstrated by a simple example: | ||
#a := [[Programmer_Guide/Command_Reference/WORD|word]] 1+1 'a b c' d 'e' f | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '#a="$#a"' // value of #a is "e", too | |||
Since the "[[Programmer_Guide/Command_Reference/WORD|word]]" built-in expects its first argument to be a number, any numerical expression occurring at the respective position will be evaluated before calling "word". Hence, this example will return "e", too, because 1+1 equals 2. Compare this with the following statements: | |||
#a := [[Programmer_Guide/Command_Reference/WORD|word]] 1+1 'a b c' d '2+2' f | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '#a="$#a"' // value of #a is "2+2" | |||
Here what the third argument of the "[[Programmer_Guide/Command_Reference/WORD|word]]" built-in will return is the string "2+2". This string does not get evaluated, because "[[Programmer_Guide/Command_Reference/WORD|word]]" expects a string argument (and not a numerical argument) at this position. | |||
Here what the third argument the "word" built-in will return is the string "2+2". This string does not get evaluated, because "word" expects a string argument at this position. | |||
Note that if the numerical expression is to contain whitespace (or if it cannot be precluded that it does, e.g. when it is built up using variable substitution), you need to quote the whole expression. This quite naturally leads to the following rule of thumb: Always quote numerical expressions that are arguments to a statement or to a built-in function. See the following example: | Note that if the numerical expression is to contain whitespace (or if it cannot be precluded that it does, e.g. when it is built up using variable substitution), you need to quote the whole expression. This quite naturally leads to the following rule of thumb: Always quote numerical expressions that are arguments to a statement or to a built-in function. See the following example: | ||
#i := [[Programmer_Guide/Command_Reference/INT|int]] 3-2 | |||
#a := [[Programmer_Guide/Command_Reference/WORD|word]] '$#i+2' a b c d e | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] '#a="$#a"' // value of #a is "d" | |||
First of all, #i gets assigned the difference between 3 and 2, that is 1. String substitution will change the string "<code>$#i+2</code>" to "<code>1+2</code>", "<code>1</code>" being the value of < | First of all, <var>#i</var> gets assigned the difference between 3 and 2, that is 1. String substitution will change the string "<code>$#i+2</code>" to "<code>1+2</code>", "<code>1</code>" being the value of <var>#i</var>. Finally, due to the fact that the "<code>[[Programmer_Guide/Command_Reference/WORD|word]]</code>" function expects its first argument to be numerical, "<code>1+2</code>" will be evaluated as a numerical expression, resulting in 3. So the fourth string argument, "d", will be returned. | ||
===If Conditions are to be Tested=== | ===If Conditions are to be Tested=== | ||
If conditions are to be tested, the "if" statement comes in handy. It actually does so in two flavours: | If conditions are to be tested, the "<code>[[Programmer_Guide/Command_Reference/IF|if]]</code>" statement comes in handy. It actually does so in two flavours: | ||
[[Programmer_Guide/Command_Reference/IF|if]] [[Programmer_Guide/Concepts/Conditional_Expressions|condition]] statement | |||
And: | And: | ||
[[Programmer_Guide/Command_Reference/IF|if]] [[Programmer_Guide/Concepts/Conditional_Expressions|condition]] then | |||
many statements | |||
else | |||
many more statements | |||
end | |||
Before going into more details about what conditions look like, a few simple examples should make things more clear: | Before going into more details about what conditions look like, a few simple examples should make things more clear: | ||
[[Programmer_Guide/Command_Reference/IF|if]] '$#a' == <nowiki>''</nowiki> [[[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'Variable #a is empty.' | |||
[[Programmer_Guide/Command_Reference/IF|if]] '1+2' != 3 then | |||
// note the use of the {{STx}} escape character, the back-tick, | |||
// for building up a string that contains a quote character | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] Wohllebenstraße, we`'re having a problem. | |||
else | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] All systems nominal. | |||
end | |||
[[Programmer_Guide/Command_Reference/IF|if]] a > c then // character-wise comparison | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'Detecting an unusual character set' | |||
else | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'Detecting a usual character set' | |||
end | |||
[[Programmer_Guide/Command_Reference/IF|if]] '2+3' != '3+2' writelog 'STX addition is not commutative' | |||
<code>if ' | Note that with the first, simple form of the "<code>[[Programmer_Guide/Command_Reference/IF|if]]</code>" command, the statement to be executed in case of the condition being true must not be another "<code>if</code>" command. (In simpler wording: ''Simple [[Programmer_Guide/Command_Reference/IF|if]] statements must not be nested.'') Hence, it is not allowed to construct a statement like the following: | ||
// BAD! NOT ALLOWED! ERRONEOUS! DON'T DO IT! | |||
// COMBINING TWO SIMPLE "IF" STATEMENTS IS FORBIDDEN! | |||
if '$a' == '$b' if '$c' != '$d' writelog a and b are equal, but c and d are not | |||
<code> | If two or more "<code>[[Programmer_Guide/Command_Reference/IF|if]]</code>" statements are to be combined, use their complex form instead: | ||
if '$a' == '$b' then | |||
if '$c' != '$d' then | |||
[[Programmer_Guide/Command_Reference/WRITELOG|writelog]] 'a and b are equal, but c and d are not' | |||
end | |||
end | |||
===Conditions, Formally Revisited=== | ===Conditions, Formally Revisited=== | ||
Line 717: | Line 655: | ||
====Simple Comparison==== | ====Simple Comparison==== | ||
Conditions basically compare two entities for being equal, or for one of them being less than, or higher than, or not higher than, or not less than, or quite unlike the other. All this is done with the comparison operators "<code>=</code>", "<code> | Conditions basically compare two entities for being equal, or for one of them being less than, or higher than, or not higher than, or not less than, or quite unlike the other. All this is done with the comparison operators "<code>=</code>", "<code><</code>", "<code>></code>", "<code><=</code>", "<code>>=</code>", and "<code>!=</code>". The arguments to these operators may either be strings, in which case a Unicode-based string comparison will take place, or numerical expressions, in which case a numerical comparison will take place. If the argument to a comparison operator looks like a numerical expression, it will be treated as such, even if the programmer did not mean that to happen. | ||
Note that unless the argument to a comparison operator is quoted, it must be separated from the operator by at least one whitespace character. This means that while "<code>'$#a'=='1'</code>" (no whitespace, but quotes) and "<code>$#a == 1</code>" (blanks between the operator and its arguments) are well-formed expressions, "<code>$#a==1</code>" is not. (Yet another reason for always quoting everything, one might feel inclined to say.) | Note that unless the argument to a comparison operator is quoted, it must be separated from the operator by at least one whitespace character. This means that while "<code>'$#a'=='1'</code>" (no whitespace, but quotes) and "<code>$#a == 1</code>" (blanks between the operator and its arguments) are well-formed expressions, "<code>$#a==1</code>" is not. (Yet another reason for always quoting everything, one might feel inclined to say.) | ||
Line 723: | Line 661: | ||
See a number of simple examples: | See a number of simple examples: | ||
if '$#a' > '$#b*2' writelog '#a is more than twice as much as #b' | |||
if 'int($#a)' == '$#a' writelog '#a is an integer' | |||
if '$#a' == '0+$#a' then | |||
writelog '#a is a number.' | |||
else | |||
writelog '#a is not a number.' | |||
end | |||
All these examples show that numerical expressions occurring in comparisons will be evaluated. Hence, the first example will compare the value of <code>#a</code> with twice the value of <code>#b</code>. The second example will compare <code>#a</code> with its integer part. If they are the same, we know that <code>#a</code> must store an integer. | All these examples show that numerical expressions occurring in comparisons will be evaluated. Hence, the first example will compare the value of <code>#a</code> with twice the value of <code>#b</code>. The second example will compare <code>#a</code> with its integer part. If they are the same, we know that <code>#a</code> must store an integer. | ||
Line 747: | Line 681: | ||
====Pattern Matching==== | ====Pattern Matching==== | ||
Besides these comparison operators, there are matching operators, too. Matching may be considered a more powerful way of comparing strings that is able not only to find out if two strings are strictly identical, but also to find out whether they are similar in a way yet to be defined. With a matching operation, its first, left-hand side argument must be the string to match. Its second, right-hand side argument must be the pattern against which to match the string. | Besides these comparison operators, there are matching operators, too. Matching may be considered a more powerful way of comparing strings that is able not only to find out if two strings are strictly identical, but also to find out whether they are similar in a way yet to be defined. With a matching operation, its first, left-hand side argument must be the string to match. Its second, right-hand side argument must be the ''pattern'' against which to match the string. | ||
===== Wildcard Pattern Matching===== | |||
''Wildcard patterns'' are similar to strings with the exception that they may contain wildcards (hence the name). With {{Stx}}, these wildcards are the asterisk, "*", for matching a string of arbitrary length, and the question mark, "?", for matching exactly one character. | |||
There are the following wildcard matching operators: | |||
{| | ; Wildcard string matching: | ||
{| style="margin-left: 1.5em;" | |||
|- | |- | ||
|<code>=SI</code> | |<code>=SI</code> | ||
Line 762: | Line 699: | ||
|- | |- | ||
|<code>=SR</code> | |<code>=SR</code> | ||
|The string argument matches the pattern case | |The string argument matches the pattern case-sensitively. | ||
|- | |- | ||
|<code>!SR</code> | |<code>!SR</code> | ||
Line 768: | Line 705: | ||
|} | |} | ||
; Wildcard name matching: Name matching is similar to string matching, the difference being that the string to match is expected to be a well-formed {{STX}} name. If this is not the case, the match will fail, even if the pattern would string-match the string. More precisely, matching a non-name string with <code>=N</code> will always return false, whereas matching a non-name string with <code>!N</code> will always succeed, regardless of the pattern used. | |||
{| style="margin-left: 1.5em;" | |||
Name matching is similar to string matching, the difference being that the string to match is expected to be a well | |||
{| | |||
|- | |- | ||
|<code>=NI</code> | |<code>=NI</code> | ||
Line 789: | Line 723: | ||
Have a glance at the following examples: | Have a glance at the following examples: | ||
if 'hallo' =SI ha* writelog 'OK' // 1 | |||
if 'hallo' =SI *ha* writelog 'OK' // 2 | |||
if 'hallo' =SI h*o writelog 'OK' // 3 | |||
if 'hallx' !NI ha*o writelog 'OK' // 4 | |||
if 'test' !SI ha* writelog 'OK' // 5 | |||
if 'test' !SI *ha* writelog 'OK' // 6 | |||
if 'test' !SI h*o writelog 'OK' // 7 | |||
if 'h-o' !NI h*o writelog 'OK' // 8 | |||
if 'test' =SI t*t* writelog 'OK' // 9 | |||
Each of these comparisons should come out true, hence causing each statement to print the string "OK". Let's have a look at a few of these examples in detail: | Each of these comparisons should come out true, hence causing each statement to print the string "OK". Let's have a look at a few of these examples in detail: | ||
Line 815: | Line 737: | ||
The pattern, "<code>ha*</code>", will match any string starting with "<code>ha</code>" – "<code>hallo</code>" does. So "<code>=SI</code>" will guarantee a positive outcome. | The pattern, "<code>ha*</code>", will match any string starting with "<code>ha</code>" – "<code>hallo</code>" does. So "<code>=SI</code>" will guarantee a positive outcome. | ||
The pattern "<code>h*o</code>" will match any string whose first character is an "h" and whose last character is an "o". Now, "<code>hallx</code>" is no such string, but since we are using a negative comparison operator, "<code>!NI</code>", the whole thing comes out true since the pattern does not match. (Using "<code>!NI</code>" instead of "<code>!SI</code>" presupposes the string to be a well-formed {{ | The pattern "<code>h*o</code>" will match any string whose first character is an "h" and whose last character is an "o". Now, "<code>hallx</code>" is no such string, but since we are using a negative comparison operator, "<code>!NI</code>", the whole thing comes out true since the pattern does not match. (Using "<code>!NI</code>" instead of "<code>!SI</code>" presupposes the string to be a well-formed {{STx}} name which, by chance, "<code>hallx</code>" is. But compare statement 8.) | ||
The pattern "<code>h*o</code>" normally matches any string starting with "h" and ending with "o". That it does not match in this expression is because the "<code>NI</code>" operator presupposes its argument to be a well-formed {{ | The pattern "<code>h*o</code>" normally matches any string starting with "h" and ending with "o". That it does not match in this expression is because the "<code>NI</code>" operator presupposes its argument to be a well-formed {{Stx}} name. Now, "<code>h-o</code>" is no such name because it contains a minus sign, "<code>-</code>". So, the match fails by necessity, hence causing the negative expression "<code>!NI</code>" to become true. | ||
If you are shaken by statement 8 of the previous example, you might want to try out for yourself the following macro: | If you are shaken by statement 8 of the previous example, you might want to try out for yourself the following macro: | ||
if 'h-o' !NI h*o writelog 'OK1' // 1 | |||
if 'h-o' !SI h*o writelog 'OK2' // 2 | |||
if 'h-o' =NI h*o writelog 'OK3' // 3 | |||
if 'h-o' =SI h*o writelog 'OK4' // 4 | |||
What this will do is print out "OK1" and "OK4", but neither "OK2" nor "OK3". Do you see why? | |||
<code> | Well, first of all, the string "<code>h-o</code>" is no valid {{STX}} name due to its containing the minus sign, "<code>-</code>". So every name-based comparison will fail before even considering the pattern argument. So the third comparison, "<code>=NI</code>", will fail though "<code>h-o</code>" is a string starting with "h" and ending with "o". Consequently, the expression inverting that condition (using "<code>!NI</code>" instead of "<code>=NI</code>") will come out true, causing the first comparison to come out true. | ||
<code> | With the string-based comparison operators, things are different. Stringly put, the pattern "<code>h*o</code>" does match the string "h-o", that string actually starting with an "h" and ending with an "o". So the positive operator, "<code>=SI</code>", will come out true, while, consequently, its negative form will come out false. | ||
===== Regular-Expression Pattern Matching===== | |||
Aside from the simple wildcard-patterns, {{Stx}} also supports full POSIX regular-expression pattern matching, using the open-source [https://github.com/laurikari/tre/ TRE] library. Here is not the place for introducing POSIX regular-expressions, but if you are familiar with that concept, you will find the following {{STx}} matching operations useful: | |||
; Regular-expression string matching: | |||
{| style="margin-left: 1.5em;" | |||
|- | |||
|<code>=RSI</code> | |||
|The string argument does match the regular expression. Case will be ignored. | |||
|- | |||
|<code>!RSI</code> | |||
|The string does not match the regular expression, not even regardless of case. | |||
|- | |||
|<code>=RSR</code> | |||
|The string argument matches the regular expression case-sensitively. | |||
|- | |||
|<code>!RSR</code> | |||
|The string argument does not match the regular expression when respecting case. | |||
|} | |||
; Regular-expression name matching: This, too, is similar to regular-expression string matching, the difference being that the string to match is expected to be a well-formed {{STX}} name. If this is not the case, the match will fail, even if the regular expression would string-match the string. | |||
{| style="margin-left: 1.5em;" | |||
|- | |||
|<code>=RNI</code> | |||
|The name argument matches the regular expression ''ignoring case''. | |||
|- | |||
|<code>!RNI</code> | |||
|The name argument does not match the regular expression, not even regardless of case. | |||
|- | |||
|<code>=RNR</code> | |||
|The name argument matches the regular expression including case. | |||
|- | |||
|<code>!RNR</code> | |||
|The name argument does not match the regular expression when respecting case. | |||
|} | |||
====Building up Complex Expressions==== | ====Building up Complex Expressions==== | ||
You may combine more than one comparison either conjunctively, using the "<code>&&</code>" operator, or disjunctively, using the "<code> | You may combine more than one comparison either conjunctively, using the "<code>&&</code>" operator, or disjunctively, using the "<code>||</code>" operator. There is, though, neither precedence nor grouping, meaning that (a) complex expressions will be evaluated strictly from left to right and that (b) you cannot use brackets for grouping sub-expressions. While the former is not strictly an offence once one gets used to it (other programming languages like APL behave very similarly), the latter may sometimes cause some programmer inconvenience. We will try addressing this issue a wee bit later. | ||
Evaluation stops as soon as the outcome is clear, but with {{ | Evaluation stops as soon as the outcome is clear, but with {{Stx}}, this is only a performance issue (there is no such thing as built-in functions with side-effects, whereas command substitution, on the other hand, will always be done regardless of where the "<code>$(...)</code>" expression actually occurs). | ||
See the following examples: | See the following examples: | ||
if 1 > 2 || 3 > 1 writelog 'condition 1 is true' | |||
if 1 > 2 || 3 > 1 && 0 == 1 writelog 'condition 2 is true' | |||
if '$#a+1' < '$#b*3' | if '$#a+1' < '$#b*3' || '$#c-7' == '$#d' writelog 'condition 3 is true' | ||
Whereas the first statement should be fairly clear, fully to understand the second statement requires being aware of {{ | Whereas the first statement should be fairly clear, fully to understand the second statement requires being aware of {{STx}} strictly evaluating from left to right: For {{STX}}, the condition "<code>1 > 2 || 3 > 1 && 0 == 1</code>" will mean the same as, for a human reader, "(1 > 2 or 3 > 1) and 0 == 1" would. The "and" ("<code>&&</code>") does not – as it normally does in logic and mathematics – take precedence over the "or" ("<code>||</code>"). | ||
{| | {| | ||
|- | |- | ||
|What you type | |What you type | ||
|What | |What {{Stx}} does | ||
|- | |- | ||
|<code>$a | |<code>$a > $b && $b < $c || $c < $b && $b < $a</code> | ||
|<code>(($a > $b && $b < $c) | |<code>(($a > $b && $b < $c) || $c < $b) && $b < $a</code> | ||
|- | |- | ||
|<code>$a < $b && $a > 0 | |<code>$a < $b && $a > 0 || $b == -1</code> | ||
|<code>($a < $b && $a > 0) | |<code>($a < $b && $a > 0) || $b == -1</code> | ||
|- | |- | ||
|<code>$a < $b | |<code>$a < $b || $b > $a && $invert == 1</code> | ||
|<code>($a < $b | |<code>($a < $b || $b > $a) && $invert == 1</code> | ||
|- | |- | ||
|<code>$a < $b | |<code>$a < $b || ( $b > $a && $invert == 1 )</code> | ||
|Syntax error (brackets are not allowed!) | |Syntax error (brackets are not allowed!) | ||
|} | |} | ||
From the theoretical standpoint, the lack of grouping (i.e. brackets) allows for a much faster evaluation. It does, though, sometimes force the programmer either to combine several "<code>if</code>"-statements, or to alter his or her expression – that is why support for bracketing in logical expressions is under active consideration for {{ | From the theoretical standpoint, the lack of grouping (i.e. brackets) allows for a much faster evaluation. It does, though, sometimes force the programmer either to combine several "<code>if</code>"-statements, or to alter his or her expression – that is why support for bracketing in logical expressions is under active consideration for {{STx}} versions to come. For the time being, here are a few examples on how to rewrite complex expressions: | ||
{| | {| | ||
Line 876: | Line 831: | ||
|What you need to do to do what you mean to do | |What you need to do to do what you mean to do | ||
|- | |- | ||
|<code>if ($a > $b && $b < $c) | |<code>if ($a > $b && $b < $c) || ($c < $b && $b < $a) then</code><br><code> statements…</code><br><code>end</code> | ||
|<code>#flag := int 0</code><code>if $a > $b && $b < $c #flag := int 1</code><code>if $c < $b && $b < $a | |<code>#flag := int 0</code><br><code>if $a > $b && $b < $c #flag := int 1</code><br><code>if $c < $b && $b < $a || $#flag == 1 then</code><br><code> statements…</code><br><code>end</code> | ||
|- | |- | ||
|<code>if $a > 0 | |<code>if $a > 0 || ( $a < 0 && $b == int($b) ) then</code><br><code> statements…</code><br><code>end</code> | ||
|<code>if $a < 0 && $b == int($b) | |<code>if $a < 0 && $b == int($b) || $a > 0 then</code><br><code> statements…</code><br><code>end</code> | ||
|- | |- | ||
|<code>if ($a > $b) && $b < $c) | |<code>if ($a > $b) && $b < $c) ||</code><br><code>($c < $b && $b < $a) ||</code><br><code> ($b < $c && $c < $a) ||</code><br><code> ($b < $a && $a < $c) then</code><br><code> Statements…</code><br><code>end</code> | ||
|<code>#flag := int 0</code><code>if $a > $b && $b < $c #flag := int 1</code><code>if $c < $b && $b < $a #flag := int 1</code><code>if $b < $c && $c < $a #flag := int 1</code><code>if $b < $a && $a < $c | |<code>#flag := int 0</code><br><code>if $a > $b && $b < $c #flag := int 1</code><br><code>if $c < $b && $b < $a #flag := int 1</code><br><code>if $b < $c && $c < $a #flag := int 1</code><br><code>if $b < $a && $a < $c || $#flag == 1 then</code><br><code> statements…</code><br><code>end</code> | ||
|} | |} | ||
====Reference==== | |||
* [[Programmer_Guide/Concepts/Conditional_Expressions|Conditional Expressions]] Reference in the {{STX}} Programmers' Guide | |||
===While Loops Loop=== | ===While Loops Loop=== | ||
While loops model our everyday concept of repeating a task until it finally has the desired outcome. With {{ | While loops model our everyday concept of repeating a task until it finally has the desired outcome. With {{Stx}}, this is done as follows: | ||
[[Programmer_Guide/Command_Reference/WHILE|while]] [[Programmer_Guide/Concepts/Conditional_Expressions|expression]] | |||
many statements… | |||
end | |||
With this code, {{STx}} will first evaluate the [[Programmer_Guide/Concepts/Conditional_Expressions|expression]] supplied to the "<code>[[Programmer_Guide/Command_Reference/WHILE|while]]</code>" statement. If this expression happens to be true, all statements between the "<code>[[Programmer_Guide/Command_Reference/WHILE|while]]</code>" and the corresponding "<code>[[Programmer_Guide/Command_Reference/END|end]]</code>" statement are being executed once. Afterwards, {{STX}} again evaluates the very same [[Programmer_Guide/Concepts/Conditional_Expressions|expression]]. If it is still true, all the statements between the "<code>[[Programmer_Guide/Command_Reference/WHILE|while]]</code>" and the "<code>[[Programmer_Guide/Command_Reference/END|end]]</code>" statement are being executed once more. Then, again, the [[Programmer_Guide/Concepts/Conditional_Expressions|expression]] is evaluated… and so on, the effect being that all statements between the "<code>[[Programmer_Guide/Command_Reference/WHILE|while]]</code>" and the "<code>[[Programmer_Guide/Command_Reference/END|end]]</code>" statement are being executed "while" (i.e. as long as) the condition happens to be true. | |||
With this | |||
===For Loops to Be Organized=== | ===For Loops to Be Organized=== | ||
The "<code>for</code>" statement is a more organized loop statement. Its syntax is the following: | The "<code>[[Programmer_Guide/Command_Reference/FOR|for]]</code>" statement is a more organized loop statement. Its syntax is the following: | ||
[[Programmer_Guide/Command_Reference/FOR|for]] startcmd to [[Programmer_Guide/Concepts/Conditional_Expressions|whilecondition]] step changecmd | |||
many statements… | |||
end | |||
First of all, "<code>startcmd</code>" – some arbitrary command – gets executed. Next, the expression "<code>[[Programmer_Guide/Concepts/Conditional_Expressions|whilecondition]]</code>" gets evaluated. If it is false, the control flow will continue with the line immediately following the "end" statement. If, on the other hand, it is true, all the statements between "[[Programmer_Guide/Command_Reference/FOR|for]]" and "end" will be executed. Next, the "<code>changecmd</code>" – again, any command of your choice – will be executed. After that, things will start all over, with "condition" being evaluated again. | |||
First of all, "<code>startcmd</code>" – some arbitrary command – gets executed. Next, the expression "<code>whilecondition</code>" gets evaluated. If it is false, the control flow will continue with the line immediately following the "end" statement. If, on the other hand, it is true, all the statements between "for" and "end" will be executed. Next, the "<code>changecmd</code>" – again, any command of your choice – will be executed. After that, things will start all over, with "condition" being evaluated again. | |||
This is functionally equivalent to the following code snippet: | This is functionally equivalent to the following code snippet: | ||
startcmd | |||
while whilecondition | |||
many statements… | |||
changecmd | |||
end | |||
<code>for | So the <code>[[Programmer_Guide/Command_Reference/FOR|for]]</code> statement does not introduce any new functionality. What it does, though, is saving space and making this certain kind of loop more readable. | ||
<code> | Note that the <code>[[Programmer_Guide/Command_Reference/FOR|for]]</code> statement is modelled after the like-named statement of fashionable programming languages like C, C++, Java, or the author's all-time favourite AWK. From a semantical point of view, the keyword "<code>to</code>" may seem slightly misleading. Suffice it to say that there is a lesser-known (and these days virtually extinct) Klingon dialect where the word "t'o" means the same as the English imperative phrase "do this while" (at least we trust this statement to be hard to disprove). | ||
As an example, counting from 1 to 10, may serve the following "[[Programmer_Guide/Command_Reference/FOR|for]]" loop: | |||
< | [[Programmer_Guide/Command_Reference/FOR|for]] #a := int 1 to $#a <= 10 step #a := int $#a + 1 | ||
writelog 'at this stage, #a=$#a' | |||
end | |||
exit | |||
===Forever and Ever=== | ===Forever and Ever=== | ||
People often feel uneasy with the rapid change and the lack of durability in today's fast-paced environment. {{ | People often feel uneasy with the rapid change and the lack of durability in today's fast-paced environment. {{Stx}} addresses these issues with the "<code>[[Programmer_Guide/Command_Reference/FOREVER|forever]]</code>" statement that allows for enduring, long-term code execution. In fact, "<code>[[Programmer_Guide/Command_Reference/FOREVER|forever]]</code>" will start a flow of execution that never ends: | ||
[[Programmer_Guide/Command_Reference/FOREVER|forever]] | |||
eternal statements… | |||
end | |||
This kind of loop will cause the statements between "<code>[[Programmer_Guide/Command_Reference/FOREVER|forever]]</code>" and "<code>end</code>" to be processed for eternity (or until your computer breaks down, whichever happens first). | |||
This kind of loop will cause the statements between "forever" and "end" to be processed for eternity (or until your computer breaks down, whichever happens first). | |||
===Ending a Loop Prematurely: Break and Continue=== | ===Ending a Loop Prematurely: Break and Continue=== | ||
If {{ | If {{STx}} encounters the "<code>[[Programmer_Guide/Command_Reference/BREAK|break]]</code>" statement within the body of a loop, this very loop gets at once terminated. Control flow will resume at the statement immediately following the "<code>end</code>" statement at the end of the loop. | ||
The "continue" statement will have a different effect: Whenever {{STX}} encounters this kind of statement, it will directly pass to the next run of this loop, ignoring any statements between the " | The "<code>[[Programmer_Guide/Command_Reference/CONTINUE|continue]]</code>" statement will have a different effect: Whenever {{STX}} encounters this kind of statement, it will directly pass to the next run of this loop, ignoring any statements between the "<code>[[Programmer_Guide/Command_Reference/BREAK|break]]</code>" statement and the "<code>end</code>" statement at the end of this loop. | ||
Consider the following example: | Consider the following example: | ||
// print odd numbers less than 20 that are not a multiple of 3 | |||
#i := 0 | |||
forever | |||
#i := int $#i + 1 | |||
if '$#i' >= 20 break | |||
// note that "int" here is an in-place function truncating the | |||
// fractional part off a number | |||
if '$#i/3' == 'int($#i/3)' || '$#i/2' == 'int($#i/2)' continue | |||
writelog '$#i is neither a multiple of 3 nor even' | |||
end | |||
writelog 'Done.' | |||
exit | |||
<code>writelog | Though not at all an example in style, the above loop surely demonstrates the use of both the "<code>[[Programmer_Guide/Command_Reference/BREAK|break]]</code>" and the "<code>[[Programmer_Guide/Command_Reference/CONTINUE|continue]]</code>" statement. The basic idea of this loop is to run forever, with each pass increasing <code>#i</code> by one. Immediately after increasing <code>#i</code> is where the "<code>break</code>" statement finds its use: <code>#i</code> is being compared against twenty, and after having reached this upper limit, the whole loop will be terminated. Execution will continue with the first statement after the "end" statement of the respective loop. In our example, this first statement is the call to the built-in "<code>writelog</code>" statement printing out the string "Done." | ||
<code> | The next issue demonstrated here is the "<code>continue</code>" statement. After having ensured that <code>#i</code> has not reached its desired upper limit, <code>#i</code> is being checked for being either a multiple of 3, being a multiple of 2, or both. If so happens, {{Stx}} will "continue" the loop, i.e. omit its remaining part (in our case, the "<code>writelog</code>" statement), and continue with its first statement. | ||
Should the loop to be "continued" be a "<code>[[Programmer_Guide/Command_Reference/FOR|for]]</code>" or "<code>[[Programmer_Guide/Command_Reference/WHILE|while]]</code>" loop, the next run of the loop will begin normally, that is with testing the expression supplied to the "<code>[[Programmer_Guide/Command_Reference/FOR|for]]</code>" or "<code>[[Programmer_Guide/Command_Reference/WHILE|while]]</code>" statement. Should it prove false, the loop will terminate. In the case of a "<code>[[Programmer_Guide/Command_Reference/FOR|for]]</code>" loop, its "<code>changecmd</code>" (see above), too, will be executed normally, i.e., before checking the condition. | |||
===Statements Considered Useful: goto, gosub, and gosubx=== | |||
Each statement in an {{STx}} script may be labelled. The name of a label may consist of letters, digits, and underscores (though it must not start with a digit). The label must be immediately followed by a colon (that's necessary because otherwise there would be no way to discern between labels and macro calls). Note that labels are local to the macro they are part of. It is hence not possible to jump to a label that is part of a macro different from the executing macro (and it is possible to use labels of the same name in more than one macro). | |||
<code> | When using a label, you may at any time divert the flow of control to that label. This is done with the "<code>[[Programmer_Guide/Command_Reference/GOTO|goto]]</code>" command. Though it has come out of fashion recently, the "<code>[[Programmer_Guide/Command_Reference/GOTO|goto]]</code>" command allows for very efficient diversions in the control flow that may be really hard to follow. | ||
<code> | Contrary to most other programming languages, {{STX}} allows "<code>[[Programmer_Guide/Command_Reference/GOTO|goto]]</code>" to be supplied not only one, but even two labels. If this is the case, {{Stx}} first tries to go to the first label. Only if there happens to be no such label, {{STx}} falls back to the second label and tries to go there. If the second label is not present either, the "<code>[[Programmer_Guide/Command_Reference/GOTO|goto]]</code>" command will do nothing (it will leave an error code in the variable <code>rc</code>, though, but most programmers do not check for errors anyway). | ||
<code> | The following loop will count from 1 to 10 (compare this with the "<code>[[Programmer_Guide/Command_Reference/FOR|for]]</code>" loop of the last example in the previous chapter): | ||
#a := int 0 | |||
looping: | |||
if '$#a' >= 10 [[Programmer_Guide/Command_Reference/GOTO|goto]] endloop | |||
#a := int $#a + 1 | |||
writelog 'at this stage, a=$#a' | |||
[[Programmer_Guide/Command_Reference/GOTO|goto]] looping | |||
endloop: | |||
writelog 'let it be' | |||
exit | |||
<code> | A different issue is the "<code>[[Programmer_Guide/Command_Reference/GOSUB|gosub]]</code>" command. It is very similar to calling a macro: A new execution environment will be created, and control will resume at the label referred to by the "<code>[[Programmer_Guide/Command_Reference/GOSUB|gosub]]</code>" command. When (and if) the control flow reaches an "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]]</code>" command, this execution environment will be destroyed, and control will resume with the line immediately following the "<code>[[Programmer_Guide/Command_Reference/GOSUB|gosub]]</code>" command that caused all that to happen. Summarizing things up, the code called by the "<code>[[Programmer_Guide/Command_Reference/GOSUB|gosub]]</code>" statement will see all variables empty. Furthermore, any changes made during the statements invoked using "<code>[[Programmer_Guide/Command_Reference/GOSUB|gosub]]</code>" will get lost on returning. This is a very powerful feature. | ||
<code> | The "<code>[[Programmer_Guide/Command_Reference/GOSUBX|gosubx]]</code>" statement differs from the "<code>[[Programmer_Guide/Command_Reference/GOSUB|gosub]]</code>" command in that no separate execution environment is being created. Hence, the code will see all variables having their current value, and, conversely, any changes in variables done by the called code will remain in effect after returning. This, too, is always a powerful, sometimes a desired feature, and seldom easy to follow. | ||
<code> | Compare the effect of the "<code>[[Programmer_Guide/Command_Reference/GOSUB|gosub]]</code>" and the "<code>[[Programmer_Guide/Command_Reference/GOSUBX|gosubx]]</code>" commands in the following example: | ||
#a := 5 | |||
[[Programmer_Guide/Command_Reference/GOSUB|gosub]] subroutine | |||
writelog 'a after gosubbing subroutine: $#a (no change!)' | |||
[[Programmer_Guide/Command_Reference/GOSUBX|gosubx]] subroutine | |||
writelog 'a after gosubxing subroutine: $#a (changed!)' | |||
exit | |||
subroutine: | |||
// should #a be unset, numerical evaluation will fail | |||
#a := int $#a + 5 | |||
writelog 'value of a in subroutine: $#a' | |||
exit | |||
Note that we do not actually dare to recommend using any of these statements, and that we strongly encourage structured programming (using "<code>[[Programmer_Guide/Command_Reference/FOR|for]]</code>" and "<code>[[Programmer_Guide/Command_Reference/WHILE|while]]</code>" loops and breaking program code into macros). There may, however, be cases where the real programmer finds that he or she can do many interesting things with the help of the statements considered harmful since Dijkstra's famous letter. | |||
Note that we do not actually dare to recommend using any of these statements, and that we strongly encourage structured programming (using "<code>for</code>" and "<code>while</code>" loops and breaking program code into macros). There may, however, be cases where the real programmer finds that he or she can do many interesting things with the help of the statements considered harmful since Dijkstra's famous letter. | |||
==Shell Items: An Overview== | ==Shell Items: An Overview== | ||
Shell items are a kind of their own. Each shell item has a handle – actually kind of an internal name – that is not normally directly known to the programmer. Shell items may be created using the " | [[Programmer_Guide/Shell_Items|Shell items]] are a kind of their own. Each shell item has a handle – actually kind of an internal name – that is not normally directly known to the programmer. Shell items may be created using the "<code>[[Programmer_Guide/Command_Reference/NEW|new]]</code>" command, and they may – and should – be disposed of properly after use which is done with the "<code>[[Programmer_Guide/Command_Reference/DELETE|delete]]</code>" command. The handle to a shell item is normally stored in an {{STX}} variable. | ||
There are different sorts of shell items, the most important being file items (representing disk files), graph items (representing graphs visible to the user), dialog items (representing GUI dialogs, menus, windows, and the like), table items (representing in-memory databases, numerical vectors, and matrices), and instances of user-defined classes. We will come to each of them later. | There are different sorts of shell items, the most important being [[Programmer_Guide/Shell_Items/File|file items]] (representing disk files), [[Programmer_Guide/Shell_Items/Graph|graph items]] (representing graphs visible to the user), [[Programmer_Guide/Shell_Items/Dialog|dialog items]] (representing GUI dialogs, menus, windows, and the like), [[Programmer_Guide/Shell_Items/Table|table items]] (representing in-memory databases, numerical vectors, and matrices), [[Programmer_Guide/Shell_Items/Value|value items]] (providing things as diverse as timers, and fast matrix operations), [[Programmer_Guide/Shell_Items/Wave|wave items]] (for handling soundfiles), and instances of user-defined classes. We will come to each of them later (or at least to most of them). | ||
Beginners occasionally find it difficult to grasp the difference between a variable and a shell item. It is often helpful to remember the following properties: | Beginners occasionally find it difficult to grasp the difference between a variable and a shell item. It is often helpful to remember the following properties: | ||
Shell items are very complex objects like files or tables that normally consist of many parts (e.g. the rows and columns of a table, or the records of a file) most of which are themselves complex objects. | # Shell items are very complex objects like files or tables that normally consist of many parts (e.g. the rows and columns of a table, or the records of a file) most of which are themselves complex objects. | ||
# Variables, on the other hand, are very simple and dumb things: They store one string each – nothing complex in that. | |||
Variables, on the other hand, are very simple and dumb things: They store one string each – nothing complex in that. | # Shell items are referred to by their handle, variables by their name. | ||
# You usually store the handle of a shell item in a variable. | |||
Shell items are referred to by their handle, variables by their name. | # You usually do not access a shell item directly, but by using the variable storing its item handle. | ||
# For using a variable, i.e. accessing its contents, you need the dollar sign ("$"). Opposed to this there is no dollar sign with item handles, because there is no such thing as a simple content of an item. | |||
You usually store the handle of a shell item in a variable. | |||
You usually do not access a shell item directly, but by using the variable storing its item handle. | |||
For using a variable, i.e. accessing its contents, you need the dollar sign ("$"). Opposed to this there is no dollar sign with item handles, because there is no such thing as a simple content of an item. | |||
You may at any time request some action from an existing shell item. For doing so, you use the following command: | You may at any time request some action from an existing shell item. For doing so, you use the following command: | ||
< | set <var>itemhandle</var> actionrequest | ||
Here "< | Here "<var>itemhandle</var>" is the internal handle of the respective {{Stx}} item, and the exact nature of "<code>actionrequest</code>" depends on what kind of item you are dealing with; you may, e.g., request a table item to add or remove one or more rows, or a file item to write or to read data. | ||
Since requesting some item to do something is such common a task, you may abbreviate it by completely omitting the "<code>set</code>" command. This means that instead of "<code>set itemhandle actionrequest</code>", you may at any time type the following command: | Since requesting some item to do something is such common a task, you may abbreviate it by completely omitting the "<code>set</code>" command. This means that instead of "<code>set itemhandle actionrequest</code>", you may at any time type the following command: | ||
< | <var>itemhandle</var> actionrequest | ||
So there would actually be no need of even knowing about the "<code>set</code>" command, were it not for the fact that with the {{ | So there would actually be no need of even knowing about the "<code>set</code>" command, were it not for the fact that with the {{STx}} online help, you find the actions each kind of item supports under the keyword "<code>set itemtype</code>" (<code>[[Programmer_Guide/Shell_Items/Table/SET_TABLE|set table]]</code>, <code>[[Programmer_Guide/Shell_Items/Dialog/SET_DIALOG|SET DIALOG]]</code>, <code>[[Programmer_Guide/Shell_Items/Graph/SET_GRAPH|SET GRAPH]]</code>, and so on). | ||
Let's have a first simple example on how shell items are normally used: | Let's have a first simple example on how shell items are normally used: | ||
// create a table whose handle is called "myhandle" | |||
[[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] myhandle | |||
// set third line (first and second line will be created empty) | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|set]] myhandle 2 'third line' // explicitly using "set" | |||
// replace empty first line with a string | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|myhandle]] 0 'first line' // not using "set" works just as well | |||
// now replace the second line, too | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|myhandle]] 1 'second line' | |||
// show the table | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] myhandle ; Showing table "myhandle" | |||
// now, just for the fun of it, delete the second row – note that, again, | |||
// either using "set" or omitting it makes no change at all | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|myhandle]] myhandle 1 /Delete | |||
// finally, show the reduced table | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] myhandle; Showing table myhandle" after deleting its second row | |||
[[Programmer_Guide/Command_Reference/DELETE|delete]] myhandle // don't forget to delete the item after use | |||
In this example, we create a table item whose handle is the string "<code>myhandle</code>". Any further access to this item is done with this handle. Note that there is no dollar sign when using "<code>myhandle</code>", because "<code>myhandle</code>" is not a variable, but an item handle. If, by mistake, we would use the expression "<code>$myhandle</code>", {{STX}} would replace this expression by the contents of the shell-global variable "<code>myhandle</code>" – an empty string in case of there being no such variable, and surely an undesired result. Bluntly put: Item handles and variables reside in different namespaces, and you may well have like-named variables and item handles (of course this is not at all recommended because it may lead to some degree of confusion.) | In this example, we create a table item whose handle is the string "<code>myhandle</code>". Any further access to this item is done with this handle. Note that there is no dollar sign when using "<code>myhandle</code>", because "<code>myhandle</code>" is not a variable, but an item handle. If, by mistake, we would use the expression "<code>$myhandle</code>", {{STX}} would replace this expression by the contents of the shell-global variable "<code>myhandle</code>" – an empty string in case of there being no such variable, and surely an undesired result. Bluntly put: Item handles and variables reside in different namespaces, and you may well have like-named variables and item handles (of course this is not at all recommended because it may lead to some degree of confusion.) | ||
All that being said, it is | All that being said, it is, common practice not to choose a handle of one's own, but to let {{Stx}} do the choosing. In this case, the automatically chosen handle is normally stored in an {{STx}} variable: | ||
// create a table item with a handle of STX's choice (e.g. T#27) | |||
// (the asterisk instructs {{STX}} to choose a handle on its own). | |||
// the handle will be stored in a variable called "#tab" | |||
#tab := [[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] * | |||
// set third line (first and second line will be created empty) | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|$#tab]] 2 'third line' | |||
// replace empty first line with a string | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|$#tab]] 0 'first line' | |||
// now replace the second line, too | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|$#tab]] 1 'second line' | |||
// show the table | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#tab ; Showing table "#tab" | |||
// now, just for fun, delete the second row | |||
[[Programmer_Guide/Shell_Items/Table/SET_TABLE|$#tab]] 1 /Delete | |||
// finally, show the reduced table | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#tab ; Showing table "#tab" after deleting second row | |||
// old-style deletion | |||
// [[Programmer_Guide/Command_Reference/DELETE|delete]] $#tab // important: delete after use | |||
// #tab := set <nowiki>''</nowiki> // clean the variable | |||
// new-style deletion - deletes the shell item whose name is stored in a variable, | |||
// ''and'' clears the variable | |||
[[Programmer_Guide/Command_Reference/DELETE|delete]] /Var #tab // no dollar here! | |||
N.B.: Whenever actively allocating a shell item (normally with the "<code>[[Programmer_Guide/Command_Reference/NEW|new]]</code>" command), you should always delete it as soon as you no longer need it. If you don't, your macro(s) will consume an unnecessarily high amount of memory which may, for large macro applications, impair performance. There is no such thing as automatic garbage collection in {{Stx}}. | |||
N.B.: Whenever actively allocating a shell item (normally with the "<code>new</code>" command), you should delete it as soon as you no longer need it. If you don't, your macro(s) will consume an unnecessarily high amount of memory which may, for large macro applications, impair performance. | |||
===Shell Tables in Detail=== | ===Shell Tables in Detail=== | ||
There are three kinds of tables: (a) plain, or simple, tables, (b) parameter tables, and (c) extended tables. | There are three kinds of tables: (a) [[Programmer_Guide/Concepts/Simple_table|plain, or simple, tables]], (b) [[Programmer_Guide/Concepts/Parameter_table|parameter tables]], and (c) [[Programmer_Guide/Concepts/Extended_table|extended tables]] (I do not mention [[Programmer_Guide/Concepts/Directory_table|directory tables]] here, and I do not even mention that I do not mention [[Programmer_Guide/Concepts/Directory_table|directory tables]] here). [[Programmer_Guide/Concepts/Simple_table|plain, or simple, tables]] are easiest to handle, and they are most useful for storing lists of strings, each table entry being one such string. [[Programmer_Guide/Concepts/Extended_table|Extended tables]] are such that tables more versatile than them cannot be conceived: [[Programmer_Guide/Concepts/Extended_table|Extended tables]] tables may consist of an arbitrary number of columns each of which may be a string, an {{STx}} name, an integer, or a general number. Access to each column may be done, at the programmer's discretion, either by its index or by its symbolic name. [[Programmer_Guide/Concepts/Parameter_table|Parameter tables]], in turn, are a somewhat reduced variant of extended tables, in that they store numbers only. On the other hand, parameter tables do so most efficiently, both memory-wise and runtime-wise. This makes [[Programmer_Guide/Concepts/Parameter_table|parameter tables]] the preferred means of storing numerical vectors and matrices, and for doing numerical operations of the utmost complicated kinds. The following chapters will deal with each kind of table in detail. | ||
====Plain, or Simple, Tables==== | ====Plain, or Simple, Tables==== | ||
Plain tables are mainly useful for storing lists of strings. As always, these strings may be of arbitrary content, allowing for e.g. logically storing multiple values per row that are, in turn, retrieved by the " | [[Programmer_Guide/Concepts/Simple_table|Plain tables]] are mainly useful for storing lists of strings. As always, these strings may be of arbitrary content, allowing for e.g. logically storing multiple values per row that are, in turn, retrieved by the "<code>readstr</code>" or even the "<code>readtab</code>" command (the [[Programmer_Guide/Command_Reference/READ|READ family of commands]]). Be this as it may, with the availability of the more efficient extended tables for database-like structured data, and of parameter tables for structured numerical data like vectors and matrices, storing structured data in plain tables is no longer recommended. | ||
The "<code>new table *</code>" command creates a plain table item. The name of this table item will be stored in the reserved variable "<code>#new</code>". Furthermore, it will be the result of the "<code>new table *</code>" command. The asterisk, though strictly instructing {{STX}} to choose an internal name on its own, should be considered a part of the command and should not normally be altered. | The "<code>[[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] *</code>" command creates a plain table item. The name of this table item will be stored in the reserved variable "<code>#new</code>". Furthermore, it will be the result of the "<code>[[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] *</code>" command. The asterisk, though strictly instructing {{STX}} to choose an internal name on its own, should be considered a part of the command and should not normally be altered. | ||
Have a look at the following example using simple tables. The macro "<code>tablewrite</code>" will create a table, fill it with a hundred strings, delete every other row, and, finally, save the table to a file. The macro "<code>tableread</code>" will, in turn, re-read the table from the file. | Have a look at the following example using simple tables. The macro "<code>tablewrite</code>" will create a table, fill it with a hundred strings, delete every other row, and, finally, save the table to a file. The macro "<code>tableread</code>" will, in turn, re-read the table from the file. | ||
[Macro tablewrite] | |||
// create the table | |||
#tab := [[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] * | |||
// the "[[Programmer_Guide/Macro_Library/StdLib#UM_and_EM|em]]" command will, after displaying its second argument in an error | |||
// dialog, immediately terminate the macro, using its first argument as the | |||
// result of the macro | |||
if '$#tab[?]' != table [[Programmer_Guide/Macro_Library/StdLib#UM_and_EM|em]] '-1 ; Cannot create table' | |||
// fill the table with a hundred strings | |||
for #i := 1 to $#i <= 100 step #i := int $#i+1 | |||
$#tab $#i 'This is string $#i out of 100' | |||
end | |||
// now we prove of changing mind and delete every other row | |||
for #i := 98 to $#i >= 0 step #i := int $#i-2 | |||
$#tab $#i /Delete | |||
end | |||
// show the contents of the table | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#tab ; See Our Table | |||
// now save the table as a text file called "table_file.txt". | |||
// the file will be stored in the {{Stx}} script directory (whose path | |||
// is available in the shell-global variable "scriptDirectory") | |||
#fileName := set '$scriptDirectory\table_file.txt' | |||
// create a text file item – we will come to this in a later chapter, | |||
// here it is shown only as kind of cliffhanger | |||
#fileItem := [[Programmer_Guide/Shell_Items/File/NEW_FILE|new file]] * '$#fileName' /Text /Write | |||
if '$#fileItem' == * [[Programmer_Guide/Macro_Library/StdLib#UM_and_EM|em]] '-1 ; failed to open the file $#fileName' | |||
// save the table to the file | |||
[[Programmer_Guide/Shell_Items/File/SET_FILE#Text_Files:_SAVE|$#fileItem save]] $#tab | |||
if '$rc' > 0 [[Programmer_Guide/Macro_Library/StdLib#UM_and_EM|em]] '-1 ; failed to save table to file $#fileName' | |||
// dispose of the file item (not the file on disk!) and the table | |||
[[Programmer_Guide/Command_Reference/DELETE|delete]] $#fileItem $#tab | |||
exit 1 int 0 | |||
[Macro tableread] | |||
// create an empty table | |||
#tab := [[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] * | |||
#fileName := set '$scriptDirectory\table_file.txt' | |||
// now, create a file item for reading a text file | |||
#fileItem := [[Programmer_Guide/Shell_Items/File/NEW_FILE|new file]] * '$#fileName' /Text /Read | |||
if '$#fileItem' == * [[Programmer_Guide/Macro_Library/StdLib#UM_and_EM|em]] -1 ; failed to open the file $#fileName | |||
[[Programmer_Guide/Shell_Items/File/SET_FILE#Text_Files:_LOAD|$#fileItem load]] $#tab | |||
if '$rc' > 0 [[Programmer_Guide/Macro_Library/StdLib#UM_and_EM|em]] -1 ; failed to load table from file $#fileName | |||
[[Programmer_Guide/Command_Reference/DELETE|delete]] $#fileItem | |||
// show the loaded table | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#tab ; See Our Loaded Table | |||
[[Programmer_Guide/Command_Reference/DELETE|delete]] $#tab | |||
exit 1 int 0 | |||
The above macro already makes use of file items we need to leave for later discussion. As you can see, using them for saving and loading tables is fairly easy. You might even at this stage start saving and loading tables of your own without having looked up any details about file items. | The above macro already makes use of file items we need to leave for later discussion. As you can see, using them for saving and loading tables is fairly easy. You might even at this stage start saving and loading tables of your own without having looked up any details about file items. | ||
Line 1,264: | Line 1,135: | ||
====Parameter Tables==== | ====Parameter Tables==== | ||
Parameter tables store numerical data only. They are optimized for numerical throughput and, hence, are the means of choice when it comes to modelling vectors and matrices. | [[Programmer_Guide/Concepts/Parameter_table|Parameter tables]] store numerical data only. They are optimized for numerical throughput and, hence, are the means of choice when it comes to modelling vectors and matrices. | ||
Technically, parameter tables are a variant of the extended tables yet to be mentioned. The reason why we deal with parameter tables first is their optimal fitness for vector and matrix arithmetic. Note that extended tables, when they are all numerical, support exactly the same operations as parameter tables – in fact, numerical-only extended tables and parameter tables are completely interchangeable. Nevertheless, parameter tables are more efficient both runtime-wise and memory-wise. So, as a rule of thumb, always use parameter tables whenever doing vector and matrix arithmetic. | Technically, parameter tables are a variant of the extended tables yet to be mentioned. The reason why we deal with parameter tables first is their optimal fitness for vector and matrix arithmetic. Note that extended tables, when they are all numerical, support exactly the same operations as parameter tables – in fact, numerical-only extended tables and parameter tables are completely interchangeable. Nevertheless, parameter tables are more efficient both runtime-wise and memory-wise. So, as a rule of thumb, always use parameter tables whenever doing vector and matrix arithmetic. | ||
There are two ways of creating a parameter table | There are two ways of creating a parameter table: | ||
# explicitly, by using the [[Programmer_Guide/Shell_Items/Table/NEW_TABLE|NEW TABLE … /Parameter]] command; or | |||
# implicitly, by using the matrix or vector result of a call to [[Programmer_Guide/Command_Reference/EVAL|eval]]. | |||
Here's a simple and semantically quite meaningless example of working with parameter tables created both ways: | |||
// create a 20-element vector initialized with all zeroes | |||
#v0 := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]](20,0,0) // 20 elements, first value 0, increment 0 | |||
// create and fill a vector | |||
#v1 := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]](30,2,4) // 30 elements, first value 2, increment 4 | |||
// create and fill a second vector | |||
#v2 := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]](30,3,9) // 30 elements, first value 3, increment 9 | |||
// add up both vectors | |||
#v3 := eval $#v1 + $#v2 | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#v3 ; v3 // show the sum vector | |||
// create a vector with 30 elements each of which is the sine | |||
// of its index (not a useful example, but useful for an example) | |||
#v4 := [[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] * * number:x /Parameter | |||
for #i := 0 to $#i < 30 step #i := int $#i + 1 | |||
// store the sine of the index, #i, in #a | |||
#a := eval sin($#i) | |||
// set the #i-th value of the vector to the value of #a | |||
$#v4 $#i $#a | |||
end | |||
// transpose a matrix | |||
// create the matrix to transpose – here we must explicitly create the | |||
// matrix because we are going to assign values to its | |||
// individual columns | |||
#mat1 := [[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] * * number:x:4 /Parameter // 4 columns | |||
// fill each of its individual columns with some values | |||
$#mat1[*,0] := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]](30,10,4) // fill the first column | |||
$#mat1[*,1] := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]](30,2,1) // fill the second column | |||
$#mat1[*,2] := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]](30,0.1,1.2) // fill the third column | |||
$#mat1[*,3] := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]](30,1.2,1.2) // fill the fourth column | |||
// now transpose mat1 and store the transposed matrix to mat2 | |||
#mat2 := [[Programmer_Guide/Command_Reference/EVAL/trn|eval trn]]($#mat1) | |||
// show the transposed matrix | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#mat2 ; transposed matrix | |||
// now, multiply the matrix by the vector and see what happens | |||
#mat3 := eval $#mat2 * $#v3 | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#mat3 ; Transposed matrix multiplied by a vector | |||
// create and initialize a matrix in a single step | |||
#mat4 := eval init(30,10,1.2) // 30 rows, 10 columns, value 1.2 | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#mat4 ; A matrix containing all 1.2 | |||
// important: delete all items not to leak memory | |||
[[Programmer_Guide/Command_Reference/DELETE|delete /Var]] #v0 #v1 #v2 #v3 #v4 #mat1 #mat2 #mat3 #mat4 | |||
exit | |||
<!-- | |||
[Macro globales] | |||
a := set <nowiki>''</nowiki> | |||
#mat := [[Programmer_Guide/Shell_Items/Table/NEW_TABLE|new table]] * * number:x:5 /Parameter | |||
#i := int 0 | |||
forever | |||
#vec := usermacro | |||
if '$#vec[?]' != table break | |||
if '$#vec[]' != 5 then | |||
um An error has occurred. | |||
exit -1 | |||
end | |||
$#mat[$#i,*] := $#vec | |||
delete $#vec | |||
#i := int $#i + 1 | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#mat ; Matrix after step $#i | |||
end | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#mat ; Resulting Matrix, each vector is a row | |||
#trnmat := [[Programmer_Guide/Command_Reference/EVAL/trn|eval trn]]($#mat) | |||
[[Programmer_Guide/Macro_Library/StdLib#ShowItem|showitem]] $#trnmat ; Transposed Matrix | |||
delete $#mat $#trnmat | |||
exit | |||
[Macro usermacro] | |||
a := set '$a' a | |||
if '$a' == 'aaaaa' exit 1 set <nowiki>''</nowiki> | |||
#v1 := [[Programmer_Guide/Command_Reference/EVAL/fill|eval fill]]( 5, 1, 2 ) | |||
exit 1 set $#v1 | |||
for #i := 0 to $#i < 10 step #i := int $#i + 1 | |||
#j := int $#i+1 | |||
$#mat1[*,$#i] := $#v4 | |||
$#mat1[*,$#j] := $#v4 | |||
end | |||
Line 1,480: | Line 1,249: | ||
XXX | XXX | ||
#i := int 7 | |||
#i := set $#i * $#i | |||
#x := set word | |||
for #i := 0 to $#i < 10 step #i := int $#i + 1 | |||
#var := set '#v$#i' | |||
$#var := int 7 // set variable | |||
readvar $#var #value // get value | |||
end | |||
XXX | XXX | ||
--> | |||
====Extended Tables==== | <!-- ====Extended Tables==== | ||
XXX | XXX--> | ||
===Shell Items: Managing Memory and Avoiding Waste=== | <!-- ===Shell Items: Managing Memory and Avoiding Waste=== | ||
Although allocating shell items and disposing of them is very easy, there are a few fine points to consider. | Although allocating shell items and disposing of them is very easy, there are a few fine points to consider. | ||
Line 1,513: | Line 1,282: | ||
XXX | XXX | ||
--> | |||
==Pitfalls== | == {{STx|caps}} Pitfalls == | ||
When invocating a macro for the second time, third time, or generally the n-th time, n being any finite positive integer, it will find its namespace empty. This is what you would expect from a full-featured programming language, but it is not what you get from many macro languages inferior to {{STX}}, or even from classical languages like the venerable FORTRAN, Basic, or – God forbid – COBOL. | When invocating a macro for the second time, third time, or generally the n-th time, n being any finite positive integer, it will find its namespace empty. This is what you would expect from a full-featured programming language, but it is not what you get from many macro languages inferior to {{STX}}, or even from classical languages like the venerable FORTRAN, Basic, or – God forbid – COBOL. | ||
Line 1,522: | Line 1,292: | ||
This consistently holds true even when invocating a macro recursively. Its new instance will find all its variables empty, while the calling instance will find its local variables untouched by the macro call. | This consistently holds true even when invocating a macro recursively. Its new instance will find all its variables empty, while the calling instance will find its local variables untouched by the macro call. | ||
Hence, local variables are no means at all for passing information to and from a macro. For passing data to a macro, you should use macro arguments. For a macro to return results, you should use the return mechanism (using the verbose form of the "exit" command). Technically, another means of passing data to and from a macro is using shell-global (or even global) variables, but in the era of structured and modular programming, this is considered harmful. | Hence, local variables are no means at all for passing information to and from a macro. For passing data to a macro, you should use macro arguments. For a macro to return results, you should use the return mechanism (using the verbose form of the "<code>[[Programmer_Guide/Command_Reference/EXIT|exit]]</code>" command). Technically, another means of passing data to and from a macro is using shell-global (or even global) variables, but in the era of structured and modular programming, this is considered harmful to one's career. | ||
Using one and the same variable name in different scopes is technically valid. You may, e.g., have a global variable called <code>@i</code>, a shell-global variable <code>i</code>, and a local variable called <code>#i</code>. Allow for it to be misleading and a source of programmer error, though. | Using one and the same variable name in different scopes is technically valid. You may, e.g., have a global variable called <code>@i</code>, a shell-global variable <code>i</code>, and a local variable called <code>#i</code>. Allow for it to be misleading and a source of programmer error, though. | ||
Line 1,528: | Line 1,298: | ||
Misspelled variable names tend to be the cause for great confusion. What makes it worse is that there is no way of automatically detecting misspelled variable names: Since variables need not be explicitly declared nor initialized, any syntactically well-formed variable name will be valid at any stage – this is an issue all programming languages without explicit variable declarations share, among them illustrious names like FORTRAN, Basic, or PROLOG. There is no real remedy for this issue other than placing debug statements in the macro source that occasionally print out the values of some important variables, and to check the printed values against one's expectations. | Misspelled variable names tend to be the cause for great confusion. What makes it worse is that there is no way of automatically detecting misspelled variable names: Since variables need not be explicitly declared nor initialized, any syntactically well-formed variable name will be valid at any stage – this is an issue all programming languages without explicit variable declarations share, among them illustrious names like FORTRAN, Basic, or PROLOG. There is no real remedy for this issue other than placing debug statements in the macro source that occasionally print out the values of some important variables, and to check the printed values against one's expectations. | ||
More than the occasional pitfall results from the fact that the expression in an assignment is a rather complex issue. Though this expression is, in last consequence, a string (remember that with {{ | More than the occasional pitfall results from the fact that the expression in an assignment is a rather complex issue. Though this expression is, in last consequence, a string (remember that with {{Stx}}, all variables are being created equal, i.e. as string variables), the way this string is computed may be a surprisingly complex, heart-warmingly powerful, and sometimes even error-prone affair, bearing the potential for unexpected side-effects. That being said, preventing most kinds of trouble is easy by following two rules of thumb: (a) put everything under apostrophes; and (b) use the appropriate expression type selector when directly assigning a value (as opposed to the result of a function call or a built-in function). So, instead of writing "<code>name := hugo</code>", always type "<code>name := set 'hugo'</code>". | ||
Users, especially those familiar with other programming languages, but not with macro languages like the UNIX shells, often confuse when to prefix the name of a variable with the dollar sign and when not to. The rule is easy: If you want to mention a variable, you do not use the dollar sign. If you want to use a variable, that is, if you want its occurrence be replaced by its actual contents, you do use the dollar sign. Compiled to a rule of thumb: If the variable is the target of an assignment, you do not use a dollar sign. If the variable is the source of an assignment, or if it is the argument to a function or macro call, you do use the dollar sign. | Users, especially those familiar with other programming languages, but not with macro languages like the UNIX shells, often confuse when to prefix the name of a variable with the dollar sign and when not to. The rule is easy: If you want to ''mention'' a variable, you do not use the dollar sign. If you want to ''use'' a variable, that is, if you want its occurrence be replaced by its actual contents, you do use the dollar sign. Compiled to a rule of thumb: If the variable is the target of an assignment, you do not use a dollar sign. If the variable is the source of an assignment, or if it is the argument to a function or macro call, you do use the dollar sign. | ||
When using string constants, you should generally use quotation, i.e., place them between single quotes. | When using string constants, you should generally use quotation, i.e., place them between single quotes. | ||
Where two or more quoted string constants meet (but see the below exception), they will get concatenated without inserting any whitespace characters. This is a standard feature with many programming languages, especially macro languages, but it is sometimes met with surprise. So please be aware that if you want two string constants to be separated by a blank, you will either need to quote the whole string, or to add the whitespace character at the beginning or at the end of the respective string literal. Hence, the statement "<code>#var := set 'abc' 'def'</code>" will set #var to "abcdef", whereas each of the statements "<code>#var := set 'abc ' 'def'</code>", "<code>#var := set 'abc' ' def'</code>", and "<code>#var := set 'abc def'</code>" will set #var to "abc def". | Where two or more quoted string constants meet (but see the below exception), they will get concatenated without inserting any whitespace characters. This is a standard feature with many programming languages, especially macro languages, but it is sometimes met with surprise. So please be aware that if you want two string constants to be separated by a blank, you will either need to quote the whole string, or to add the whitespace character at the beginning or at the end of the respective string literal. Hence, the statement "<code>#var := set 'abc' 'def'</code>" will set <var>#var</var> to "abcdef", whereas each of the statements "<code>#var := set 'abc ' 'def'</code>", "<code>#var := set 'abc' ' def'</code>", and "<code>#var := set 'abc def'</code>" will set <var>#var</var> to "abc def". | ||
If a quoted string constant is an argument to a statement or to a built-in function (i.e. not to a user macro), there will be no string concatenation. Hence, each quoted string and each unquoted string is exactly one argument to the function. | If a quoted string constant is an argument to a statement or to a built-in function (i.e. not to a user macro), there will be no string concatenation. Hence, each quoted string and each unquoted string is exactly one argument to the function. | ||
Line 1,540: | Line 1,310: | ||
If a statement or a built-in function expects a numerical argument, this argument may either be a numerical constant or a numerical expression. Either way, it is more secure to quote even a numerical argument (don't quote me on that, though). | If a statement or a built-in function expects a numerical argument, this argument may either be a numerical constant or a numerical expression. Either way, it is more secure to quote even a numerical argument (don't quote me on that, though). | ||
It is easy to mix up the reserved variables "rc" and "result". The former ("rc") contains the numerical return code of the {{ | It is easy to mix up the reserved variables "<var>rc</var>" and "<var>result</var>". The former ("<var>rc</var>") contains the numerical return code of the {{STx}} statement or built-in function most-recently executed (or the reason code why some {{STX}} statement could not be executed). The latter ("<var>result</var>") contains whatever information the most recently called user-defined macro (or class method) chose to return. | ||
Unlike most other programming languages and many macro languages, {{ | Unlike most other programming languages and many macro languages, {{Stx}} evaluates logical expressions strictly from left to right. Furthermore, it does not support grouping (i.e. using brackets). Although this is not a bad thing in general, it may be cause for error and misunderstanding. When writing down a logical expression, be aware that no rules of precedence will apply. If in doubt, use a second (third, fourth…) "<code>[[Programmer_Guide/Command_Reference/IF|if]]</code>" statement within the scope of each other. | ||
Note that whitespace characters may at any time freely be interspersed within a number. This means that e.g. the string "<code>123 456</code>", although containing a blank character, will be interpreted as the number "123456". This is not a bad thing in itself, but you should be aware of this fact. | Note that whitespace characters may at any time freely be interspersed within a number. This means that e.g. the string "<code>123 456</code>", although containing a blank character, will be interpreted as the number "123456". This is not a bad thing in itself, but you should be aware of this fact. | ||
Line 1,548: | Line 1,318: | ||
Never forget to free an object (both shell-items and user-defined classes) after use. If you don't, and automatic cleanup will only be done at the termination of the running shell which, when using a complex set of macros, may be very late. | Never forget to free an object (both shell-items and user-defined classes) after use. If you don't, and automatic cleanup will only be done at the termination of the running shell which, when using a complex set of macros, may be very late. | ||
Be careful not to mix up internal {{ | Be careful not to mix up internal {{STx}} item handles on the one hand and variable names on the other hand. Remember that accessing an {{STX}} item by its name means just using this name – there is no dollar sign involved. If, on the other hand, your item handle is stored in an {{Stx}} variable, you will need to refer to the contents of this variable, i.e. use this variable – here there is a need for using the dollar sign. |
Latest revision as of 09:43, 1 October 2019
The following text is based on an STx workshop of 2005, which took place in the Acoustics Research Institute (ARI), and which was described as "our weekly time-wasting meeting" in a colleague's blog. It explains the internal structure and the features of the STx 3.7 release, and has been carefully adapted to the current STx version (4.0) in 2014.
Contents
- 1 Whom this is for
- 2 How to write an STx Script: The Easy Part
- 3 Constants
- 4 Variables
- 5 Control Structures
- 5.1 Calls to Macros and Built-In Functions
- 5.2 Macros
- 5.3 Statements and Built-in Functions
- 5.4 If Conditions are to be Tested
- 5.5 Conditions, Formally Revisited
- 5.6 While Loops Loop
- 5.7 For Loops to Be Organized
- 5.8 Forever and Ever
- 5.9 Ending a Loop Prematurely: Break and Continue
- 5.10 Statements Considered Useful: goto, gosub, and gosubx
- 6 Shell Items: An Overview
- 7 STx Pitfalls
Whom this is for
This manual is a general introduction to the programming language that is part of the STx environment. Reading this manual will allow you to implement procedural functions and then to proceed to object-oriented classes for a wide range of tasks, with a natural focus on numerical applications, sound processing, and visualization. As a reader of this manual, you should be slightly familiar with programming in general, and it would be great if you were an avid user of STx (on the other hand, being faintly familiar with starting up STx, loading sound files, and maybe even starting the spectrogram function should suffice).
This manual is intended for reading from start to end (not necessarily without interruptions). It is not a reference manual (there is such a thing, too: the STx Quick Reference), meaning that it will abstract from, you might even bluntly say: omit, many a detail in order not to depress the reader with a seemingly abundant amount of material. Instead, a careful selection has been taken on what is, and what is not, necessary for achieving common goals. Of course this selectivity (or, if you prefer, these omissions) try hard not to give any false impressions of what can, and what can't, be done with STx (and how). We are well aware that writing such an introductory programmer's (or, as we hope, programmers') manual is a slippery slope, and we hope for the reader's (or, as we hope, the readers') pardon if the depth covering each topic is too shallow, or to deep, or both (we are not quite sure whether the latter is logically possible, but one never knows). We appreciate any comments or criticism, both on this manual and on STx.
If you want to dig deeper into any specific topic, or if you are looking for information on a specific issue, we recommend having a look either at the online help of STx (just start STx and select the "Help" menu, or press the "F1" key), or, even better, at the STx Programmer Guide. Both are a vast and ever-growing collection of deep, sometimes even exhausting wisdom. They are, in fact, the official reference manual.
When new to STx programming, we recommend first reading this manual (you might skip paragraphs you find particularly uninteresting), probably trying out all the examples by yourself, preferably even modifying and improving them. Afterwards, you will have a basis firm enough for using the STx Programmer Guide, or even the STx Quick Reference, as a reference manual. And you will surely have a sound basis for solving all your future programming tasks with STx, disposing of the need for any other programming languages or environments.
How to write an STx Script: The Easy Part
This chapter deals with quite a basic fact, answering the first question that may come to your mind: How can I create and edit a script, and what need I do to submit it to STx for execution?
Well, the first part is easy: Just use a text editor of your choosing. Since Windows does not come equipped with the editor of your choosing, you should probably get it from www.vim.org, unless, of course, you already did so.
There is a faint convention to use the filename extension ".STS" for STx scripts, but everything else will really work equally well. So: just start editing. In its simplest case, your script will contain only one macro. The start of the macro is indicated by macro header, and it should end with an EXIT
statement:
[macro TowersOfHanoi] um 'Rapid prototyping of the towers of Hanoi problem. Program logic not yet present.' exit
There is, of course, a more formal definition of what an STx script is built up from, but at this stage you should probably not bother with all these other thingies.
After crafting your STx script, you still have STx to notice its presence, and to execute it. The simplest way for working on a script is pushing the large button labelled "Script file" (on the top of the main STx window, labelled (and called) "Workspace". After pressing this button, a file dialog titled "Select script file" will open, allowing you to, well, select your shiny new script file.
To the right of the "Script file" area, there is another area labelled "Macro". If your script file contains more than one macro function, this drop-down thingy will allow you to select which macro to execute.
There is a third field, labelled "Arguments", where you can supply arguments for your macro. And there are two awesome checkboxes labelled "Debug" and "Console", respectively. When checking "Debug", STx will start your amazing macro program in the gorgeous STx macro debugger. When checking "Console", your macro will run in the interactive STx console (an interactive version of the STx command interpreter). When checking both, either may happen, or both.
The awesome buttons, fields and checkboxes mentioned here build up what we call the gorgeous Script Controller. You will not find much more information in Script Controller.
Note: If you do not see any of the fields mentioned here (bluntly put: if you do not see the Script Controller at all), it is likely that it is not there. In this case, try using the Workspace menu item "Scripts > Show Scripts" to unhide it.
Constants
Our STx does not strictly discern between string constants and numerical constants. Normally each constant is considered a string, there only being exceptions dependent on the context where the constant occurs. If, for example, a constant occurs as part of a numerical expression, STx tries to get its numerical value.
Generally, you may, but you need not, put single quotes around constants. With a few exceptions depending on context, this will not change the way STx handles the constant. So each of the following strings is a valid STx constant:
string1 'string1' -12.34 '-12.34'
Regardless of the presence or absence of single quotes, both the first and the second argument will be considered string constants. Also regardless of the presence or absence of single quotes, both the third and the fourth constant will be considered numerical constants when occurring in a numerical context, or string constants when occurring in a string context.
If a string constant contains whitespace characters, it depends on the context whether STx considers it one string constant or more than one string constant. If you want to make sure that a string constant is considered one constant, you should always put single quotes around the whole affair:
'Hello World' Hello World
While the first string is always considered one string constant, the second one may, depending on where it is occurring, be considered one string constant denoting the string "Hello World", or two string constants, denoting the strings "Hello" and "World", respectively.
These issues will be dealt with in more detail below. For the moment, the curious reader may have a look at the following assignment statements:
#a := num '5' * '3' // value of #a will be 15 #b := set 5 * 3 // value of #a will be "5 * 3"
The first statement is a numerical assignment (denoted by the keyword "num
"). Both constants, 5 and 3, will be considered numerical constants, even if surrounded by quote characters. On the other hand, the second statement is a string assignment (denoted by the keyword "set
"). So the argument will, logically, be interpreted as one string constant, "5 * 3", even though it contains whitespace and lacks any quote character. What happens physically is that all separate words will be concatenated to the one string constant expected, inserting exactly one blank between each pair of words. The following statements show the consequences of this procedure:
#b := set 5 * 3 // value of #b will be "5 * 3" #b := set 5 * 3 // value of #b will be "5 * 3", too #b := set '5 * 3' // value of #b will also be "5 * 3"
In the second statement, though the words "5", "*", and "3" are separated by more than one whitespace character, the one string that will be built up from them is "5 * 3" – when concatenating them, they get separated by exactly one space character.
You may influence the way concatenation works by quoting some, or all, of the words to concatenate. Quoting a word will prevent STx from automatically inserting a blank before and after that word. So, compare the above statements with the statements below:
#b := set 5 '*' 3 // #b is "5*3" (no space) #b := set '5' * 3 // #b is "5* 3" (one space) #b := set 5 ' * ' 3 // #b is "5 * 3" (three spaces)
With the first statement, the word in the middle, "*", is quoted. This indicates STx on concatenation not to insert a space character either before or after this word, resulting in #b being set to "5*3" (no intervening whitespace).
With the second statement, the first word is quoted and will, hence, be concatenated to its right successor (there is no left predecessor) without inserting space. The second and the third word are not quoted and will be concatenated with an additional space in between them. This results in #b being assigned "5* 3" (no whitespace between "5" and "*", one blank between "*" and "3").
With the third statement, concatenation will not add any additional blanks either before or after the word in the middle. The whitespace that is part of the word, i.e. part of the quotation (three blanks before and after the asterisk, each), will be unaltered, though. So what results it #b being assigned the string "5 * 3" (exactly three blanks both before and after the asterisk).
Within a constant, you may alter the meaning of special characters by using the STx escape character "`", the backwards single quote, sometimes called back-tick. At the current stage, we can only use this feature for defining a string constant that contains single quote characters themselves:
#a := set 'Rome is a city but `'Rome`' is a four-letter word' #a := set Rome is a city but `'Rome`' is a four-letter word
Both statements will assign the string "Rome is a city but 'Rome' is a four-letter word" to a variable called #a (although it may not always be easy later to retrieve the value of this variable). We have to leave these issues open for later discussion.
Variables
The names of STx variables start with an optional one-character prefix indicating the scope of this variable (the lack of such a prefix indicating shell-global scope, see below). Besides this prefix, they may consist of letters and digits, although their first actual character must be a letter. Names are not case-sensitive, meaning that e.g. „freq", „Freq", and „FREQ", are names of the same variable.
Our STx discerns four kinds of scope and, hence, four kinds of variables:
Scope | Prefix | Description |
Global | @
|
Global variables are known to, and may be changed by, every shell instance, both running or yet to start. Global variables are the only kind of variables guaranteed to be persistent over interactive macro calls during one STx run. |
Shell | no prefix | "Shell-global" variables are global to the running shell. This implies that they are known to, any may be changed by, any macro invoked by a normal macro call. The variable will not, though, be known to other running shells or to shells yet to start. So, in general, shell-global variables will not be persistent through interactive calls to a user-defined macro.N.B.: Do not mix up shell-global variables with item handles that will be dealt with in chapter 5 (page 1). For the time being, suffice it to say that item handles look like shell-global variables, but that they reside in different namespaces and that they behave differently. |
Local | #
|
Local variables are valid only during the runtime of one invocation of a shell macro. On each invocation of a shell macro, a separate namespace containing all its local variables is being created. This namespace is destroyed as soon as the macro finishes. Note that this implies that on invocating a macro recursively, it will find itself starting with a fresh, empty copy of all its local variables, while the calling instance will find its namespace, i.e. its local variables, untouched. |
Member | &
|
When adhering to the object-oriented programming paradigm, you will define classes and instantiate them. Each instance of a class will have its own set of variables called member variables. The introductory chapters will stick to the clean ole' procedural way of programming. |
As a rule of thumb, most of the time you will be using local variables whose names start with "#
", e.g. variables like #i, #depth or #title.
Variables need neither be declared nor to be explicitly initialized. If you want to introduce a variable, just start using it. Note that a variable is empty, i.e. contains the empty string, when being referred to without having been explicitly set to a value different from the void.
Typing
There is really not much about typing in STx. All variables store strings, like it is the case with most scripting languages. Of course these strings are at liberty only to consist of digits, a comma, and an optional sign character, making them look like numbers, smell like numbers, taste like numbers, and being treated like numbers by numerical STx functions like addition, multiplication, or even the controversial subtraction.
Setting Variables
Setting a variable, i.e. assigning a value to the variable, is done with the ":=
" operator. Its general format is
variable := expression
You should never use this kind of assignment with STx, though. Instead, always use one of the following assignments:
variable := SET string_expression // typed string assignment variable := INT numerical_expression // typed integer assignment variable := NUM numerical_expression // typed numerical assignment variable := EVAL amazing_expression // awesome assignment using EVAL
See the following chapters for the reasons of this recommendation.
Simple string assignment
In its simplest form, this "expression" is a simple string constant, just like in the following example:
#adress := 'Reichsratsstrasse 17' // discouraged, see further below - always use SET
Or even:
#adress := Reichsratsstrasse 17 // discouraged, see further below - always use SET
Although generally valid, either usage is strongly discouraged, because it may lead to often surprising, seldom desired results if or when the string to be assigned starts with the name of a built-in function or a user-defined macro (note that words so far not reserved may become reserved words any time now, and that, when in a large software-building project, you never know how your colleagues call their helper macros today).
Typed string assignment
You may indicate your expressed desire for the expression to be a string constant by prefixing it with the type-selector statement "set
":
#address := set Reichsratsstrasse 17 // this is better
or, even better:
#address := set 'Reichsratsstrasse 17' // and this is the way to go
In this case, the assignment will even work if one day the STx macro language should be added a built-in function "Reichsratsstrasse" (which is, we dare to admit, unlikely) – or if one of your colleagues' macros happens to be likely called.
Typed numerical assignment
Although the value assigned is invariably a string, this string may be the result of a numerical computation. You may indicate your desire for it to be so by using one out of the following type-selector statements: "int
", "num
", and "eval
".
The "int
" type-selector will cause your expression being evaluated as an integer expression. More precisely (more precisely less wrongly), the expression will be evaluated numerically, and the result will be converted to an integer whose textual representation will be the string to be assigned to the destination variable. The calculation itself will be done with the point floating, though (see the below examples for what that means).
The "num
" type-selector will cause your expression being evaluated as a numerical expression, provided it is such. The textual representation of the numerical result will be the string to assign to the destination variable.
The "eval
" type-selector is the most powerful of them all. Firstly, it does everything the "num
" type-selector does. So, when evaluating a plain numerical expression, it is your free choice whether to use "num
" or "eval
" (we might one day choose spontaneously to fade out the "num
" type-selector, but do not allow this to bias your choice). But, secondly: The "eval
" type-selector is capable of much, much more: It does vector and matrix operations of the most sophisticated kind, calculates averages, converts units, dances the Fourier transform, and so on. The actual number of functions available to "eval
" expressions is more than 70 and counting (see the interactive STx help topics "eval").
For an example, compare the following statements:
#a := num 3*3.4 // result is 10.19999999999 #a := int 3*3.4 // result is 10 (!) #a := num 3*int(3.4) // result is 9 #a := int 3*int(3.4) // result is 9 #a := num int(3*3.4) // result is 10 (!) #a := num int(3*int(3.4)) // result is 9
Here the first statement will cause #a
to be assigned the result of the floating point multiplication of 3 and 3.4, i.e. about 10.2. In the second statement, the same multiplication will be calculated, and only the result of this calculation, i.e. 10.2 (roughly...), will get truncated to an integer – this integer in turn being 10. Only the third and the fourth statement (both!) will cause the second factor, 3.4, to be broken down to an integer before multiplication. This usage, though, is strictly not a feature of the type selector-based assignment, but of the numerical function (we will come to these later) "int". (Note that since the product of two integers is an integer itself, in the third and fourth statement it does not make any difference whether we use the "num" or the "int" type selector.)
Do not mix up the type selector of the assignment with any built-in type-conversion functions: Whereas the type-selector always immediately follows the assignment operator, ":=
", type-conversion functions never do. Furthermore, the arguments of type-conversion functions are always enclosed in brackets, whereas the type selector never uses brackets.
So in the third to sixth example, the type selectors are "num
", "int
", "num
", and again "num
", whereas the strings "int(3.4)
", "int(3*3.4)
", and "int(3*int(3.4))
" are calls to the built-in type conversion function, "int
". (If this sounds confusing for the moment, you should not worry: In practice, things are much easier, and it is not normally necessary to think these things over).
Function calls
Syntactically, every built-in STx function, every macro, and every method of a user-defined class may be the source of an assignment. If this happens to be the case, the respective function or macro is executed, and its result is assigned to the destination variable. See it for yourself:
#i := word 2 one two three four // #i will be "three"
The built-in "word
" function takes an integer index and a list of strings as its arguments. From this list, it selects and returns the string with the respective index. Hence, the expression "word 2 one two three four
" will select, and return, the third word, „three". When part of an assignment as in the above example, this very string "three" will be assigned to the respective destination variable, in this case: to the local variable #i
.
The same holds true if the source of the assignment is the name of a class method. In this case, the respective method is being called, and its result (which may as well be the empty string) gets assigned to the destination variable.
The issue of function calls will be dealt with in full detail in chapter Calls to STx Macros and Built-in Functions.
Accessing Variables
If and when you want to access the value of a variable (bluntly put: to read out its content), you need to put a dollar sign in front of the variable name. What happens internally is that, before actually executing a line from the macro file, STx replaces all occurrences of variable names that are prefixed by a dollar sign by the content of the respective variables. See for yourself:
#i := int 7 writelog 'The current value of variable #i is $#i' #i := int $#i + 1 // #i will be set to 8 #heading := set 'This is page $#curpage out of $#totpage'
The first line in this example will assign the value 7 to a local variable called #i. This is nothing new; note that we are using the type-selector "int
" to make sure the assigned value is interpreted as an integer, though in this case this is strictly redundant because 7 cannot help being an integer anyway.
The second line will print out the text "The current value of variable #i is 7". This should not be surprising, for the name of a variable is only replaced by its content if preceded by a dollar sign. Hence, the first occurrence of "#i" – although we know that there is a variable called #i – does not get replaced by that variable's value. The second, though, does, because it is preceded by the dollar sign. STx is one reliable piece of software, strictly and indiscriminately following the instructions laid out by its master.
What the third line does is increase the value of the local variable #i by 1. What's more interesting is how it does so. First, STx replaces all dollar-prefixed variable names by the contents of the respective variables. Hence, STx will replace the string "$#i" by the current value of #i which, in our example, happens to be 7. This replacement will change the current statement from "#i := int $#i + 1
" to "#i := int 7 + 1
". Now this is one fine integer expression that in turn gets evaluated to 8, thereby causing 8 to be the string that is finally assigned to the destination variable.
The fourth and last line demonstrates that substitution also works within string constants, and that it does so even if they are put under quotation marks (in our case, apostrophes). Some macro languages, e.g. the well-known UNIX shells, let certain kinds of quotation marks prevent substitution. Users familiar with such shells should bear in mind that STx is a kind of its own. (Note that if you really want to suppress the special meaning of a character like the dollar sign, you may precede it with the STx escape character "`
", the so-called back-tick.)
The aspiring STx guru may find it instructive to consider the following example (anyone else will find no harm in completely skipping it):
#var := set 'one' writelog '#var now containing "$#var"' #var := set 'two' writelog '#var now containing "$#var"' $#var := set 'three' // N.B.: substitution will make this // "two := set 'three'" writelog '#var still containing "$#var"' writelog '...but there suddenly is a variable called two' writelog '...and its value is "$two"'
The first line is well familiar. It assigns the string "one" to a local variable called "#var
". Consequently, the second line will print exactly the following string:
#var now containing "one"
The third line changes the value of #var
to "two", hence the third line will print out the following string:
#var now containing "two"
No surprises yet. But what will the fourth line do? Well, not to be surprised about the answer to this semi-rhetorical question, we must analyze the statement carefully. It reads "$#var := set 'three'
" – did you notice that the assignment target is preceded by a dollar sign? Alas, this instructs STx to replace the variable name by its content; but the content of variable "#var
" is "two." Hence the statement gets, by substitution, altered to "two := set 'three'
". This is a perfectly valid assignment statement, only that the target of the assignment is a shell-global variable called "two
". So we assign the string "three" to a variable called "two
". The next statements only illustrate this fact by printing out the respective values.
Read Functions
The STx read
family of functions supply a means for parsing the contents of a string or a variable, that is for splitting them into several pieces, and for storing some or all of these pieces into one or more other variables (or even in the same variable). That being said, it should be noticed that everything is much easier than this description implicates. See for yourself:
readstr 'one two three four five' #a #b #c /Delete // #a is now "one", #b is now "two", #c is "three four five" writelog 'now a="$#a", b="$#b", and c="$#c".'
The readstr
command parses its first argument into (at most) as many blank-separated strings as there are variables. If the number of variables is higher than the number of available words, the remaining variables will either be cleared (if supplying the /Delete
option), or they will be left untouched (if omitting the /Delete
option). If, on the other hand, the number of variables is lower than the number of available words, the last variable gets all the remaining words. Note that if there is more than one whitespace character between two words, this will not do any harm.
So what the above example does is parse the string "one two three four five
" into three substrings (there are three variables supplied, #a, #b, and #c). The first substring will be the first word, "one". The second substring will be the second word, "two". Since there are more words than variables, the third substring will catch all the rest, that is, the string "three four five". It's really simple, isn't it?
There is one additional feature you may, or may not, find convenient. You may as well parse strings that are separated by exactly one character of your choice. If, for some reason, you prefer semicolons over blanks, you might have done the above example as follows:
readstr 'one;two;three;four;five' #a ';' #b ';' #c /Delete // #a is now "one", #b is now "two", #c is "three;four;five" writelog 'now a="$#a", b="$#b", and c="$#c".'
The syntactical difference between those two examples is that the latter explicitly names the separator character between each pair of variables. The difference in semantics is that now there must be exactly one separator character between two words. So this variant of the readstr
command empowers you to read empty words, two. Consider the following statement (we may abbreviate the /Delete
option to /D
, if we do not care for the reduced legibility):
readstr 'one;;two;three;four;five' #a ';' #b ';' #c /D // #a is "one", #b is empty, #c is " two;three;four;five" writelog 'now a="$#a", b="$#b", and c="$#c".'
Looking awfully identical, doesn't it? Well, instead of one semicolon, there are now two semicolons between the first two words, "one" and "two". Believe it or not, this is making all the difference in the world: STx will consider these successive semicolons three separate words, the first being "one", the second one being empty, and the third one being, in general, "two" (in our case where there are only three variables supplied, the third word will get the rest of the string). So, with this example, variable #a will get the string "one", variable #b will be cleared, and variable #c will get the rest, that is the string, "two;three;four;five". Cool, isn't it?
Note that if you omit the /Delete
option, target variables corresponding to empty words will not be cleared, that is, they will keep whatever value that had before calling readstr
. That being said, you can easily foretell the results of the following statements:
b := set 'old value before calling readstr' readstr 'one;;two;three;four;five' #a ';' #b ';' #c writelog 'now a="$#a", b="$#b", and c="$#c".'
As you rightly foretold, the readstr
command in this example will not change the value of the second variable, #b, since the second word in this string is empty.
Although technically a consequence of the above, it may not immediately be clear that the space character, too, may explicitly named as separating arguments – and that this does cause a difference to the default readstr
behaviour. Look for yourself (and notice that there are two space characters between the "a" and the "b" in the first string constant):
readstr 'a b c' #a #b #c /Delete // #a is now "a", #b is "b", #c is "c" writelog 'a="$#a", b="$#b", c="$#c"' readstr 'a b c' #a ' ' #b ' ' #c /Delete // #a is now "a", #b is empty, #c is "b c" writelog 'a="$#a", b="$#b", c="$#c"'
As you already know, there is one important difference between both readstr
variants: When explicitly naming the separation character, STx considers consecutive occurrences of the separation character to separate empty strings. When not naming a separation, consecutive occurrences of whitespace are considered one single separator character, thereby causing no empty word to be read. So, in the above example, the first readstr
command will read "a" into #a, "b" into #b, and "c" into #c, whereas the second readstr
command will read "a" into #a, the empty word into #b, and the remaining string, "b c", into the last variable, #c.
Of course the string argument supplied to readstr
may even be the result of variable substitution. Less prosaically put, you might as well supply code like the following:
#three := set 'THREE' readstr 'one two $#three four' #a #b #c #d writelog 'now a="$#a", b="$#b", c="$#c", d="$#d"'
Before executing the command, STx will, as usual, look for any variable name prefixed with a dollar sign. If there happens to be any, they will be replaced by the contents of the respective variables. So in the above example, the text "$#three
" will get replaced by the contents of variable #three, thereby causing the STx readstr
command to actually be processed to be the following:
readstr 'one two THREE four' #a #b #c #d
Only the strong survive the following example (anyone else will find no harm in skipping it).
readstr 'x y z' #a #b #c writelog 'now a="$#a", b="$#b", c="$#c"' readstr 'one two three' $#a $#b $#c writelog 'gee, now a="$#a", b="$#b", c="$#c"' writelog 'but mysteriously, x="$x", y="$y", z="$z"'
When listening very carefully, you might hear the above example speak for itself. If not, you will find the key in the third line that contains the second readstr
command. Did you notice the variables being prefixed by a dollar sign each? Now the dollar sign indicates STx that variable substitution is desired (and required) before the command is to be processed. So all STx does is replace the two strings "$#a
", "$#b
", and "$#c
" by the contents of the respective variables. Since, at this stage, their contents are "x", "y", and "z", the command actually to get executed will look as follows:
readstr 'one two three' x y z
Now this clearly is a request to parse the string "one two three" into the three shell-global variables "x", "y", and "z".
When reading directly from one variable, you might prefer a variant of readstr
, the readvar
command. readvar
works similar to readstr
, as the following example shows:
#var := set 'one two three' readvar #var #a #b #c /Delete
The first argument to readvar
is one variable to read from. The remaining arguments are the variables where to store the words read. Otherwise that readvar
reads from a variable as opposed to reading from a literal string, there is no difference between readvar
and readstr
.
You will undoubtedly ask why there is such a thing as a readvar
command. After all, all that
readvar #var #a #b #c
does may as well be done using the following command:
readstr '$#var' #a #b #c
You are, of course, right in principle. The difference between the two commands is that the latter depends upon variable substitution which introduces a second step when evaluating and executing the command. Hence, the former is simply faster.
Note that there is a readtable
function, too. We will come to that much later when dealing with the versatile (both are) STx table feature.
Special Variables
There are a number of reserved variables that serve special purposes. At this stage, it suffices to give a short overview of the most important special variables.
Variable | Description |
rc | After executing an STx statement or built-in function (not a user-defined macro!), this variable contains its numerical return code, 0 indicating success, and values different from 0 indicating different kinds of failure. "rc" gets set after each invocation of an STx statement or built-in function, meaning that an error code will get reset when executing the next statement. If you later need to map a numerical error code to a textual error message (e.g. for presenting it to the user), you may always use the EMSG command. |
emsg | This variable contains a textual description of the value of the "rc" variable. It, too, gets reset with each new STx statement. If you have saved the value of rc e.g. in a local variable and you later need to map it to a textual error message (e.g. for presenting it to the user), you may always use the EMSG command. (And don't confuse the emsg variable and the EMSG command - they are not the same). |
#argc, #argv | On macro invocation, "argc" contains the number of arguments supplied to the respective macro, while "argv" contains the actual arguments (all of them). See below. |
result | After returning from a user-defined macro call, the variable "result" contains the value returned by the respective macro, i.e. the value the macro supplied as an argument to the "exit " call. If the macro was left without returning an argument, "result" is empty (this is not considered an error – honestly, there is not much that is ever considered an error as far as STx is concerned).
|
#read | After using one of the STx READ commands (read , readstr , readvar , readtab ), this variable contains the number of arguments actually read.
|
#new | When allocating an STx shell item, its item name will be stored in a variable called "#new". On the issue of shell items, please be patient until chapter Shell Items: An Overview, or divert to the Shell Items chapter in the Reference Manual. |
In general, even the reserved variable may be target of an assignment (this is sometimes used with the "#argv" variable for implementing default macro arguments). You should not be surprised, though, that assigning a value to either "rc" or "emsg" will not have the desired outcome: Since the assignment statement is built-in STx statement itself, executing it will reset "rc" to 0 and "emsg" to the empty string, thereby indicating that the assignment statement itself was successful (which it was).
Control Structures
Calls to Macros and Built-In Functions
You might argue that we have been through that already, but, unfortunately, this is only part true: There is much more to passing an argument to a function than we have explored when calling the built-in "word
" function, or the built-in "writelog
" command. This chapter tells why and what.
Macros
Semantically, an STx macro is like a procedure or function of any procedural programming language of your choice. Syntactically, a macro starts with a macro definition surrounded by square brackets, e.g. the following line:
[Macro mymacro]
There is no clear end to a macro. Execution continues until the control flow reaches an "exit
" statement, until a new macro starts, or until the file comes to an end (whichever happens first).
Calling a macro
For calling a macro, you simply type the macro name. In the above case, if STx encounters a line starting whose first word is the string "mymacro
", it will execute the like-named macro. STx supports recursive macro calls.
There are a few commands providing different, not always cleaner ways of calling a macro, namely the MACRO, the MACROX and the SHELL command. Please do not use these commands unless there is very good reason to (there normally isn't).
Returning From a Macro
Macro execution will stop when STx encounters the "exit
" statement. Normally, control flow resumes at the line immediately following the statement that caused the macro to execute. You may supply the following optional arguments to an exit statement:
exit level result
When calling "exit
" without any arguments or when calling "exit 1
", control flow will, as said above, resume with the next statement after the macro call.
In general, the first argument to "exit
" indicates how many call levels to skip. When calling "exit 2
", control flow will not resume at the line after the statement calling the macro, but at the line after that statement calling whichever macro was in turn calling our macro. When calling "exit 3
", control flow will resume at the line after the statement calling the macro calling the macro calling our macro. When calling "exit 4
", control flow will resume at the line after the statement calling the macro calling the macro calling the macro calling our macro, while on "exit 5
" execution will resume with the line after the statement calling the macro calling the macro calling the macro calling the macro calling our macro, and so on. There is limited use to this feature, and you are at the safe side when, at least in the beginning, always using "exit 1
".
There are two special cases, one when supplying 0 for the exit level, the other when supplying a negative number.
When supplying 0 for the exit level, i.e. when executing exit 0
, the executing shell will be terminated, effectively ending execution of the whole user script (both the running macro and all calling macros).
When supplying a negative level argument (e.g. exit -2
), STx will return from as many macro levels as needed to find a macro level where there is a non-empty local variable #onexitall defined. Since this is a bit complicated and not very clean, you probably should try not to use this feature.
Of more interest than the first is the second argument to "exit
". It is the result of the macro, that is the number or string the macro is to return. Like with assignment operations, this result may, and should, start with a type selector, one out of "set
", "int
", "num
", and "eval
". Consider the following "exit
" statements:
exit 1 set 'Hallo Welt' exit 1 int 5/3 exit 1 num 3/5
The first statement will exit the running macro and return the friendly string "Hello World". The second statement will exit the running macro and return whichever integer results on dividing 5 by 3. Finally, the third statement will exit the running macro and return the quotient of 3 and 5.
There are several interchangeable ways for the caller to retrieve whatever value the macro has returned. First of all, the result is stored in the reserved shell variable "result
", rendering the following code snipped a working example:
greetings // call the "greetings" macro writelog '$result' // ...and print its results exit // ...and rest after a long day's work [Macro greetings] exit 1 set 'Hello World'
An even more elegant feature is using command substitution, a concept both known and feared from the UNIX shells:
#var := set '$(greetings)' writelog '$(greetings)'
The first statement executes the macro "greetings" and assigns its result to a variable called "var
". The second statement, too, calls the "greetings" macro, but directly pastes its result into a "writelog
" statement.
Though command substitution is the most general way, a macro may be directly called, too – just like any built-in STx function:
#var := greetings
The latter format greatly improves legibility and is therefore he recommended way of calling a user-defined macro.
Supplying and Retrieving Arguments
There are, they say, many ways leading to Rome, and this surely holds true for retrieving and parsing the arguments a user-defined macro has been supplied with. The easiest way is disposing of the problem in the macro definition, i.e. by writing something like this:
[Macro multiply #mand #mor] #prod := eval $#mand * $#mor writelog 'multiplying $#mand by $#mor results in $#prod' exit
As you may have noticed, the macro name may be followed by one or more variable names. In this case the arguments supplied to the macro are being automatically parsed and stored to the like-named variables. We call this the implicit argument parsing feature. Hence, when calling "multiply 7 9
", the starting macro will find its local variable #mand
set to 7 and its local variable #mor
set to 9. This will cause the above macro to print the message "multiplying 7 by 9 results in 63". If there are fewer arguments than variables, the remaining variables will be empty. If there are more arguments than variables, the whole remaining part of the arguments will be stored to the last variable. When you think it over, this is completely analogous to the readstr
and readvar
functions already known.
What's also analogous to readstr
and readvar
is the possibility explicitly to name a separator character of one's choice. See for yourself:
[Macro multiply #mand';'#mor] #prod := eval $#mand * $#mor writelog 'multiplying $#mand by $#mor results in $#prod' exit
While the first variant of the user-defined macro "multiply" will expect its arguments to be separated by an arbitrary number of whitespace characters, the second version will expect them to be separated by exactly one semicolon. This comes in handy when passing arguments that may contain, or contain, or must contain, or should contain, or would contain were it not for the fact that this is impossible with the default way of argument parsing, whitespace characters.
A further and generally slightly more flexible, though a little more laborious way of parsing one's arguments is using the special variables "#argc
" and "#argv
". On start-up of a macro, these variables get set to the number of arguments and to the actual argument list, respectively. So the above example may be re-written for explicit argument parsing as follows:
[Macro multiply] readvar #argv #mand #mor #prod := eval $#mand * $#mor writelog 'multiplying $#mand by $#mor results in $#prod' exit
Why is this more flexible in general? – Because of the greater versatility of the readvar
and readstr
commands that even allow parsing argument lists built up from a variable number of arguments. We will come to that with the macro countSTXProgrammers
of chapter 4.1.1.3 on page 1.
That being said, we may well continue with a real-world example. Consider e.g. the following macro:
[Macro declare #person #attribute] writelog 'I hereby declare $#person an $#attribute' exit
Now, if you would like to declare, say, Toni an STx hero, you might call the macro as follows:
declare Toni STx hero
This will work fine because the first word, "Toni", gets stored to the first variable, #person
, and the rest of the arguments, "STX hero", gets stored to the second, and last, variable, #attribute
. But what if you want to declare Christian Gottschall an STx beginner? See for yourself:
declare Christian Gottschall STx beginner
This will invariably lead to the first word, "Christian", being stored to #person
, and the whole rest, "Gottschall STx beginner", being stored to #attribute
, hence causing Christian being declared a (sic!) "Gottschall STx beginner", which he surely isn't.
You might be tempted to attacking this problem by using quotes, but in fact this is utterly impossible. You will find any conceivable combination (and even most unconceivable combinations) of quotes and escape characters to have a meaning very different from that intended. So the only general solution to this problem is using a different separator character:
[Macro declare #person';'#attribute] writelog 'I hereby declare $#person an $#attribute' exit
Or:
[Macro declare] readvar #argv #person ';' #attribute writelog 'I hereby declare $#person an $#attribute' exit
Now you may declare whomever you want whatever you please:
declare Toni ; STx hero declare Christian Gottschall ; STx beginner
Though looking unusual from the standpoint of several other programming languages, it is good STx practice to quote the macro arguments, either as a whole, or argument-wise, or even (don't read this loudly!) on a per-word basis:
declare 'Jonnie White ; STx guru' declare 'Jonnie White' ; 'STX guru' declare 'Jonnie' ' White' ; 'STX ' 'guru'
In general, neither form of quotation will do any harm. Remember, though, that, with STx, writing several quoted strings will cause them to be implicitly concatenated with any intervening whitespace removed. Hence, although the above examples work as desired (note the quotations of the fourth statement containing space characters), the following won't:
declare 'Jonnie' 'White' ; 'STX' 'guru'
The latter statement will print out the text "I hereby declare JonnieWhite an STXguru": Although there is a whitespace character between each pair of quoted words, the quotations themselves do not contain any blank character, thereby causing the respective words to be concatenated without any intervening space. Compare this with the third statement in the previous example where there is a space character at the beginning of the second, and at the end of the third quoted word. (We've already had a few words on concatenation in chapter on page 1.)
Note that when using the implicit argument parsing feature (or when using readvar
with the default separation characters (whitespace), you need to be careful not to let the concatenation feature come in your way. Consider the following macro call:
usermacro 'one' 'two' 'three'
Here, by string concatenation, the two strings "one", "two," and "three" will get concatenated to one single string, "onetwothree". Consequently, the macro will, whatever way of parsing it uses, get only one argument – the string "onetwothree." If you want the macro actually to be called with three arguments, you will need to use one of the following statements:
usermacro one two three usermacro 'one two three' usermacro 'one ' 'two ' ' three'
It will work either way, additional whitespace characters never doing any harm.
As a rule of thumb, the easiest thing would be always to quote the whole arguments to a macro call, like is done in the second statement of the above example.
Combined example
The following example builds up several things we are at this stage familiar with (and several others we are not). You need not fully understand the macro at this stage, but you might notice several familiar issues.
[Macro countSTXProgrammers] // read 0 both into #count and #totcount readstr '0 0' #count #totcount // (1) // the "forever" loop will never terminate by itself. // it will only stop at a "break" statement forever // (2) // read the first person into #person, and the // remaining persons into #argv readvar #argv #person ';' #argv /Delete // (3) // return both the total number of persons counted and // the number of STx persons if '$#read' == 0 then // (4) [[{Programmer_Guide/Command_Reference/EXIT|exit]] 1 set '$#totcount;$#count' // (5) end #totcount := int $#totcount+1 #index := keyword '$#person' Toni Jonnie Christian // (6) if '$#index' >= 0 then #count := int $#count+1 end end
The countSTXProgrammers
macro gets an arbitrary number of arguments separated by semicolons. Each argument is considered the name of a person. What the macro does, slightly arbitrarily, is count both the total number of persons supplied, and the number of STx programmers amongst them (STx programmers being considered Toni, Jonnie, and Christian only). The most important statements are the following (see the respective numbers commented in the macro source):
- (1)
- This shows an idiomatic use of the
readstr
command for initializing multiple variables. Here the string "0 0" is parsed into two variables,#count
and#totcount
, effectively setting them both to zero. Of course you might as well use the two separate assignment statements "#count := int 0
", and "#totcount := int 0
". - (2)
- The
forever
keyword starts kind of an eternal loop running until abreak
statement will be met. Both issues are dealt with in separate chapters (Forever and Ever and Ending a Loop Prematurely: Break and Continue). - (3)
- The
readvar
statement will be utterly familiar to you. Note, though, that one of the destination variables is the same as the source variable. This is not a problem at all. Parsing will proceed normally, and the assignment will take place only after parsing has finished. So thisreadvar
statement will result in the first word of #argv being parsed into #person. The remaining contents of #argv will, in turn, be parsed to%… #argv itself(!), effectively removing the first word from #argv. So with each pass of the loop, the next "first" name from the list will both be stored in #person and be removed from #argv. Isn't that cool? - (4)
- This line introduces the
if
command. You need not worry about this command; it will be dealt with properly in chapterc 4.1.3. - (5)
- The
exit
statement will return from the macro, and it will do so – due to theif
statement – on the condition that #read is zero which is the case as soon as there are no more names to read. The first argument to theexit
statement is 1, indicating that only the current macro is to end (and that execution shall continue with the calling macro). The second argument to theexit
statement is the result of the macro, in this case: the string to return (indicated by the string assignment type selector,set
). This string is built up from #totcount, a semicolon, and #count, and will hence contain the total number of persons and the number of STx programmers, separated by a semicolon. - (6)
- This statement uses the built-in
keyword
function to investigate whether the current person is on the list of STx programmers.
Statements and Built-in Functions
Arguments to STx statements and to built-in functions differ in several respects from macro calls:
- One quoted string is always considered one argument, even if it contains whitespace characters.
- Hence there is no automatic string concatenation either.
- If (and only if) the statement or the function expects a numerical argument, you may supply a numerical expression instead of a number. This expression will be evaluated before executing the statement, or calling the built-in function, respectively.
The first two issues are easily demonstrated by the following example:
#a := word 2 'a b c' d 'e' f writelog '#a="$#a"' // value of #a is "e"
Both quoted strings will be considered one argument each, and no concatenation will take place. Hence, the built-in word
function will return the string "e", "e" being the third argument (index 2).
More surprising is the third issue, but it, too, can be demonstrated by a simple example:
#a := word 1+1 'a b c' d 'e' f writelog '#a="$#a"' // value of #a is "e", too
Since the "word" built-in expects its first argument to be a number, any numerical expression occurring at the respective position will be evaluated before calling "word". Hence, this example will return "e", too, because 1+1 equals 2. Compare this with the following statements:
#a := word 1+1 'a b c' d '2+2' f writelog '#a="$#a"' // value of #a is "2+2"
Here what the third argument of the "word" built-in will return is the string "2+2". This string does not get evaluated, because "word" expects a string argument (and not a numerical argument) at this position.
Note that if the numerical expression is to contain whitespace (or if it cannot be precluded that it does, e.g. when it is built up using variable substitution), you need to quote the whole expression. This quite naturally leads to the following rule of thumb: Always quote numerical expressions that are arguments to a statement or to a built-in function. See the following example:
#i := int 3-2 #a := word '$#i+2' a b c d e writelog '#a="$#a"' // value of #a is "d"
First of all, #i gets assigned the difference between 3 and 2, that is 1. String substitution will change the string "$#i+2
" to "1+2
", "1
" being the value of #i. Finally, due to the fact that the "word
" function expects its first argument to be numerical, "1+2
" will be evaluated as a numerical expression, resulting in 3. So the fourth string argument, "d", will be returned.
If Conditions are to be Tested
If conditions are to be tested, the "if
" statement comes in handy. It actually does so in two flavours:
if condition statement
And:
if condition then many statements else many more statements end
Before going into more details about what conditions look like, a few simple examples should make things more clear:
if '$#a' == '' [[writelog 'Variable #a is empty.' if '1+2' != 3 then // note the use of the STx escape character, the back-tick, // for building up a string that contains a quote character writelog Wohllebenstraße, we`'re having a problem. else writelog All systems nominal. end
if a > c then // character-wise comparison writelog 'Detecting an unusual character set' else writelog 'Detecting a usual character set' end
if '2+3' != '3+2' writelog 'STX addition is not commutative'
Note that with the first, simple form of the "if
" command, the statement to be executed in case of the condition being true must not be another "if
" command. (In simpler wording: Simple if statements must not be nested.) Hence, it is not allowed to construct a statement like the following:
// BAD! NOT ALLOWED! ERRONEOUS! DON'T DO IT! // COMBINING TWO SIMPLE "IF" STATEMENTS IS FORBIDDEN! if '$a' == '$b' if '$c' != '$d' writelog a and b are equal, but c and d are not
If two or more "if
" statements are to be combined, use their complex form instead:
if '$a' == '$b' then if '$c' != '$d' then writelog 'a and b are equal, but c and d are not' end end
Conditions, Formally Revisited
Simple Comparison
Conditions basically compare two entities for being equal, or for one of them being less than, or higher than, or not higher than, or not less than, or quite unlike the other. All this is done with the comparison operators "=
", "<
", ">
", "<=
", ">=
", and "!=
". The arguments to these operators may either be strings, in which case a Unicode-based string comparison will take place, or numerical expressions, in which case a numerical comparison will take place. If the argument to a comparison operator looks like a numerical expression, it will be treated as such, even if the programmer did not mean that to happen.
Note that unless the argument to a comparison operator is quoted, it must be separated from the operator by at least one whitespace character. This means that while "'$#a'=='1'
" (no whitespace, but quotes) and "$#a == 1
" (blanks between the operator and its arguments) are well-formed expressions, "$#a==1
" is not. (Yet another reason for always quoting everything, one might feel inclined to say.)
See a number of simple examples:
if '$#a' > '$#b*2' writelog '#a is more than twice as much as #b' if 'int($#a)' == '$#a' writelog '#a is an integer' if '$#a' == '0+$#a' then writelog '#a is a number.' else writelog '#a is not a number.' end
All these examples show that numerical expressions occurring in comparisons will be evaluated. Hence, the first example will compare the value of #a
with twice the value of #b
. The second example will compare #a
with its integer part. If they are the same, we know that #a
must store an integer.
The most interesting of these examples is the third one. What it does is check if the variable #a
is numeric, i.e. if it contains a number. For understanding how it does so you need to remember that STx evaluates numerical expressions only. Now, should #a
contain a number, say, 42, variable substitution will alter the expression "if '$#a' == '0+$#a'
" to "if '42' == '0+42'
". Now since "0+42
" is a numerical expression (remember that single quotes do not matter), it will, in turn, be evaluated to 42 (0+42 equalling 42). This leads to the final statement "if 42 == 42
". Since the number 42 is equal to 42, the condition will come out true, causing the "then
" branch to be executed.
If, on the other hand, #a
contains a value that is not a number, say, the string "Hello", variable substitution will alter the "if
" statement to "if 'Hello' == '0+Hello
". Due to "0+Hello
" not being a numerical expression, there will be no evaluation, resulting in the string "Hello
" to be compared against the string "0+Hello
". Since they are not identical, the "else
" branch will be executed.
Note that there is one potential pitfall (and note the alliteration, too). Should #a consist of digits separated by whitespace, it will be considered a number (one number) anyway. So e.g. the string "123 440 . 12
" will be considered the number "123440.12". This is a feature allowing for separating numbers into groups of digits (e.g. a group of ones, tenths, and hundreds, and a group of thousands, ten-thousands, and hundred-thousands).
Pattern Matching
Besides these comparison operators, there are matching operators, too. Matching may be considered a more powerful way of comparing strings that is able not only to find out if two strings are strictly identical, but also to find out whether they are similar in a way yet to be defined. With a matching operation, its first, left-hand side argument must be the string to match. Its second, right-hand side argument must be the pattern against which to match the string.
Wildcard Pattern Matching
Wildcard patterns are similar to strings with the exception that they may contain wildcards (hence the name). With STx, these wildcards are the asterisk, "*", for matching a string of arbitrary length, and the question mark, "?", for matching exactly one character.
There are the following wildcard matching operators:
- Wildcard string matching
=SI
|
The string argument does match the pattern. Case will be ignored. |
!SI
|
The string does not match the pattern, not even regardless of case. |
=SR
|
The string argument matches the pattern case-sensitively. |
!SR
|
The string argument does not match the pattern when respecting case. |
- Wildcard name matching
- Name matching is similar to string matching, the difference being that the string to match is expected to be a well-formed STx name. If this is not the case, the match will fail, even if the pattern would string-match the string. More precisely, matching a non-name string with
=N
will always return false, whereas matching a non-name string with!N
will always succeed, regardless of the pattern used.
=NI
|
The name argument matches the pattern ignoring case. |
!NI
|
The name argument does not match the pattern, not even regardless of case. |
=NR
|
The name argument matches the pattern including case. |
!NR
|
The name argument does not match the pattern when respecting case. |
Have a glance at the following examples:
if 'hallo' =SI ha* writelog 'OK' // 1 if 'hallo' =SI *ha* writelog 'OK' // 2 if 'hallo' =SI h*o writelog 'OK' // 3 if 'hallx' !NI ha*o writelog 'OK' // 4 if 'test' !SI ha* writelog 'OK' // 5 if 'test' !SI *ha* writelog 'OK' // 6 if 'test' !SI h*o writelog 'OK' // 7 if 'h-o' !NI h*o writelog 'OK' // 8 if 'test' =SI t*t* writelog 'OK' // 9
Each of these comparisons should come out true, hence causing each statement to print the string "OK". Let's have a look at a few of these examples in detail:
The pattern, "ha*
", will match any string starting with "ha
" – "hallo
" does. So "=SI
" will guarantee a positive outcome.
The pattern "h*o
" will match any string whose first character is an "h" and whose last character is an "o". Now, "hallx
" is no such string, but since we are using a negative comparison operator, "!NI
", the whole thing comes out true since the pattern does not match. (Using "!NI
" instead of "!SI
" presupposes the string to be a well-formed STx name which, by chance, "hallx
" is. But compare statement 8.)
The pattern "h*o
" normally matches any string starting with "h" and ending with "o". That it does not match in this expression is because the "NI
" operator presupposes its argument to be a well-formed STx name. Now, "h-o
" is no such name because it contains a minus sign, "-
". So, the match fails by necessity, hence causing the negative expression "!NI
" to become true.
If you are shaken by statement 8 of the previous example, you might want to try out for yourself the following macro:
if 'h-o' !NI h*o writelog 'OK1' // 1 if 'h-o' !SI h*o writelog 'OK2' // 2 if 'h-o' =NI h*o writelog 'OK3' // 3 if 'h-o' =SI h*o writelog 'OK4' // 4
What this will do is print out "OK1" and "OK4", but neither "OK2" nor "OK3". Do you see why?
Well, first of all, the string "h-o
" is no valid STx name due to its containing the minus sign, "-
". So every name-based comparison will fail before even considering the pattern argument. So the third comparison, "=NI
", will fail though "h-o
" is a string starting with "h" and ending with "o". Consequently, the expression inverting that condition (using "!NI
" instead of "=NI
") will come out true, causing the first comparison to come out true.
With the string-based comparison operators, things are different. Stringly put, the pattern "h*o
" does match the string "h-o", that string actually starting with an "h" and ending with an "o". So the positive operator, "=SI
", will come out true, while, consequently, its negative form will come out false.
Regular-Expression Pattern Matching
Aside from the simple wildcard-patterns, STx also supports full POSIX regular-expression pattern matching, using the open-source TRE library. Here is not the place for introducing POSIX regular-expressions, but if you are familiar with that concept, you will find the following STx matching operations useful:
- Regular-expression string matching
=RSI
|
The string argument does match the regular expression. Case will be ignored. |
!RSI
|
The string does not match the regular expression, not even regardless of case. |
=RSR
|
The string argument matches the regular expression case-sensitively. |
!RSR
|
The string argument does not match the regular expression when respecting case. |
- Regular-expression name matching
- This, too, is similar to regular-expression string matching, the difference being that the string to match is expected to be a well-formed STx name. If this is not the case, the match will fail, even if the regular expression would string-match the string.
=RNI
|
The name argument matches the regular expression ignoring case. |
!RNI
|
The name argument does not match the regular expression, not even regardless of case. |
=RNR
|
The name argument matches the regular expression including case. |
!RNR
|
The name argument does not match the regular expression when respecting case. |
Building up Complex Expressions
You may combine more than one comparison either conjunctively, using the "&&
" operator, or disjunctively, using the "||
" operator. There is, though, neither precedence nor grouping, meaning that (a) complex expressions will be evaluated strictly from left to right and that (b) you cannot use brackets for grouping sub-expressions. While the former is not strictly an offence once one gets used to it (other programming languages like APL behave very similarly), the latter may sometimes cause some programmer inconvenience. We will try addressing this issue a wee bit later.
Evaluation stops as soon as the outcome is clear, but with STx, this is only a performance issue (there is no such thing as built-in functions with side-effects, whereas command substitution, on the other hand, will always be done regardless of where the "$(...)
" expression actually occurs).
See the following examples:
if 1 > 2 || 3 > 1 writelog 'condition 1 is true' if 1 > 2 || 3 > 1 && 0 == 1 writelog 'condition 2 is true' if '$#a+1' < '$#b*3' || '$#c-7' == '$#d' writelog 'condition 3 is true'
Whereas the first statement should be fairly clear, fully to understand the second statement requires being aware of STx strictly evaluating from left to right: For STx, the condition "1 > 2 || 3 > 1 && 0 == 1
" will mean the same as, for a human reader, "(1 > 2 or 3 > 1) and 0 == 1" would. The "and" ("&&
") does not – as it normally does in logic and mathematics – take precedence over the "or" ("||
").
What you type | What STx does |
$a > $b && $b < $c || $c < $b && $b < $a
|
(($a > $b && $b < $c) || $c < $b) && $b < $a
|
$a < $b && $a > 0 || $b == -1
|
($a < $b && $a > 0) || $b == -1
|
$a < $b || $b > $a && $invert == 1
|
($a < $b || $b > $a) && $invert == 1
|
$a < $b || ( $b > $a && $invert == 1 )
|
Syntax error (brackets are not allowed!) |
From the theoretical standpoint, the lack of grouping (i.e. brackets) allows for a much faster evaluation. It does, though, sometimes force the programmer either to combine several "if
"-statements, or to alter his or her expression – that is why support for bracketing in logical expressions is under active consideration for STx versions to come. For the time being, here are a few examples on how to rewrite complex expressions:
What you mean to do | What you need to do to do what you mean to do |
if ($a > $b && $b < $c) || ($c < $b && $b < $a) then statements… end
|
#flag := int 0 if $a > $b && $b < $c #flag := int 1 if $c < $b && $b < $a || $#flag == 1 then statements… end
|
if $a > 0 || ( $a < 0 && $b == int($b) ) then statements… end
|
if $a < 0 && $b == int($b) || $a > 0 then statements… end
|
if ($a > $b) && $b < $c) || ($c < $b && $b < $a) || ($b < $c && $c < $a) || ($b < $a && $a < $c) then Statements… end
|
#flag := int 0 if $a > $b && $b < $c #flag := int 1 if $c < $b && $b < $a #flag := int 1 if $b < $c && $c < $a #flag := int 1 if $b < $a && $a < $c || $#flag == 1 then statements… end
|
Reference
- Conditional Expressions Reference in the STx Programmers' Guide
While Loops Loop
While loops model our everyday concept of repeating a task until it finally has the desired outcome. With STx, this is done as follows:
while expression many statements… end
With this code, STx will first evaluate the expression supplied to the "while
" statement. If this expression happens to be true, all statements between the "while
" and the corresponding "end
" statement are being executed once. Afterwards, STx again evaluates the very same expression. If it is still true, all the statements between the "while
" and the "end
" statement are being executed once more. Then, again, the expression is evaluated… and so on, the effect being that all statements between the "while
" and the "end
" statement are being executed "while" (i.e. as long as) the condition happens to be true.
For Loops to Be Organized
The "for
" statement is a more organized loop statement. Its syntax is the following:
for startcmd to whilecondition step changecmd many statements… end
First of all, "startcmd
" – some arbitrary command – gets executed. Next, the expression "whilecondition
" gets evaluated. If it is false, the control flow will continue with the line immediately following the "end" statement. If, on the other hand, it is true, all the statements between "for" and "end" will be executed. Next, the "changecmd
" – again, any command of your choice – will be executed. After that, things will start all over, with "condition" being evaluated again.
This is functionally equivalent to the following code snippet:
startcmd while whilecondition many statements… changecmd end
So the for
statement does not introduce any new functionality. What it does, though, is saving space and making this certain kind of loop more readable.
Note that the for
statement is modelled after the like-named statement of fashionable programming languages like C, C++, Java, or the author's all-time favourite AWK. From a semantical point of view, the keyword "to
" may seem slightly misleading. Suffice it to say that there is a lesser-known (and these days virtually extinct) Klingon dialect where the word "t'o" means the same as the English imperative phrase "do this while" (at least we trust this statement to be hard to disprove).
As an example, counting from 1 to 10, may serve the following "for" loop:
for #a := int 1 to $#a <= 10 step #a := int $#a + 1 writelog 'at this stage, #a=$#a' end exit
Forever and Ever
People often feel uneasy with the rapid change and the lack of durability in today's fast-paced environment. STx addresses these issues with the "forever
" statement that allows for enduring, long-term code execution. In fact, "forever
" will start a flow of execution that never ends:
forever eternal statements… end
This kind of loop will cause the statements between "forever
" and "end
" to be processed for eternity (or until your computer breaks down, whichever happens first).
Ending a Loop Prematurely: Break and Continue
If STx encounters the "break
" statement within the body of a loop, this very loop gets at once terminated. Control flow will resume at the statement immediately following the "end
" statement at the end of the loop.
The "continue
" statement will have a different effect: Whenever STx encounters this kind of statement, it will directly pass to the next run of this loop, ignoring any statements between the "break
" statement and the "end
" statement at the end of this loop.
Consider the following example:
// print odd numbers less than 20 that are not a multiple of 3 #i := 0 forever #i := int $#i + 1 if '$#i' >= 20 break // note that "int" here is an in-place function truncating the // fractional part off a number if '$#i/3' == 'int($#i/3)' || '$#i/2' == 'int($#i/2)' continue writelog '$#i is neither a multiple of 3 nor even' end writelog 'Done.' exit
Though not at all an example in style, the above loop surely demonstrates the use of both the "break
" and the "continue
" statement. The basic idea of this loop is to run forever, with each pass increasing #i
by one. Immediately after increasing #i
is where the "break
" statement finds its use: #i
is being compared against twenty, and after having reached this upper limit, the whole loop will be terminated. Execution will continue with the first statement after the "end" statement of the respective loop. In our example, this first statement is the call to the built-in "writelog
" statement printing out the string "Done."
The next issue demonstrated here is the "continue
" statement. After having ensured that #i
has not reached its desired upper limit, #i
is being checked for being either a multiple of 3, being a multiple of 2, or both. If so happens, STx will "continue" the loop, i.e. omit its remaining part (in our case, the "writelog
" statement), and continue with its first statement.
Should the loop to be "continued" be a "for
" or "while
" loop, the next run of the loop will begin normally, that is with testing the expression supplied to the "for
" or "while
" statement. Should it prove false, the loop will terminate. In the case of a "for
" loop, its "changecmd
" (see above), too, will be executed normally, i.e., before checking the condition.
Statements Considered Useful: goto, gosub, and gosubx
Each statement in an STx script may be labelled. The name of a label may consist of letters, digits, and underscores (though it must not start with a digit). The label must be immediately followed by a colon (that's necessary because otherwise there would be no way to discern between labels and macro calls). Note that labels are local to the macro they are part of. It is hence not possible to jump to a label that is part of a macro different from the executing macro (and it is possible to use labels of the same name in more than one macro).
When using a label, you may at any time divert the flow of control to that label. This is done with the "goto
" command. Though it has come out of fashion recently, the "goto
" command allows for very efficient diversions in the control flow that may be really hard to follow.
Contrary to most other programming languages, STx allows "goto
" to be supplied not only one, but even two labels. If this is the case, STx first tries to go to the first label. Only if there happens to be no such label, STx falls back to the second label and tries to go there. If the second label is not present either, the "goto
" command will do nothing (it will leave an error code in the variable rc
, though, but most programmers do not check for errors anyway).
The following loop will count from 1 to 10 (compare this with the "for
" loop of the last example in the previous chapter):
#a := int 0 looping: if '$#a' >= 10 goto endloop #a := int $#a + 1 writelog 'at this stage, a=$#a' goto looping endloop: writelog 'let it be' exit
A different issue is the "gosub
" command. It is very similar to calling a macro: A new execution environment will be created, and control will resume at the label referred to by the "gosub
" command. When (and if) the control flow reaches an "exit
" command, this execution environment will be destroyed, and control will resume with the line immediately following the "gosub
" command that caused all that to happen. Summarizing things up, the code called by the "gosub
" statement will see all variables empty. Furthermore, any changes made during the statements invoked using "gosub
" will get lost on returning. This is a very powerful feature.
The "gosubx
" statement differs from the "gosub
" command in that no separate execution environment is being created. Hence, the code will see all variables having their current value, and, conversely, any changes in variables done by the called code will remain in effect after returning. This, too, is always a powerful, sometimes a desired feature, and seldom easy to follow.
Compare the effect of the "gosub
" and the "gosubx
" commands in the following example:
#a := 5 gosub subroutine writelog 'a after gosubbing subroutine: $#a (no change!)' gosubx subroutine writelog 'a after gosubxing subroutine: $#a (changed!)' exit subroutine: // should #a be unset, numerical evaluation will fail #a := int $#a + 5 writelog 'value of a in subroutine: $#a' exit
Note that we do not actually dare to recommend using any of these statements, and that we strongly encourage structured programming (using "for
" and "while
" loops and breaking program code into macros). There may, however, be cases where the real programmer finds that he or she can do many interesting things with the help of the statements considered harmful since Dijkstra's famous letter.
Shell Items: An Overview
Shell items are a kind of their own. Each shell item has a handle – actually kind of an internal name – that is not normally directly known to the programmer. Shell items may be created using the "new
" command, and they may – and should – be disposed of properly after use which is done with the "delete
" command. The handle to a shell item is normally stored in an STx variable.
There are different sorts of shell items, the most important being file items (representing disk files), graph items (representing graphs visible to the user), dialog items (representing GUI dialogs, menus, windows, and the like), table items (representing in-memory databases, numerical vectors, and matrices), value items (providing things as diverse as timers, and fast matrix operations), wave items (for handling soundfiles), and instances of user-defined classes. We will come to each of them later (or at least to most of them).
Beginners occasionally find it difficult to grasp the difference between a variable and a shell item. It is often helpful to remember the following properties:
- Shell items are very complex objects like files or tables that normally consist of many parts (e.g. the rows and columns of a table, or the records of a file) most of which are themselves complex objects.
- Variables, on the other hand, are very simple and dumb things: They store one string each – nothing complex in that.
- Shell items are referred to by their handle, variables by their name.
- You usually store the handle of a shell item in a variable.
- You usually do not access a shell item directly, but by using the variable storing its item handle.
- For using a variable, i.e. accessing its contents, you need the dollar sign ("$"). Opposed to this there is no dollar sign with item handles, because there is no such thing as a simple content of an item.
You may at any time request some action from an existing shell item. For doing so, you use the following command:
set itemhandle actionrequest
Here "itemhandle" is the internal handle of the respective STx item, and the exact nature of "actionrequest
" depends on what kind of item you are dealing with; you may, e.g., request a table item to add or remove one or more rows, or a file item to write or to read data.
Since requesting some item to do something is such common a task, you may abbreviate it by completely omitting the "set
" command. This means that instead of "set itemhandle actionrequest
", you may at any time type the following command:
itemhandle actionrequest
So there would actually be no need of even knowing about the "set
" command, were it not for the fact that with the STx online help, you find the actions each kind of item supports under the keyword "set itemtype
" (set table
, SET DIALOG
, SET GRAPH
, and so on).
Let's have a first simple example on how shell items are normally used:
// create a table whose handle is called "myhandle" new table myhandle // set third line (first and second line will be created empty) set myhandle 2 'third line' // explicitly using "set" // replace empty first line with a string myhandle 0 'first line' // not using "set" works just as well // now replace the second line, too myhandle 1 'second line' // show the table showitem myhandle ; Showing table "myhandle" // now, just for the fun of it, delete the second row – note that, again, // either using "set" or omitting it makes no change at all myhandle myhandle 1 /Delete // finally, show the reduced table showitem myhandle; Showing table myhandle" after deleting its second row delete myhandle // don't forget to delete the item after use
In this example, we create a table item whose handle is the string "myhandle
". Any further access to this item is done with this handle. Note that there is no dollar sign when using "myhandle
", because "myhandle
" is not a variable, but an item handle. If, by mistake, we would use the expression "$myhandle
", STx would replace this expression by the contents of the shell-global variable "myhandle
" – an empty string in case of there being no such variable, and surely an undesired result. Bluntly put: Item handles and variables reside in different namespaces, and you may well have like-named variables and item handles (of course this is not at all recommended because it may lead to some degree of confusion.)
All that being said, it is, common practice not to choose a handle of one's own, but to let STx do the choosing. In this case, the automatically chosen handle is normally stored in an STx variable:
// create a table item with a handle of STX's choice (e.g. T#27) // (the asterisk instructs STx to choose a handle on its own). // the handle will be stored in a variable called "#tab" #tab := new table * // set third line (first and second line will be created empty) $#tab 2 'third line' // replace empty first line with a string $#tab 0 'first line' // now replace the second line, too $#tab 1 'second line' // show the table showitem $#tab ; Showing table "#tab" // now, just for fun, delete the second row $#tab 1 /Delete // finally, show the reduced table showitem $#tab ; Showing table "#tab" after deleting second row // old-style deletion // delete $#tab // important: delete after use // #tab := set '' // clean the variable // new-style deletion - deletes the shell item whose name is stored in a variable, // and clears the variable delete /Var #tab // no dollar here!
N.B.: Whenever actively allocating a shell item (normally with the "new
" command), you should always delete it as soon as you no longer need it. If you don't, your macro(s) will consume an unnecessarily high amount of memory which may, for large macro applications, impair performance. There is no such thing as automatic garbage collection in STx.
Shell Tables in Detail
There are three kinds of tables: (a) plain, or simple, tables, (b) parameter tables, and (c) extended tables (I do not mention directory tables here, and I do not even mention that I do not mention directory tables here). plain, or simple, tables are easiest to handle, and they are most useful for storing lists of strings, each table entry being one such string. Extended tables are such that tables more versatile than them cannot be conceived: Extended tables tables may consist of an arbitrary number of columns each of which may be a string, an STx name, an integer, or a general number. Access to each column may be done, at the programmer's discretion, either by its index or by its symbolic name. Parameter tables, in turn, are a somewhat reduced variant of extended tables, in that they store numbers only. On the other hand, parameter tables do so most efficiently, both memory-wise and runtime-wise. This makes parameter tables the preferred means of storing numerical vectors and matrices, and for doing numerical operations of the utmost complicated kinds. The following chapters will deal with each kind of table in detail.
Plain, or Simple, Tables
Plain tables are mainly useful for storing lists of strings. As always, these strings may be of arbitrary content, allowing for e.g. logically storing multiple values per row that are, in turn, retrieved by the "readstr
" or even the "readtab
" command (the READ family of commands). Be this as it may, with the availability of the more efficient extended tables for database-like structured data, and of parameter tables for structured numerical data like vectors and matrices, storing structured data in plain tables is no longer recommended.
The "new table *
" command creates a plain table item. The name of this table item will be stored in the reserved variable "#new
". Furthermore, it will be the result of the "new table *
" command. The asterisk, though strictly instructing STx to choose an internal name on its own, should be considered a part of the command and should not normally be altered.
Have a look at the following example using simple tables. The macro "tablewrite
" will create a table, fill it with a hundred strings, delete every other row, and, finally, save the table to a file. The macro "tableread
" will, in turn, re-read the table from the file.
[Macro tablewrite] // create the table #tab := new table * // the "em" command will, after displaying its second argument in an error // dialog, immediately terminate the macro, using its first argument as the // result of the macro if '$#tab[?]' != table em '-1 ; Cannot create table' // fill the table with a hundred strings for #i := 1 to $#i <= 100 step #i := int $#i+1 $#tab $#i 'This is string $#i out of 100' end // now we prove of changing mind and delete every other row for #i := 98 to $#i >= 0 step #i := int $#i-2 $#tab $#i /Delete end // show the contents of the table showitem $#tab ; See Our Table // now save the table as a text file called "table_file.txt". // the file will be stored in the STx script directory (whose path // is available in the shell-global variable "scriptDirectory") #fileName := set '$scriptDirectory\table_file.txt' // create a text file item – we will come to this in a later chapter, // here it is shown only as kind of cliffhanger #fileItem := new file * '$#fileName' /Text /Write if '$#fileItem' == * em '-1 ; failed to open the file $#fileName' // save the table to the file $#fileItem save $#tab if '$rc' > 0 em '-1 ; failed to save table to file $#fileName' // dispose of the file item (not the file on disk!) and the table delete $#fileItem $#tab exit 1 int 0 [Macro tableread] // create an empty table #tab := new table * #fileName := set '$scriptDirectory\table_file.txt' // now, create a file item for reading a text file #fileItem := new file * '$#fileName' /Text /Read if '$#fileItem' == * em -1 ; failed to open the file $#fileName $#fileItem load $#tab if '$rc' > 0 em -1 ; failed to load table from file $#fileName delete $#fileItem // show the loaded table showitem $#tab ; See Our Loaded Table delete $#tab exit 1 int 0
The above macro already makes use of file items we need to leave for later discussion. As you can see, using them for saving and loading tables is fairly easy. You might even at this stage start saving and loading tables of your own without having looked up any details about file items.
Parameter Tables
Parameter tables store numerical data only. They are optimized for numerical throughput and, hence, are the means of choice when it comes to modelling vectors and matrices.
Technically, parameter tables are a variant of the extended tables yet to be mentioned. The reason why we deal with parameter tables first is their optimal fitness for vector and matrix arithmetic. Note that extended tables, when they are all numerical, support exactly the same operations as parameter tables – in fact, numerical-only extended tables and parameter tables are completely interchangeable. Nevertheless, parameter tables are more efficient both runtime-wise and memory-wise. So, as a rule of thumb, always use parameter tables whenever doing vector and matrix arithmetic.
There are two ways of creating a parameter table:
- explicitly, by using the NEW TABLE … /Parameter command; or
- implicitly, by using the matrix or vector result of a call to eval.
Here's a simple and semantically quite meaningless example of working with parameter tables created both ways:
// create a 20-element vector initialized with all zeroes #v0 := eval fill(20,0,0) // 20 elements, first value 0, increment 0 // create and fill a vector #v1 := eval fill(30,2,4) // 30 elements, first value 2, increment 4 // create and fill a second vector #v2 := eval fill(30,3,9) // 30 elements, first value 3, increment 9 // add up both vectors #v3 := eval $#v1 + $#v2 showitem $#v3 ; v3 // show the sum vector // create a vector with 30 elements each of which is the sine // of its index (not a useful example, but useful for an example) #v4 := new table * * number:x /Parameter for #i := 0 to $#i < 30 step #i := int $#i + 1 // store the sine of the index, #i, in #a #a := eval sin($#i) // set the #i-th value of the vector to the value of #a $#v4 $#i $#a end // transpose a matrix // create the matrix to transpose – here we must explicitly create the // matrix because we are going to assign values to its // individual columns #mat1 := new table * * number:x:4 /Parameter // 4 columns // fill each of its individual columns with some values $#mat1[*,0] := eval fill(30,10,4) // fill the first column $#mat1[*,1] := eval fill(30,2,1) // fill the second column $#mat1[*,2] := eval fill(30,0.1,1.2) // fill the third column $#mat1[*,3] := eval fill(30,1.2,1.2) // fill the fourth column // now transpose mat1 and store the transposed matrix to mat2 #mat2 := eval trn($#mat1) // show the transposed matrix showitem $#mat2 ; transposed matrix // now, multiply the matrix by the vector and see what happens #mat3 := eval $#mat2 * $#v3 showitem $#mat3 ; Transposed matrix multiplied by a vector // create and initialize a matrix in a single step #mat4 := eval init(30,10,1.2) // 30 rows, 10 columns, value 1.2 showitem $#mat4 ; A matrix containing all 1.2 // important: delete all items not to leak memory delete /Var #v0 #v1 #v2 #v3 #v4 #mat1 #mat2 #mat3 #mat4 exit
STx Pitfalls
When invocating a macro for the second time, third time, or generally the n-th time, n being any finite positive integer, it will find its namespace empty. This is what you would expect from a full-featured programming language, but it is not what you get from many macro languages inferior to STx, or even from classical languages like the venerable FORTRAN, Basic, or – God forbid – COBOL.
One important implication of this fact is that local variables are not a means for storing data persistently (bluntly put, local variables are no such thing as the local "static" variables of Kernighan's and Ritchie's fine C programming language).
This consistently holds true even when invocating a macro recursively. Its new instance will find all its variables empty, while the calling instance will find its local variables untouched by the macro call.
Hence, local variables are no means at all for passing information to and from a macro. For passing data to a macro, you should use macro arguments. For a macro to return results, you should use the return mechanism (using the verbose form of the "exit
" command). Technically, another means of passing data to and from a macro is using shell-global (or even global) variables, but in the era of structured and modular programming, this is considered harmful to one's career.
Using one and the same variable name in different scopes is technically valid. You may, e.g., have a global variable called @i
, a shell-global variable i
, and a local variable called #i
. Allow for it to be misleading and a source of programmer error, though.
Misspelled variable names tend to be the cause for great confusion. What makes it worse is that there is no way of automatically detecting misspelled variable names: Since variables need not be explicitly declared nor initialized, any syntactically well-formed variable name will be valid at any stage – this is an issue all programming languages without explicit variable declarations share, among them illustrious names like FORTRAN, Basic, or PROLOG. There is no real remedy for this issue other than placing debug statements in the macro source that occasionally print out the values of some important variables, and to check the printed values against one's expectations.
More than the occasional pitfall results from the fact that the expression in an assignment is a rather complex issue. Though this expression is, in last consequence, a string (remember that with STx, all variables are being created equal, i.e. as string variables), the way this string is computed may be a surprisingly complex, heart-warmingly powerful, and sometimes even error-prone affair, bearing the potential for unexpected side-effects. That being said, preventing most kinds of trouble is easy by following two rules of thumb: (a) put everything under apostrophes; and (b) use the appropriate expression type selector when directly assigning a value (as opposed to the result of a function call or a built-in function). So, instead of writing "name := hugo
", always type "name := set 'hugo'
".
Users, especially those familiar with other programming languages, but not with macro languages like the UNIX shells, often confuse when to prefix the name of a variable with the dollar sign and when not to. The rule is easy: If you want to mention a variable, you do not use the dollar sign. If you want to use a variable, that is, if you want its occurrence be replaced by its actual contents, you do use the dollar sign. Compiled to a rule of thumb: If the variable is the target of an assignment, you do not use a dollar sign. If the variable is the source of an assignment, or if it is the argument to a function or macro call, you do use the dollar sign.
When using string constants, you should generally use quotation, i.e., place them between single quotes.
Where two or more quoted string constants meet (but see the below exception), they will get concatenated without inserting any whitespace characters. This is a standard feature with many programming languages, especially macro languages, but it is sometimes met with surprise. So please be aware that if you want two string constants to be separated by a blank, you will either need to quote the whole string, or to add the whitespace character at the beginning or at the end of the respective string literal. Hence, the statement "#var := set 'abc' 'def'
" will set #var to "abcdef", whereas each of the statements "#var := set 'abc ' 'def'
", "#var := set 'abc' ' def'
", and "#var := set 'abc def'
" will set #var to "abc def".
If a quoted string constant is an argument to a statement or to a built-in function (i.e. not to a user macro), there will be no string concatenation. Hence, each quoted string and each unquoted string is exactly one argument to the function.
If a statement or a built-in function expects a numerical argument, this argument may either be a numerical constant or a numerical expression. Either way, it is more secure to quote even a numerical argument (don't quote me on that, though).
It is easy to mix up the reserved variables "rc" and "result". The former ("rc") contains the numerical return code of the STx statement or built-in function most-recently executed (or the reason code why some STx statement could not be executed). The latter ("result") contains whatever information the most recently called user-defined macro (or class method) chose to return.
Unlike most other programming languages and many macro languages, STx evaluates logical expressions strictly from left to right. Furthermore, it does not support grouping (i.e. using brackets). Although this is not a bad thing in general, it may be cause for error and misunderstanding. When writing down a logical expression, be aware that no rules of precedence will apply. If in doubt, use a second (third, fourth…) "if
" statement within the scope of each other.
Note that whitespace characters may at any time freely be interspersed within a number. This means that e.g. the string "123 456
", although containing a blank character, will be interpreted as the number "123456". This is not a bad thing in itself, but you should be aware of this fact.
Never forget to free an object (both shell-items and user-defined classes) after use. If you don't, and automatic cleanup will only be done at the termination of the running shell which, when using a complex set of macros, may be very late.
Be careful not to mix up internal STx item handles on the one hand and variable names on the other hand. Remember that accessing an STx item by its name means just using this name – there is no dollar sign involved. If, on the other hand, your item handle is stored in an STx variable, you will need to refer to the contents of this variable, i.e. use this variable – here there is a need for using the dollar sign.