Article: Q63095
Product(s): See article
Version(s): 4.00 4.00b 4.50
Operating System(s): MS-DOS
Keyword(s): ENDUSER | SR# S900608-167 B_BasicCom | mspl13_basic
Last Modified: 8-AUG-1990
This article provides a code example of a FUNCTION procedure that
always returns the string equivalent of a real number in the format
"WholeNumber.Fraction". This works around the usual behavior of the
STR$ function, which, depending on how a real number is represented in
binary, returns either a string in the form of "WholeNumber.Fraction"
or a string in scientific notation.
This information applies to Microsoft QuickBASIC versions 4.00, 4.00b,
and 4.50, to Microsoft BASIC Compiler versions 6.00 and 6.00b for
MS-DOS and MS OS/2, and to Microsoft BASIC Professional Development
System (PDS) versions 7.00 and 7.10 for MS-DOS and MS OS/2.
STR$ merely converts a SINGLE or DOUBLE into a string based on a real
number's internal representation. Because some real numbers cannot be
represented exactly in binary, the STR$ function returns some numbers
in scientific notation and others in normal notation. The RealString$
FUNCTION below never returns scientific notation, but always returns
normal notation.
Code Example
------------
The code example below provides a FUNCTION procedure that consistently
converts a real number into a string of the format "WholeNumber.Fraction".
Please note that this function is provided as a general example of how
to approach solving this programming problem. It represents only one
solution; others may be possible.
1. Description: The RealString$ function converts a DOUBLE-precision
number into a string, rounding the result to the specified
precision.
2. The parameters are as follows:
a. RealNumber - Any DOUBLE type value.
Note: You can also pass INTEGER, LONG, and SINGLE types in
conjunction with the BASIC keyword CDBL. Of course, STR$ with
the INTEGER and LONG types will yield the same result. For
example:
a$ = RealString$(CDBL(b!),10)
where b! is a variable of type SINGLE
b. MaxPrecision - Indicates the maximum number of decimal places
you are willing to tolerate to the right of the decimal point.
For this function to work properly, MaxPrecision MUST be greater
than or equal to the total number of fractional decimal places
of the value being passed in RealNumber. For example, if you are
calling this function with .005 in RealNumber, you MUST pass a
number no less than 3 in MaxPrecision. A number greater than 3
still returns "0.005" in RealString$. For example, passing 5
through MaxPrecision returns ".005" NOT "0.00500". Zeros are NOT
padded in the string to give a result the length of
MaxPrecision.
WARNING: If MaxPrecision is substantially greater than the
fractional decimal places in the value passed through parameter
RealNumber, extraneous results may be returned in RealString$.
For example:
a$ = RealString$(.05,30) returns
"0.0500000007450580596923828125"
3. Returns: RealString$ returns a string containing the string
equivalent of RealNumber. For real numbers greater than -1 and less
than 1, RealString$ always appends a zero in the 1's digit
position. For example, a$ = RealString$(.005,3) returns "0.005" in
a$.
Note: The maximum number of possible digits to the right of the
decimal point is determined by the parameter MaxPrecision. (See
above.)
'******************** RealString FUNCTION **************************
FUNCTION RealString$ (RealNumber AS DOUBLE, MaxPrecision AS INTEGER)
IF MaxPrecision < 0 OR MaxPrecision > 50 THEN
RealString$ = ""
ELSE
' Break the RealNumber into its whole and decimal parts
whole& = FIX(RealNumber#)
decimal# = ABS(RealNumber# - whole&)
' Add in a rounding factor. This compensates for numbers
' such as .05 that can't be stored exactly in binary. For
' example, .05 is represented as .04999. This is the means
' to round it up to .05 as it should be.
decimal# = decimal# + 5 * 10 ^ -(MaxPrecision% + 1)
IF decimal# >= 1 THEN
whole& = whole& + 1
END IF
' Convert the fractional part to a string
DecString$ = ""
FOR i% = 1 TO MaxPrecision + 1
DecDigit% = FIX(decimal# * 10)
DecString$ = DecString$ + RIGHT$(STR$(DecDigit%), 1)
decimal# = decimal# * 10 - FIX(decimal# * 10)
NEXT i%
' Ensure that the fractional part is equal
' in length to the maximum precision allowed
IF LEN(DecString$) > MaxPrecision THEN
DecString$ = LEFT$(DecString$, MaxPrecision%)
' Remove all trailing zeros from the end of the
' fractional part
i% = LEN(DecString$)
WHILE i% > 0
IF MID$(DecString$, i%, 1) = "0" THEN
DecString$ = LEFT$(DecString$, i% - 1)
ELSE
i% = 0
END IF
i% = i% - 1
WEND
END IF
' Include the decimal point before the fraction
IF DecString$ <> "" THEN
DecString$ = "." + DecString$
END IF
' Compose a string of the whole and fractional parts of
' the real number
RealString$ = STR$(whole&) + DecString$
END IF ' (END of IF from very top of function)
END FUNCTION
Sample Program Using This Function
----------------------------------
DECLARE FUNCTION RealString$ (RealNumber AS DOUBLE, MaxPrecision AS_
INTEGER)
CLS
PRINT RealString$(3.1415927, 7)
PRINT RealString$(437.137827#, 5)
PRINT RealString$(.000173, 10)
PRINT RealString$(99.997,2)
END
Sample Output
-------------
3.14157927
437.13783
0.000173
100