|
MFCalc
Martin Carradus takes us through his multi function calculator
!MFCalc performs numerical calculations on expressions typed into its Input slot when the 'Calculate' button is clicked upon. The slot may be cleared out either by typing Control with 'U', or by clicking on the 'Clear Entry' button.
The following can be typed into the 'Input' slot:-
Remarks
Remarks begin with '/*' and end with '*/', and are totally ignored by !MFCalc.
Numerics
These are of Five types:
- Integers: e.g. 1234
- Exponented Integers: e.g. 1234E1 or 1234e+2
- Hexadecimal constants, which begin with '&', '&h' or '&H': e.g. &H1ab or &HADD. They are case insensitive.
- Octal constants, which begin with '&o' or '&O': e.g. &O0137
- Floating constants: They always have a decimal point in them and can also be exponented: e.g. .1234 or 1.25 or 2.3456e-6 or 0.001E+5
Variables
A named storage location containing a numeric value.
These always begin with an alpha character or underscore, then continue with any alphanumeric or underscore. These are case sensitive and become available with the value zero wherever they are mentioned. Integer variables end with a '%', otherwise they are taken to be floating point, unless declared otherwise (see below). They also can be of any length up to the 256 character limit of the Input slot.
Examples: 'a' or 'A%' or '_HELLO' or 'var1%' or 'non_zero'.
Assignment of Variables
This is simply done by typing the variables name then '=' then some numeric expression to the right of that, which in turn may contain variable names. e.g. num0=1.0 or A%=B%+6 or _hello=_goodbye
These statements may optionally be preceded by the key word 'LET', in upper or lower case.
e.g. LET a=0.0 or let no%=2
Note that if you assign an integer variable a floating value, then it will be truncated to an integer.
e.g. let a%=-2.5 gives a% the value -2
Multiple Assignments
Simply concatenate the assignments together with a colon between.
Example: a%=b%*6:let c%=a%+4:d=c%^2
Operators
You have already seen some of these above, they are:
- Addition and Subtraction: '+' and '-' e.g. 1+2 or 4.0-2.0
- You may also negate expressions: e.g. a%=-b%
- Multiplication and Division: '*' and '/' e.g. 6*7 or 1.0/2.0
!MFCalc will complain about division by zero by outputting a comment
if this happens.
Integer Division: 'div' or 'DIV', performs the division and then
truncates it to an integer. Also complains about division by zero.
e.g. '5.0 div 2.0' gives the value '2'.
Modular Arithmetic: 'mod' or 'MOD', gives the remainder upon division.
e.g. 5 MOD 2 gives the value '1'. Complains about MOD with respect to
zero.
Lastly raising a% to the power b% is denoted by 'a%^b%', and, unlike
the rest binds to the right, so 'a%^b%^c%' is taken as 'a%^(b%^c%)'.
So 3^1^2 is 3, not 9.
!MFCalc complains about raising zero to a zero or negative power.
Also beware of raising negative quantities to fractional powers.
As you can see, these expressions may be bracketed, but otherwise the
precedence of these operators is in reverse order to the list above,
going from performing powers at highest precedence, to addition and
subtraction at lowest precedence.
In all this, if !MFCalc can't make sense of the input line, it outputs
a 'Syntax Error' comment.
Combinations
The number of ways of picking r objects from n without regard
to the order of the objects is usually denoted by:-
n
C
r
The 'combination' of r objects from n.
e.g. The number of ways of picking 1 object from 5 is 5,
but the number of ways of picking 2 objects from 5 is 10.
The operator in !MFCalc you use is 'NCR' or 'ncr', so
'5 ncr 1' yields a value of 5,
and '5 ncr 2' yields a value of 10.
You may use floating values, but !MFCalc complains if they
have any fractional places. !MFcalc also complains if either
of the values supplied is negative, or if the second value exceeds
the first.
If you exceed several hundred for either of the arguments, you will
get a message about 'Stack Overflow', as recursion is employed.
Lastly NCR binds the least of all the operators, so all calculations
on both sides are performed first before the combination is taken.
e.g. '6-1 ncr 3-1' will give you the value of '5 ncr 2'.
NB NCR is not associative, so !MFcalc will error multiple NCRs
in one expression e.g. '12 NCR 2 NCR 2' will cause a syntax error.
Minimum and Maximum of Values
The operators 'MIN' and 'min' or 'MAX' and 'max' between two
expressions give the minimum or maximum of the two values e.g.
'2 min 1' gives a value of 1, but '2 max 1' gives a value of 2
These operators bind more tightly than addition or subtraction, but
less than multiplication and division or any higher operator, so,
'2 min 1 + 3 min 2', will give the sum '1 + 2' giving 3,
but '2*6 min 3*8' will produce '12 min 24', giving 12.
Also they are left associative, so combinations of 'min' and 'max' are
obeyed from left to right e.g.
'2 max 1 min 4' becomes '2 min 4', which becomes 2
You will note that you can obtain the minimum or maximum of more than
two values by simply concatenating them between the same operator e.g.
'x% min y% min z%' gives the minimum of x% and y% and z% and
'x% max y% max z%' gives the maximum of x% and y% and z%.
Both 'min' and 'max' attempt to keep the resulting value as an integer
whenever possible e.g.
'2.0 max 4.0' will produce the integer value of '4', not '4.0',
but '2 max 4.7' will produce the floating value '4.7'
Lowest Common Multiple (LCM) and Greatest Common Divisor (GCD)
The Greatest Common Divisor of two Numbers is the largest number less
than or equal to them that divides into both numbers exactly.
e.g. '6 gcd 21' gives 3, as 6=3*2 and 21=3*7, so 3 is the largest
common factor. !MFCalc uses a method called 'Euclid's Algorithm' to
obtain this number. If two numbers share no common factor greater than
or equal to two, then their GCD is one. This is true particularly when
both are prime.
The Lowest Common Multiple of two numbers is the lowest number that
shares the factors of both numbers.
e.g. '6 lcm 21' gives 42=3*2*7, as 6=3*2 and 21=3*7, 42 is the
smallest number that shares the factors of both numbers.
Note that that the LCM times the GCD of a and b is equal to (a times b)
for all a and b.
LCM and GCD both complain if either of the supplied numbers is zero or
negative or fractional. Both, also, always give out integer results.
(No decimal places shown.)
The operators 'gcd' and 'lcm' are on a par with 'min' and 'max' in
terms of priority - higher than addition or subtraction, but lower
than multiplication or division or raising to powers.
e.g. '3*2 gcd 3*7', will give the answer '3', as the multiplication is
performed first before the 'gcd'.
These operators are also associative, so you may obtain the LCM of x%,
y% and z%, by typing 'x% lcm y% lcm z%' in the 'Input:' slot and
clicking on 'Calculate'. If there is no operator of a higher
priority,'lcm' and 'gcd' are obeyed from left to right.
Mathematical Functions
There are twenty one of these available, and if the arguments are out of
range, !MFCalc complains. These functions need not have the
input expression in brackets and bind more tightly than any of the
operators. With trigonometric functions there are buttons on the
'SetUp' panel to select calculation in degrees, radians or grads.
- 'ABS' or 'abs' - Absolute positive value without the sign.
- 'ACS' or 'acs' - Arccosine.
- 'ASN' or 'asn' - Arcsine.
- 'ATN' or 'atn' - Arctangent.
- 'COS' or 'cos' - Cosine of the Angle given.
- 'COSH' or 'cosh' - Hyperbolic cosine.
- 'EXP' or 'exp' - Raising e to the given power.
- 'FRAC' or 'frac' - Fractional part of the given number. Negative if the number is negative.
- 'INT' or 'int' - Truncate to the integer below.
- 'FIX' or 'fix' - Round to nearest integer. If negative, rounds down, if positive, rounds up.
- 'LN' or 'ln' - Natural logarithm to base e.
- 'LOG' or 'log' - Logarithm to base ten.
- 'RND', 'rnd', "RAND", or "rand" - Gives out a random number in the range 0 - 1 (inclusive). Does not take in any value as an argument.
- 'SGN' or 'sgn' - The 'sign' of the number, -1 if negative, 0 if zero, +1 if positive.
- 'SIN' or 'sin' - Sine of supplied angle.
- 'SINH' or 'sinh' - Hyperbolic sine.
- 'SQR' or 'sqr' - Square root, complains if supplied number is negative. Additionally, 'SQRT' or 'sqrt' will also be accepted.
- 'TAN' or 'tan' - Tangent of supplied angle.
- 'TANH' or 'tanh' - Hyperbolic tangent.
- 'CBR' or 'cbr' - Cube root, turns negative if supplied number is negative. Additionally, 'CBRT' or 'cbrt' will also be accepted.
- 'LNFACT' or 'lnfact' - The natural logarithm of the factorial of the supplied argument. The factorial of a number is the product of the number with all the ones below it down to 1, so 3 factorial is 3.2.1=6 and 4 factorial is 4.3.2.1=24. The natural logarithm is taken because factorials grow large very quickly, so the logarithm slows this process down and stops numeric overflow at higher values. As with 'ncr', very large arguments cause a 'Stack Overflow' message.
In addition, the key word 'PI' or 'pi' in a calculation gives out the
numeric value of the trigonometric constant pi.
All of these function names are reserved words and should not be used
for variable names.
As already mentioned, these functions bind more tightly than any other
operator, so, for example, the value of 'SIN 90 +2' is 3.0, not the
Sine of 92, when in degree mode. Note that 'SIN' is separated from 90
so that the reserved word 'SIN' can be recognised. If you had typed
'SIN90+2', a variable name 'SIN90' would be created, given a floating
value of 0.0 and 2.0 added, so 2.0 would be output. Note, also, that
if you had really wanted the Sine of 90+2, you should have used
brackets and put 'SIN(90+2)'.
The Form of the Calculated Value
The form (integer or floating point (double precision)) of the
expression being calculated is always at the level of the highest
operand, so, 2+3 gives an integer value of 5, and 5/2 gives an integer
value of 2, but 2+3.0 gives a floating value of 5.0, and 5/2.0 gives a
floating value of 2.5. If you had meant an integer divide to be
performed, then you should have used 'DIV', i.e. '5 DIV 2.0' gives out
an integer value of 2.
Getting back the Last Calculated Value
The special variable 'ANS' or 'ans' is assigned the value of the last
calculation. If this is an integer, ANS takes an integer value, if
this is a floating value, ANS is floating, otherwise ANS starts with
an integer value of 0. Like any other variable, ANS may be assigned a
value. e.g. let ANS=2.5 , but it will not appear in the list produced
by the LIST instruction (see below).
Declarations
Additionally you may declare variables to be integer or floating
(double precision) using the key words 'INTEGER' or 'DOUBLE'
respectively followed by a comma separated list of variable names of
any type, either integer or float. You may also assign these variable
names by using '='.
e.g.
INTEGER a, b, c=3
or
DOUBLE A%=1.2,D,e=2.71828
You may not concatenate these declarations.
PRIME Keyword
Prime numbers are not divisible by anything other than 1 and themselves.
If a number is not prime, it can be represented as the product of
primes and powers of primes less than itself, called factors. The
number is said to be 'composite'.
'PRIME' or 'prime' followed by any expression will determine if the
integer produced is a prime number. Complains if the number is
fractional. If the number is negative, the sign is removed before
prime determination. If the number is floating point, it is
integerised, but complains if the result would be too large for an
integer. Indicates if the number is zero or +1 or -1 or even. This is
done in integer arithmetic, so !MFCalc may crash if the number is
too large.
Gives the smallest factor if the number is not prime.
e.g. 'PRIME 2^31-1' will produce the REM '/* 2147483647 is Prime */'.
'PRIME 13*17' will produce the REM '/* 221 is not Prime, Factor 13 */'.
'PRIME' or 'prime' followed by an expression can also be put,
separated by a colon, at the very end of a colon-separated list of
assignments and 'input' statements (see below).
e.g. A 'Mersenne' prime is one of the form '2^n-1', so by repeating
the line: 'input a%:prime 2^a%-1', using 'Entry Back' and 'Calculate',
you may investigate for what values of a%, 2^a%-1 is prime.
Lastly, PRIME sets ANS to the value of the found lowest factor of the
supplied expression, or to the value 1 if the expression is prime. So,
for example, start with the line 'm%=1:ANS=1'. Then repeat the line:
'm%=m%*ANS:PRIME <expression>:/m%', where <expression>: is the
expression you wish to factorise, and continue until you get a prime.
You will then have found all the prime factors of <expression>:.
Getting Rid of all the Variables
The command 'CLEAR' or 'clear' will remove all the variables and start
from scratch again.
Getting Rid of One Variable
The command 'CLEAR' or 'clear' followed by the variable name will cause
the variable to be removed from the list of defined variable names.
e.g. CLEAR a%
A message indicates if the variable name was already undefined.
Checking the Variables Present
The single command 'LIST' or 'list' and clicking 'Calculate' will
cause an alphabetical list of the variable names and values to appear
in the Input Slot. After each line, press the Return (Enter) key and a
new line appears, or a message indicating the end of the list.
Pressing the Escape key at any time terminates the list and a message
indicating that appears. A message also indicates when no variables
have been defined. The list also shows if the variable has not been
explicitly assigned a value. Also the form of representation of the
assigned value in the list gives the variable's data type - it is
represented as an integer without a decimal point if the variable is
to be taken as holding an integer value - with a decimal point if it
is a floating variable e.g. a variable with the a value of six would
be listed with a value '6' if it was an integer variable, but '6.0' if
it was a floating (double precision) variable.
The Field Length and Number of Decimal Places
Just type numeric values into the slots provided, then press
the 'Enter' key (Return). !MFCalc requires that you press 'Enter'
before these values are recognised. Clearing them out to nothing just
leaves the values unaltered. !MFCalc complains if you try to make the
field length less than the number of places, or the number of places
more than the field length. Not available in hexadecimal or octal
notation. Also the choice of representation in floating, exponential
or general format is only available in decimal notation, not hex or
octal.
The INPUT command
The key word 'INPUT' followed by a single variable name e.g. INPUT a%
will cause a prompt REM of the form:- '/* Type a%, Enter or Escape */
to appear in the 'Input:' slot together with a red caret. Typing the
value at the caret and pressing 'Enter' ('Return') will assign the
variable a% with the typed literal. Pressing the 'Escape' key at this
point leaves the variable unassigned with its default value of zero.
Note that !MFCalc will only accept literals e.g. 123, 2.31, &123,
&o127, 1E2, 0.5e-04 or a single variable name at the prompt, not whole
expressions. The only exception is that you may put a minus '-' sign
in front.
Also Note pressing the 'Backspace' or 'Delete' key, if you make a
mistake, will erase the mistaken entry, and assign the variable the
new value on pressing 'Enter' ('Return'). Pressing the 'Escape' key,
as mentioned above, aborts the current INPUT statement. In this case,
no new value is assigned. This is useful if you do not wish to alter
the value of an INPUT variable. If you use the cursor keys to move
backwards and forwards in the 'Input:' slot, they will have no effect.
Inherited Unassignment: If you use INPUT and enter an unassigned
variable name at the prompt, then the INPUT variable remains
unassigned.
INPUT statements can be concatenated, so, you may put e.g.
INPUT a%:INPUT b%:INPUT c%
And !MFCalc will prompt for the value a%, then b%, then c%
The INPUT statement allows you to give your own prompt by supplying a
double quoted character string, which is converted to your own REM before
the prompt caret e.g. 'INPUT "Give Value"; a%' will give:
'/* Give Value */ followed by a prompt for the value. If you
supply '/*' or '*/' in the quoted string, they are removed from the
REM. If you require a double quote, supply "" within the string, which
is converted to a " within your REM. The semi-colon afterwards, when
supplied, suppresses a '?' from being added to the REM.
The PRINT Command
'PRINT' followed by any expression simply gives the value of the
expression when the 'Calculate' button is clicked on. The advantage is
that this command can occur after a colon at the end of any series of
assignments or INPUTs e.g.
'INPUT a%:b%=1:INPUT c%:PRINT a%+b%+c%' will prompt for the values of
a% and c% then give the value of a%+b%+c%. PRINT should only occur at
the end of an 'Input:' line.
Similarly to INPUT, PRINT can be followed by a string of characters in
double quotes before the expression. The string will be output as a
comment before the value of the expression when 'Calculate' is clicked
on. You may separate the string from the expression by a semi-colon,
but it has no effect.
As described below, you may bring this line back and so perform a
calculation many times over with different input values.
The 'Entry Back' and 'Entry Forward' Buttons
Every time you click on 'Calculate', the 'Input:' slot value is stored
away on a stack. The 'Entry Back' button, when clicked,
successively brings back previous 'Input:' lines into the 'Input:'
slot. There is enough storage to hold up to twenty previous lines.
When this is exceeded, the 'stack' shifts up, so lines more than
twenty ago are lost.
The 'Entry Forward' button moves forward in time right up to the
current line, reversing the effect of 'Entry Back'.
Neither of these buttons have any effect until you have entered
several lines.
If you go back to a previous line, then enter a different line and
click on 'Calculate', then immediately the entries forward in time
from the chosen line are lost, but those previous in time to the
chosen line can still be obtained by clicking on the 'Entry Back'
button.
'Input:' Lines are not recorded in the memory stack if they are blank
or they repeat the same calculation as the previous entry on the
stack. This means you can repeat previous calculations without it
affecting the previous line memory stack.
Lastly, the 'Clear Entries' button completely clears out the memory
stack and all knowledge of any previous lines is lost. Remember to
distinguish this from the 'Clear Entry' button, that just clears out
the 'Input:' slot and has no effect on the memory stack.
The 'Save Entries' and 'Retrieve Entries' Buttons
The 'Save Entries' Button copies all the entries in the memory stack
to an internal file, which can be brought back by clicking on the
'Retrieve Entries' Button, even after you have Quit the whole
application. Both buttons give out a message indicating how many
entries were Saved/Retrieved, but in the case of 'Retrieve Entries',
Press the 'Enter' ('Return') key afterwards. The entry you will then
see in the 'Input:' slot is the last one you were looking at when you
Saved the Entries.
If you have a constant set of calculations you perform, this saves
having to type them in time and again.
Getting Out of the Application
You may select 'Quit' from both the icon bar menu and the 'Help' menu
to terminate the application. Selecting the 'Close' icon on the
'SetUp' panel simply removes the panel and does not clear any of the
variable names, but typing 'QUIT' or 'quit' in the 'Input' slot and
clicking 'Calculate' both removes the panel and clears the variable
names. Lastly, typing 'EXIT' or 'exit' and clicking 'Calculate'
completely terminates the application. In certain circumstances, the
'Escape' key has the same effect as 'QUIT'.
Bugs
1. It has been discovered that the value returned for the Sine of 90
degrees is marginally greater than one, so if you immediately perform
an Arc Sine on this value, !MFCalc objects to the number input to Arc
Sine being greater than one. i.e. ASN SIN 90 in degree mode produces
an error message, not a return value of 90. The same is true of SIN -90.
The same is not true of the COS of zero.
2. If the resulting value of integer arithmetic with '+', '-', '*' or
'^' overflows the 32 bits used for its (signed) representation, then
the calculation is converted to a floating one. The same is not true
for floating calculations. If you wish to calculate with large
numbers, then you could use floating arithmetic e.g. represent one
hundred million as '100000000.0'. However, if any integer literal in a
calculation exceeds about 8 digits, it is automatically converted so
as to be taken as floating literal and the whole calculation is
performed as a floating one anyway.
Integer '/', MOD or DIV are calculated in integer storage. If the
numbers involved are large (roughly over 8 digits), it can cause
integer overflow and !MFCalc crashes.
Also realise that even if the calculation is done in floating point
format, because of the way the number is stored, only 16 - 17
significant digits can be held. If, for example, the result of the
calculation exceeds 10^17, then only the first 16 - 17 digits will
appear before the decimal point, the rest being zero.
3. For various reasons, it is best to perform repeated raising to
powers (using '^') as all integer as all float (not mixed arithmetic).
e.g. 3 ^ 2 ^ 2 or 3.0 ^ 2.0 ^ 2.0, but not 3 ^ 2.0 ^ 2.0
However, with a single '^', any mixture of integer or float is
acceptable. As mentioned above, if the two operands are integer, the
result is an integer, but otherwise the result is floating.
Also beware of large numeric values (overflow) developing during
repeated exponentiation. It can produce very odd output results,
particularly when displaying in 'Floating' mode.
Reserved Words
A list of the reserved words (both in upper and lower case) used by
!MFCalc follow. They should not be used as variable names. If you get
an odd result or a syntax error, then you are probably using a
reserved word in the wrong place, or think you are using one when
you have typed it incorrectly or not separated it properly from other
alphanumerics. Note the reserved words are either all upper case or
all lower case and not any mixture:
"ABS",
"ACS", "ANS", "ASN", "ATN", "CBR", "CBRT", "CLEAR", "COS", "COSH", "DIV", "DOUBLE", "EXIT", "EXP", "FIX", "FRAC", "GCD", "INPUT", "INT", "INTEGER", "LCM", "LET", "LIST", "LN", "LNFACT", "LOG", "MAX", "MIN", "MOD", "NCR", "PI", "PRIME", "PRINT", "QUIT", "RAND", "REM", "RND", "SGN", "SIN", "SINH", "SQR", "SQRT", "TAN", "TANH", "abs", "acs", "ans", "asn", "atn", "cbr", "cbrt", "clear", "cos", "cosh", "div", "double", "exit", "exp", "fix", "frac", "gcd", "input", "int", "integer", "lcm", "let", "list", "ln", "lnfact", "log", "max", "min", "mod", "ncr", "pi", "prime", "print", "quit", "rand", "rem", "rnd", "sgn", "sin", "sinh", "sqr", "sqrt", "tan", "tanh".
Eighty eight reserved words in all, forty four in lower and forty four in upper case.
Have fun!
Queries, curses, praise to:
Martin Carradus, Leaf Mindcraft, c/o 27 Wells Road, Ilkley, West Yorkshire, LS29 9JE.
Send s.a.e. if you require a reply.
NB. Please read the Text file !ReadMe within this application for conditions of use.
(Hold down 'Shift' and double click with the Mouse over the !MFCalc icon to find it.)
Martin Carradus July 2004.
|