note description: "Decimal coefficients. They hold the significant digits. Can be seen as arrays of decimal values. Zero-based index. - Index '0' is the least significant digit, - index 'count-'1 is the most significant digit." library: "Gobo Eiffel Decimal Arithmetic Library" copyright: "Copyright (c) 2004, Paul G. Crismer and others" license: "MIT License" date: "$Date: 2016-05-06 19:15:38 +0000 (Fri, 06 May 2016) $" revision: "$Revision: 98678 $" deferred class MA_DECIMAL_COEFFICIENT feature {NONE} -- Initialization default_create -- Process instances of classes with no creation clause. -- (Default: do nothing.) -- (from ANY) do end make (a_capacity: INTEGER_32) -- Make with a_capacity. require a_capacity_positive: a_capacity > 0 deferred ensure capacity_set: capacity >= a_capacity count_zero: count = 0 lower_set: lower = 0 upper_set: upper = capacity - 1 end make_copy (other: like Current) -- Make a copy of other. require other_not_void: other /= Void deferred ensure capacity_set: capacity >= other.capacity equal_other: is_equal (other) end feature -- Access generating_type: TYPE [detachable MA_DECIMAL_COEFFICIENT] -- Type of current object -- (type of which it is a direct instance) -- (from ANY) external "built_in" ensure -- from ANY generating_type_not_void: Result /= Void end generator: STRING_8 -- Name of current object's generating class -- (base class of the type of which it is a direct instance) -- (from ANY) external "built_in" ensure -- from ANY generator_not_void: Result /= Void generator_not_empty: not Result.is_empty end item (index: INTEGER_32): INTEGER_32 -- Item at index require valid_index: valid_index (index) deferred end msd_index: INTEGER_32 -- Index of most significant (non-zero) digit deferred ensure msd_index_small_enough: Result < count msd_index_large_enough: Result >= 0 index_of_msd_or_zero: Result > 0 implies item (Result) /= 0 end feature -- Measurement capacity: INTEGER_32 -- Capacity deferred ensure definition: capacity >= count end count: INTEGER_32 -- Number of decimal digits deferred end lower: INTEGER_32 -- Lower index deferred ensure definition: Result = 0 end subcoefficient (index_start, index_end: INTEGER_32): MA_DECIMAL_COEFFICIENT -- Subcoefficient made of digits in range [index_start..index_end] require index_start_big_enough: index_start >= 0 index_end_big_enough: index_end >= index_start index_end_small_enough: index_end <= count - 1 deferred ensure subcoefficient_not_void: Result /= Void end upper: INTEGER_32 -- Upper index deferred ensure definition: Result <= capacity - 1 end feature -- Comparison frozen deep_equal (a: detachable ANY; b: like arg #1): BOOLEAN -- Are a and b either both void -- or attached to isomorphic object structures? -- (from ANY) do if a = Void then Result := b = Void else Result := b /= Void and then a.is_deep_equal (b) end ensure -- from ANY instance_free: class shallow_implies_deep: standard_equal (a, b) implies Result both_or_none_void: (a = Void) implies (Result = (b = Void)) same_type: (Result and (a /= Void)) implies (b /= Void and then a.same_type (b)) symmetric: Result implies deep_equal (b, a) end frozen equal (a: detachable ANY; b: like arg #1): BOOLEAN -- Are a and b either both void or attached -- to objects considered equal? -- (from ANY) do if a = Void then Result := b = Void else Result := b /= Void and then a.is_equal (b) end ensure -- from ANY instance_free: class definition: Result = (a = Void and b = Void) or else ((a /= Void and b /= Void) and then a.is_equal (b)) end frozen is_deep_equal (other: MA_DECIMAL_COEFFICIENT): BOOLEAN -- Are Current and other attached to isomorphic object structures? -- (from ANY) require -- from ANY other_not_void: other /= Void external "built_in" ensure -- from ANY shallow_implies_deep: standard_is_equal (other) implies Result same_type: Result implies same_type (other) symmetric: Result implies other.is_deep_equal (Current) end is_equal (other: MA_DECIMAL_COEFFICIENT): BOOLEAN -- Is other attached to an object of the same type -- as current object and identical to it? -- (from COMPARABLE) require -- from ANY other_not_void: other /= Void deferred ensure -- from ANY symmetric: Result implies other ~ Current consistent: standard_is_equal (other) implies Result ensure then -- from COMPARABLE trichotomy: Result = (not (Current < other) and not (other < Current)) end is_greater alias ">" (other: like Current): BOOLEAN -- Is Current greater than other? require -- from PART_COMPARABLE other_exists: other /= Void do Result := (three_way_comparison (other) = 1) ensure then -- from COMPARABLE definition: Result = (other < Current) end is_greater_equal alias ">=" (other: MA_DECIMAL_COEFFICIENT): BOOLEAN -- Is current object greater than or equal to other? -- (from COMPARABLE) require -- from PART_COMPARABLE other_exists: other /= Void do Result := not (Current < other) ensure then -- from COMPARABLE definition: Result = (other <= Current) end is_less alias "<" (other: like Current): BOOLEAN -- Is Current less than other? require -- from PART_COMPARABLE other_exists: other /= Void do Result := (three_way_comparison (other) = -1) ensure then -- from COMPARABLE asymmetric: Result implies not (other < Current) end is_less_equal alias "<=" (other: MA_DECIMAL_COEFFICIENT): BOOLEAN -- Is current object less than or equal to other? -- (from COMPARABLE) require -- from PART_COMPARABLE other_exists: other /= Void do Result := not (other < Current) ensure then -- from COMPARABLE definition: Result = ((Current < other) or (Current ~ other)) end max (other: MA_DECIMAL_COEFFICIENT): MA_DECIMAL_COEFFICIENT -- The greater of current object and other -- (from COMPARABLE) require -- from COMPARABLE other_exists: other /= Void do if Current >= other then Result := Current else Result := other end ensure -- from COMPARABLE current_if_not_smaller: Current >= other implies Result = Current other_if_smaller: Current < other implies Result = other end min (other: MA_DECIMAL_COEFFICIENT): MA_DECIMAL_COEFFICIENT -- The smaller of current object and other -- (from COMPARABLE) require -- from COMPARABLE other_exists: other /= Void do if Current <= other then Result := Current else Result := other end ensure -- from COMPARABLE current_if_not_greater: Current <= other implies Result = Current other_if_greater: Current > other implies Result = other end frozen standard_equal (a: detachable ANY; b: like arg #1): BOOLEAN -- Are a and b either both void or attached to -- field-by-field identical objects of the same type? -- Always uses default object comparison criterion. -- (from ANY) do if a = Void then Result := b = Void else Result := b /= Void and then a.standard_is_equal (b) end ensure -- from ANY instance_free: class definition: Result = (a = Void and b = Void) or else ((a /= Void and b /= Void) and then a.standard_is_equal (b)) end frozen standard_is_equal (other: MA_DECIMAL_COEFFICIENT): BOOLEAN -- Is other attached to an object of the same type -- as current object, and field-by-field identical to it? -- (from ANY) require -- from ANY other_not_void: other /= Void external "built_in" ensure -- from ANY same_type: Result implies same_type (other) symmetric: Result implies other.standard_is_equal (Current) end three_way_comparison (other: MA_DECIMAL_COEFFICIENT): INTEGER_32 -- If current object equal to other, 0; -- if smaller, -1; if greater, 1 -- (from COMPARABLE) require -- from COMPARABLE other_exists: other /= Void do if Current < other then Result := -1 elseif other < Current then Result := 1 end ensure -- from COMPARABLE equal_zero: (Result = 0) = (Current ~ other) smaller_negative: (Result = -1) = (Current < other) greater_positive: (Result = 1) = (Current > other) end feature -- Status report conforms_to (other: ANY): BOOLEAN -- Does type of current object conform to type -- of other (as per Eiffel: The Language, chapter 13)? -- (from ANY) require -- from ANY other_not_void: other /= Void external "built_in" end is_one: BOOLEAN -- Is Current one? do Result := (msd_index = 0 and then item (0) = 1) ensure definition: Result = (msd_index = 0 and then item (0) = 1) end is_significant: BOOLEAN -- Has Current any non-zero digit? do Result := not is_zero ensure definition: Result = not is_zero end is_zero: BOOLEAN -- Is this coefficient only composed of zeros? local i: INTEGER_32 do Result := True from i := count - 1 until i < 0 loop if item (i) /= 0 then Result := False i := -1 else i := i - 1 end end end same_type (other: ANY): BOOLEAN -- Is type of current object identical to type of other? -- (from ANY) require -- from ANY other_not_void: other /= Void external "built_in" ensure -- from ANY definition: Result = (conforms_to (other) and other.conforms_to (Current)) end valid_index (index: INTEGER_32): BOOLEAN -- Is index valid ? do Result := (index >= 0 and then index < count) ensure definition: Result = (index >= 0 and then index < count) end feature {MA_DECIMAL} -- Status setting set_from_substring (s: STRING_8; coefficient_begin, coefficient_end: INTEGER_32) -- Set from s, skip the decimal point if it is present. require s_not_void: s /= Void coefficient_begin: coefficient_begin >= 1 coefficient_end: coefficient_end <= s.count coefficient_end_ge_begin: coefficient_end >= coefficient_begin deferred end feature {MA_DECIMAL, MA_DECIMAL_PARSER, MA_DECIMAL_COEFFICIENT} -- Element change grow (a_capacity: INTEGER_32) -- Grow coefficient so that it can contain up to a_capacity digits. require a_capacity_large_enough: a_capacity >= count deferred ensure new_capacity: capacity >= a_capacity adapted_count: count = a_capacity end put (v, index: INTEGER_32) -- Put v at index-th item. require valid_index: index >= 0 and index < capacity valid_v: v >= 0 and v <= 9 deferred ensure item_set: item (index) = v adapted_count: (index + 1 > old count) implies count = index + 1 end feature -- Duplication frozen clone (other: detachable ANY): like other obsolete "Use `twin' instead. [2017-05-31]" -- Void if other is void; otherwise new object -- equal to other -- -- For non-void other, clone calls copy; -- to change copying/cloning semantics, redefine copy. -- (from ANY) do if other /= Void then Result := other.twin end ensure -- from ANY instance_free: class equal: Result ~ other end copy (other: MA_DECIMAL_COEFFICIENT) -- Update current object using fields of object attached -- to other, so as to yield equal objects. -- (from ANY) require -- from ANY other_not_void: other /= Void type_identity: same_type (other) deferred ensure -- from ANY is_equal: Current ~ other end frozen deep_clone (other: detachable ANY): like other obsolete "Use `deep_twin' instead. [2017-05-31]" -- Void if other is void: otherwise, new object structure -- recursively duplicated from the one attached to other -- (from ANY) do if other /= Void then Result := other.deep_twin end ensure -- from ANY instance_free: class deep_equal: deep_equal (other, Result) end frozen deep_copy (other: MA_DECIMAL_COEFFICIENT) -- Effect equivalent to that of: -- copy (other . deep_twin) -- (from ANY) require -- from ANY other_not_void: other /= Void do copy (other.deep_twin) ensure -- from ANY deep_equal: deep_equal (Current, other) end frozen deep_twin: MA_DECIMAL_COEFFICIENT -- New object structure recursively duplicated from Current. -- (from ANY) external "built_in" ensure -- from ANY deep_twin_not_void: Result /= Void deep_equal: deep_equal (Current, Result) end frozen standard_clone (other: detachable ANY): like other obsolete "Use `standard_twin' instead. [2017-05-31]" -- Void if other is void; otherwise new object -- field-by-field identical to other. -- Always uses default copying semantics. -- (from ANY) do if other /= Void then Result := other.standard_twin end ensure -- from ANY instance_free: class equal: standard_equal (Result, other) end frozen standard_copy (other: MA_DECIMAL_COEFFICIENT) -- Copy every field of other onto corresponding field -- of current object. -- (from ANY) require -- from ANY other_not_void: other /= Void type_identity: same_type (other) external "built_in" ensure -- from ANY is_standard_equal: standard_is_equal (other) end frozen standard_twin: MA_DECIMAL_COEFFICIENT -- New object field-by-field identical to other. -- Always uses default copying semantics. -- (from ANY) external "built_in" ensure -- from ANY standard_twin_not_void: Result /= Void equal: standard_equal (Result, Current) end to_twin: like Current -- Cloned version of Current deferred ensure twin_not_void: Result /= Void twin_not_current: Result /= Current tiwn_equal_current: Result.is_equal (Current) end frozen twin: MA_DECIMAL_COEFFICIENT -- New object equal to Current -- twin calls copy; to change copying/twinning semantics, redefine copy. -- (from ANY) external "built_in" ensure -- from ANY twin_not_void: Result /= Void is_equal: Result ~ Current end feature {MA_DECIMAL, MA_DECIMAL_COEFFICIENT} -- Basic operations integer_add (other: like Current) -- Integer add of other to Current. require other_not_void: other /= Void same_count: count <= other.count deferred ensure new_count: count >= old count end integer_multiply (a, b: like Current) -- Multiply a, b into Current. require a_not_void: a /= Void b_not_void: b /= Void capacity_sufficient: capacity > a.count + b.count deferred end integer_quick_add_msd (other, digits_count: INTEGER_32) -- Integer add of other (between 0 and 9) to Current, -- restricted to digits_count most significant digits. require other_limits: other >= 0 and other <= 9 digits_count: digits_count >= 0 and digits_count <= count deferred ensure new_count: count <= digits_count + 1 end integer_quick_subtract_msd (other, digits_count: INTEGER_32) -- Integer subtract of other (between 0 and 9) to Current -- restricted to digits_count most significant digits. require other_limits: other >= 0 and other <= 9 digits_count: digits_count >= 0 and digits_count <= count deferred ensure new_count: count <= digits_count + 1 end integer_subtract (other: like Current) -- Integer subtract of other to Current, -- provided that result will not be negative. require other_not_void: other /= Void other_smaller: other <= Current deferred end keep_head (a_count: INTEGER_32) -- Keep head of 'a_count' digits. require a_count_valid: a_count > 0 and a_count <= count deferred ensure adapted_count: count = a_count unchanged_capacity: capacity = old capacity end shift_left (a_count: INTEGER_32) -- Shift items left. require a_count_greater_zero: a_count > 0 deferred ensure adapted_count: count = old count + a_count zero_shifted: item (0) = 0 end shift_right (a_count: INTEGER_32) -- Shift items right. require a_count_greater_zero: a_count > 0 a_count_less_count: a_count <= count deferred ensure adapted_count: count = old count zero_shifted: item (count - 1) = 0 end strip_leading_zeroes -- Strip leading zeroes. local index: INTEGER_32 do index := msd_index set_count (index + 1) ensure no_leading_zero: count > 1 implies item (count - 1) /= 0 end feature -- Basic operations frozen as_attached: attached MA_DECIMAL_COEFFICIENT obsolete "Remove calls to this feature. [2017-05-31]" -- Attached version of Current. -- (Can be used during transitional period to convert -- non-void-safe classes to void-safe ones.) -- (from ANY) do Result := Current end frozen default: detachable MA_DECIMAL_COEFFICIENT -- Default value of object's type -- (from ANY) do end frozen default_pointer: POINTER -- Default value of type POINTER -- (Avoid the need to write p.default for -- some p of type POINTER.) -- (from ANY) do ensure -- from ANY instance_free: class end default_rescue -- Process exception for routines with no Rescue clause. -- (Default: do nothing.) -- (from ANY) do end frozen do_nothing -- Execute a null action. -- (from ANY) do ensure -- from ANY instance_free: class end feature {MA_DECIMAL} -- Implementation set_count (a_count: INTEGER_32) -- Set count to a_count. require a_count_small_enough: a_count <= capacity a_count_large_enough: a_count >= 0 deferred ensure count_set: count = a_count end feature -- Output Io: STD_FILES -- Handle to standard file setup -- (from ANY) once create Result Result.set_output_default ensure -- from ANY instance_free: class io_not_void: Result /= Void end out: STRING_8 -- New string containing terse printable representation -- of current object -- (from ANY) do Result := tagged_out ensure -- from ANY out_not_void: Result /= Void end print (o: detachable ANY) -- Write terse external representation of o -- on standard output. -- (from ANY) do if o /= Void then Io.put_string (o.out) end ensure -- from ANY instance_free: class end frozen tagged_out: STRING_8 -- New string containing terse printable representation -- of current object -- (from ANY) external "built_in" ensure -- from ANY tagged_out_not_void: Result /= Void end feature -- Platform Operating_environment: OPERATING_ENVIRONMENT -- Objects available from the operating system -- (from ANY) once create Result ensure -- from ANY instance_free: class operating_environment_not_void: Result /= Void end feature {NONE} -- Retrieval frozen internal_correct_mismatch -- Called from runtime to perform a proper dynamic dispatch on correct_mismatch -- from MISMATCH_CORRECTOR. -- (from ANY) local l_msg: STRING_8 l_exc: EXCEPTIONS do if attached {MISMATCH_CORRECTOR} Current as l_corrector then l_corrector.correct_mismatch else create l_msg.make_from_string ("Mismatch: ") create l_exc l_msg.append (generating_type.name) l_exc.raise_retrieval_exception (l_msg) end end invariant count_less_or_equal_capacity: count <= capacity count_not_negative: count >= 0 lower_upper_consistent: lower <= upper -- from COMPARABLE irreflexive_comparison: not (Current < Current) -- from ANY reflexive_equality: standard_is_equal (Current) reflexive_conformance: conforms_to (Current) end -- class MA_DECIMAL_COEFFICIENT
Generated by ISE EiffelStudio