_TITLE "Math regulator demo for krovit" ' b+ start 2020-08-18 x$ = mr$("3", "/", "2") '1.5
y$ = mr$(x$, "-", ".0000000000000000000000001") '1.49999... 24 nines no rounding
PRINT "Differnce to 1.5 is "; mr$
(y$
, "-", "1.5")
FUNCTION mr$
(a$
, op$
, b$
) ' catchy? mr$ for math regulator DIM ca$
, cb$
, aSgn$
, bSgn$
, postOp$
, SGN$
op$
= _TRIM$(op$
) 'save fixing each time ca$
= _TRIM$(a$
): cb$
= _TRIM$(b$
) 'make copies in case we change 'strip signs and decimals
aSgn$
= "-": ca$
= MID$(ca$
, 2) aSgn$ = "": ca$ = ca$
ca$
= MID$(ca$
, 1, dp
- 1) + MID$(ca$
, dp
+ 1) adp = 0
bSgn$
= "-": cb$
= MID$(cb$
, 2) bSgn$ = "": cb$ = cb$
cb$
= MID$(cb$
, 1, dp
- 1) + MID$(cb$
, dp
+ 1) bdp = 0
IF op$
= "+" OR op$
= "-" OR op$
= "/" THEN 'add or subtr even up strings on right of decimal 'even up the right sides of decimals if any
dp = adp + bdp
'now according to signs and op$ call add$ or subtr$
IF aSgn$
= bSgn$
THEN 'really add postOp$ = aSgn$ + add$(ca$, cb$)
ELSE 'have a case of subtraction IF aSgn$
= "-" THEN postOp$
= subtr$
(cb$
, ca$
) ELSE postOp$
= subtr$
(ca$
, cb$
) IF bSgn$
= "-" THEN 'really add but switch b sign bSgn$ = ""
postOp$ = subtr$(cb$, ca$)
postOp$ = add$(ca$, cb$)
bSgn$ = "-"
postOp$ = aSgn$ + add$(ca$, cb$)
postOp$ = subtr$(ca$, cb$)
postOp$
= SGN$
+ mult$
(ca$
, cb$
) postOp$
= SGN$
+ divide$
(ca$
, cb$
) 'put dp back
lpop
= LEN(postOp$
) ' put decimal back postOp$
= MID$(postOp$
, 1, lpop
- dp
) + "." + MID$(postOp$
, lpop
- dp
+ 1) mr$ = trim0$(postOp$)
di$
= MID$(nInverse$
(d$
, 100), 2) 'chop off decimal point after ndi$ = mult$(n$, di$)
divide$ = trim0$(ndi$)
FUNCTION nInverse$
(n$
, DP
AS INTEGER) 'assume decimal at very start of the string of digits returned, no rounding DIM m$
(1 TO 9), si$
, r$
, outstr$
, d$
m$(i) = mult$(si$, n$)
outstr$ = ""
outstr$ = "." 'everything else n > 1 is decimal 8/17
r$ = "10"
outstr$ = outstr$ + "0" ' add 0 to the output string
r$ = r$ + "0"
outstr$ = outstr$ + d$
r$ = subtr$(r$, mult$(d$, n$)) 'r = r -d*n
r$ = r$ + "0" 'add another place
FUNCTION mult$
(a$
, b$
) 'assume both positive integers prechecked as all digits strings DIM f18$
, f1$
, t$
, build$
, accum$
'find the longer number and make it a mult of 18 to take 18 digits at a time from it
f1$ = b$
f1$ = a$
FOR dp
= LEN(f1$
) TO 1 STEP -1 'dp = digit position of the f1$ build$ = "" 'line builder
co = 0
'now taking 18 digits at a time Thanks Steve McNeill
v18
= VAL(MID$(f18$
, m
* 18 - g
* 18 + 1, 18)) build$
= MID$(t$
, 2) + build$
accum$ = build$
accum$
= add$
(accum$
, build$
+ STRING$(LEN(f1$
) - dp
, "0")) mult$ = accum$
FUNCTION subtr$
(sum$
, minus$
) ' assume both numbers are positive all digits DIM ts$
, tm$
, sign$
, LG$
, sm$
, t$
, result$
ts$ = TrimLead0$(sum$): tm$ = TrimLead0$(minus$)
IF trim0
(ts$
) = trim0$
(tm$
) THEN subtr$
= "0":
EXIT FUNCTION 'OK proceed with function knowing they are not equal tenE18 = 1000000000000000000 'yes!!! no dang E's
IF LTE
(ts$
, tm$
) THEN ' which is bigger? minus is bigger sign$ = "-"
sign$ = ""
'now taking 18 digits at a time From Steve I learned we can do more than 1 digit at a time
VB
= VAL(MID$(LG$
, m
* 18 - g
* 18 + 1, 18)) vs
= VAL(MID$(sm$
, m
* 18 - g
* 18 + 1, 18)) p = (m - g) * 18
p = p - 1
result$ = t$ + result$
subtr$ = sign$ + result$
FUNCTION add$
(a$
, b$
) 'add 2 positive integers assume a and b are just numbers no spaces or - signs 'first thing is to set a and b numbers to same length and multiple of 18 so can take 18 digits at a time
DIM fa$
, fb$
, t$
, new$
, result$
'now taking 18 digits at a time Thanks Steve McNeill
sa
= VAL(MID$(fa$
, m
* 18 - g
* 18 + 1, 18)) sb
= VAL(MID$(fb$
, m
* 18 - g
* 18 + 1, 18)) result$ = new$ + result$
add$ = result$
' String Math Helpers -----------------------------------------------
'this function needs TrimLead0$(s$)
FUNCTION LTE
(a$
, b$
) ' a$ Less Than or Equal b$ comparison of 2 strings ca$ = TrimLead0$(a$): cb$ = TrimLead0(b$)
LTE = -1
LTE = -1
LTE = 0
' ------------------------------------- use these for final display
FUNCTION TrimLead0$
(s$
) 'for treating strings as number (pos integers) copys$
= _TRIM$(s$
) 'might as well remove spaces too i = 1: find = 0
i = i + 1: find = 1
IF copys$
= "" THEN TrimLead0$
= "0" ELSE TrimLead0$
= copys$
copys$
= _TRIM$(s$
) 'might as well remove spaces too TrimTail0$ = copys$
i
= LEN(copys$
): find
= 0 i = i - 1: find = 1
TrimTail0$
= MID$(copys$
, 1, dp
- 1) TrimTail0$
= MID$(copys$
, 1, i
)
cs$ = s$
cs$ = TrimLead0$(cs$)
cs$ = TrimTail0$(cs$)
IF si$
= "-" THEN trim0$
= si$
+ cs$
ELSE trim0$
= cs$