Source code for abjad.math

"""
Abjad's math library.
"""

import collections
import fractions
import itertools
import math
import numbers
import typing


[docs] def all_are_equal(argument) -> bool: """ Is true when ``argument`` is an iterable collection of equal items. .. container:: example >>> abjad.math.all_are_equal([99, 99, 99, 99, 99, 99]) True >>> abjad.math.all_are_equal(17) False .. container:: example Is true when ``argument`` is empty: >>> abjad.math.all_are_equal([]) True """ try: first_element = None for element in argument: if first_element is None: first_element = element else: if not element == first_element: return False return True except TypeError: return False
[docs] def all_are_integer_equivalent(argument) -> bool: """ Is true when ``argument`` is an iterable collection with integer-equivalent items. .. container:: example >>> import fractions >>> items = [1, '2', 3.0, fractions.Fraction(4, 1)] >>> abjad.math.all_are_integer_equivalent(items) True >>> abjad.math.all_are_integer_equivalent([1, '2', 3.5, 4]) False """ try: return all(is_integer_equivalent(_) for _ in argument) except TypeError: return False
[docs] def all_are_integer_equivalent_numbers(argument) -> bool: """ Is true when ``argument`` is an iterable collection with integer-equivalent items. .. container:: example >>> import fractions >>> items = [1, 2, 3.0, fractions.Fraction(4, 1)] >>> abjad.math.all_are_integer_equivalent_numbers(items) True >>> abjad.math.all_are_integer_equivalent_numbers([1, 2, 3.5, 4]) False """ try: return all(is_integer_equivalent_number(_) for _ in argument) except TypeError: return False
[docs] def all_are_nonnegative_integer_equivalent_numbers(argument) -> bool: """ Is true when ``argument`` is an iterable collection of nonnegative integer-equivalent numbers. .. container:: example >>> import fractions >>> items = [0, 0.0, fractions.Fraction(0), 2, 2.0, fractions.Fraction(2)] >>> abjad.math.all_are_nonnegative_integer_equivalent_numbers(items) True >>> items = [0, 0.0, fractions.Fraction(0), -2, 2.0, fractions.Fraction(2)] >>> abjad.math.all_are_nonnegative_integer_equivalent_numbers(items) False """ try: return all(is_nonnegative_integer_equivalent_number(_) for _ in argument) except TypeError: return False
[docs] def all_are_nonnegative_integer_powers_of_two(argument) -> bool: """ Is true when ``argument`` is an iterable collection of nonnegative integer powers of two. .. container:: example >>> items = [0, 1, 1, 1, 2, 4, 32, 32] >>> abjad.math.all_are_nonnegative_integer_powers_of_two(items) True >>> abjad.math.all_are_nonnegative_integer_powers_of_two(17) False .. container:: example Is true when ``argument`` is empty: >>> abjad.math.all_are_nonnegative_integer_powers_of_two([]) True """ try: return all(is_nonnegative_integer_power_of_two(_) for _ in argument) except TypeError: return False
[docs] def all_are_nonnegative_integers(argument) -> bool: """ Is true when ``argument`` is an iterable collection of nonnegative integers. .. container:: example >>> abjad.math.all_are_nonnegative_integers([0, 1, 2, 99]) True >>> abjad.math.all_are_nonnegative_integers([0, 1, 2, -99]) False """ try: return all(is_nonnegative_integer(_) for _ in argument) except TypeError: return False
[docs] def all_are_pairs_of_types(argument, first_type, second_type) -> bool: """ Is true when ``argument`` is an iterable collection whose members are all of length 2, and where the first member of each pair is an instance of ``first_type`` and where the second member of each pair is an instance of ``second_type``. .. container:: example >>> items = [(1., 'a'), (2.1, 'b'), (3.45, 'c')] >>> abjad.math.all_are_pairs_of_types(items, float, str) True >>> abjad.math.all_are_pairs_of_types('foo', float, str) False .. container:: example Is true when ``argument`` is empty: >>> abjad.math.all_are_pairs_of_types([], float, str) True """ try: return all( ( len(_) == 2 and isinstance(_[0], first_type) and isinstance(_[1], second_type) ) for _ in argument ) except (KeyError, TypeError): return False
[docs] def all_are_positive_integers(argument) -> bool: """ Is true when ``argument`` is an iterable collection of positive integers. .. container:: example >>> abjad.math.all_are_positive_integers([1, 2, 3, 99]) True >>> abjad.math.all_are_positive_integers(17) False """ try: return all(is_positive_integer(_) for _ in argument) except TypeError: return False
[docs] def are_relatively_prime(argument) -> bool: """ Is true when ``argument`` is an iterable collection of relative primes. .. container:: example >>> abjad.math.are_relatively_prime([13, 14, 15]) True >>> abjad.math.are_relatively_prime([13, 14, 15, 16]) False >>> abjad.math.are_relatively_prime('text') False .. container:: example Returns true when ``argument`` is empty: >>> abjad.math.are_relatively_prime([]) True """ if not isinstance(argument, collections.abc.Iterable): return False if not all(isinstance(_, numbers.Number) for _ in argument): return False all_factors: typing.Set[int] = set([]) for number in argument: current_factors = factors(number) current_factors_ = set(current_factors) if all_factors & current_factors_: return False all_factors.update(current_factors_) return True
[docs] def arithmetic_mean(argument) -> int | float | fractions.Fraction: """ Gets arithmetic mean of ``argument``. .. container:: example >>> abjad.math.arithmetic_mean([1, 2, 2, 20, 30]) 11 >>> abjad.math.arithmetic_mean([1, 2, 20]) Fraction(23, 3) >>> abjad.math.arithmetic_mean([2, 2, 20.0]) 8.0 Raises exception when ``argument`` is not iterable. """ if not isinstance(argument, collections.abc.Sequence): raise TypeError(argument) total = sum(argument) length = len(argument) if isinstance(total, float): return total / length result = fractions.Fraction(sum(argument), len(argument)) int_result = int(result) if int_result == result: return int_result else: return result
[docs] def binomial_coefficient(n, k) -> int: """ Gets binomial coefficient of ``n`` choose ``k``. .. container:: example >>> for k in range(8): ... print(k, '\t', abjad.math.binomial_coefficient(8, k)) ... 0 1 1 8 2 28 3 56 4 70 5 56 6 28 7 8 """ return math.factorial(n) // (math.factorial(n - k) * math.factorial(k))
[docs] def cumulative_products(argument): """ Gets cumulative products of ``argument``. .. container:: example >>> abjad.math.cumulative_products([1, 2, 3, 4, 5, 6, 7, 8]) [1, 2, 6, 24, 120, 720, 5040, 40320] >>> abjad.math.cumulative_products([1, -2, 3, -4, 5, -6, 7, -8]) [1, -2, -6, 24, 120, -720, -5040, 40320] Raises exception when ``argument`` is not iterable. Returns new object of ``argument`` type. """ if not isinstance(argument, collections.abc.Iterable): raise TypeError(argument) if len(argument) == 0: return type(argument)([]) result = [argument[0]] for element in argument[1:]: result.append(result[-1] * element) return type(argument)(result)
[docs] def cumulative_sums(argument, start=0): """ Gets cumulative sums of ``argument``. .. container:: example >>> abjad.math.cumulative_sums([1, 2, 3, 4, 5, 6, 7, 8], start=0) [0, 1, 3, 6, 10, 15, 21, 28, 36] >>> abjad.math.cumulative_sums([1, 2, 3, 4, 5, 6, 7, 8], start=None) [1, 3, 6, 10, 15, 21, 28, 36] Raises exception when ``argument`` is not iterable. Returns new object of ``argument`` type. """ if start is None: result = [] else: result = [start] for item in argument: if result: new = result[-1] + item result.append(new) else: result.append(item) return type(argument)(result)
[docs] def difference_series(argument): """ Gets difference series of ``argument``. .. container:: example >>> abjad.math.difference_series([1, 1, 2, 3, 5, 5, 6]) [0, 1, 1, 2, 0, 1] >>> abjad.math.difference_series([9, 6, 8, 5, 7, 4, 6]) [-3, 2, -3, 2, -3, 2] Returns new object of ``argument`` type. """ result = [] for i, n in enumerate(argument[1:]): result.append(n - argument[i]) return type(argument)(result)
[docs] def divide_integer_by_ratio(n, ratio) -> list[fractions.Fraction | float]: """ Divides integer ``n`` by tuple ``ratio``. .. container:: example >>> abjad.math.divide_integer_by_ratio(1, (1, 1, 3)) [Fraction(1, 5), Fraction(1, 5), Fraction(3, 5)] .. container:: example >>> abjad.math.divide_integer_by_ratio(1.0, (1, 1, 3)) [0.2, 0.2, 0.6] """ assert isinstance(n, int | float), repr(n) assert isinstance(ratio, tuple), repr(ratio) denominator = sum(ratio) factors = [fractions.Fraction(_, denominator) for _ in ratio] result = [n * _ for _ in factors] return result
[docs] def divisors(n) -> list[int]: """ Gets positive divisors of ``n`` in increasing order. .. container:: example >>> abjad.math.divisors(84) [1, 2, 3, 4, 6, 7, 12, 14, 21, 28, 42, 84] >>> for x in range(10, 20): ... print(x, abjad.math.divisors(x)) ... 10 [1, 2, 5, 10] 11 [1, 11] 12 [1, 2, 3, 4, 6, 12] 13 [1, 13] 14 [1, 2, 7, 14] 15 [1, 3, 5, 15] 16 [1, 2, 4, 8, 16] 17 [1, 17] 18 [1, 2, 3, 6, 9, 18] 19 [1, 19] .. container:: example Allows nonpositive ``n``: >>> abjad.math.divisors(-27) [1, 3, 9, 27] Raises not implemented error on ``0``. """ if not isinstance(n, int): raise TypeError(f"must be integer: {n!r}.") if n == 0: raise NotImplementedError("all numbers divide zero evenly.") n = abs(n) divisors = [1] for i in range(2, int(math.sqrt(n)) + 1): if n % i == 0: divisors.append(i) codivisors = [n // i for i in reversed(divisors)] if divisors[-1] == codivisors[0]: divisors.pop() divisors.extend(codivisors) divisors.sort() return divisors
[docs] def factors(n) -> list[int]: """ Gets prime factors less than or equal to ``n`` . .. container:: example >>> abjad.math.factors(84) [2, 2, 3, 7] >>> for n in range(10, 20): ... print(n, abjad.math.factors(n)) ... 10 [2, 5] 11 [11] 12 [2, 2, 3] 13 [13] 14 [2, 7] 15 [3, 5] 16 [2, 2, 2, 2] 17 [17] 18 [2, 3, 3] 19 [19] ``n`` must be a positive integer. Returns factors in increasing order. """ if not is_positive_integer(n): raise TypeError(f"must be positive integer: {n!r}.") factor = 2 factors = [] while 1 < n: if n % factor == 0: factors.append(factor) n = n / factor else: factor = factor + 1 return factors
[docs] def fraction_to_proper_fraction( rational, ) -> tuple[int, fractions.Fraction]: """ Changes ``rational`` to proper fraction. .. container:: example >>> import fractions >>> abjad.math.fraction_to_proper_fraction(fractions.Fraction(116, 8)) (14, Fraction(1, 2)) """ assert isinstance(rational, fractions.Fraction), repr(rational) quotient = int(rational) residue = rational - quotient return quotient, residue
[docs] def greatest_common_divisor(*integers) -> int: """ Calculates greatest common divisor of ``integers``. .. container:: example >>> abjad.math.greatest_common_divisor(84, -94, -144) 2 Allows nonpositive input. Raises not implemented error when zero is included in input. """ common_divisors = None for positive_integer in integers: all_divisors = set(divisors(positive_integer)) if common_divisors is None: common_divisors = all_divisors else: common_divisors &= all_divisors if common_divisors == set([1]): return 1 assert isinstance(common_divisors, set) return max(common_divisors)
[docs] def greatest_power_of_two_less_equal(n, i=0) -> int: r""" Gets greatest integer power of two less than or equal to positive ``n``. .. container:: example >>> for n in range(10, 20): ... print('\t%s\t%s' % (n, abjad.math.greatest_power_of_two_less_equal(n))) ... 10 8 11 8 12 8 13 8 14 8 15 8 16 16 17 16 18 16 19 16 Greatest-but-``i`` integer power of ``2`` less than or equal to positive ``n``: >>> for n in range(10, 20): ... print('\t%s\t%s' % (n, abjad.math.greatest_power_of_two_less_equal(n, i=1))) ... 10 4 11 4 12 4 13 4 14 4 15 4 16 8 17 8 18 8 19 8 """ if n <= 0: raise ValueError(f"must be positive: {n!r}.") return 2 ** (int(math.log(n, 2)) - i)
[docs] def integer_equivalent_number_to_integer(number) -> int | float: """ Changes integer-equivalent ``number`` to integer. .. container:: example Returns integer-equivalent number as integer: >>> abjad.math.integer_equivalent_number_to_integer(17.0) 17 .. container:: example Returns noninteger-equivalent number unchanged: >>> abjad.math.integer_equivalent_number_to_integer(17.5) 17.5 """ if is_integer_equivalent_number(number): return int(number) else: return number
[docs] def integer_to_base_k_tuple(n, k) -> tuple[int, ...]: """ Changes nonnegative integer ``n`` to base-`k` tuple. .. container:: example Gets base-10 digits of 1066: >>> abjad.math.integer_to_base_k_tuple(1066, 10) (1, 0, 6, 6) .. container:: example Gets base-2 digits of 1066: >>> abjad.math.integer_to_base_k_tuple(1066, 2) (1, 0, 0, 0, 0, 1, 0, 1, 0, 1, 0) .. container:: example Gets base-26 digits of 1066: >>> abjad.math.integer_to_base_k_tuple(1066, 26) (1, 15, 0) """ assert isinstance(n, int), repr(n) assert 0 <= n, repr(n) if n == 0: return (0,) result = [] current_exponent = math.trunc(math.log(n, k)) remainder = n while 0 <= current_exponent: current_power = k**current_exponent current_digit = remainder // current_power result.append(current_digit) remainder -= current_digit * current_power current_exponent -= 1 return tuple(result)
[docs] def integer_to_binary_string(n) -> str: r""" Changes positive integer ``n`` to binary string. .. container:: example >>> for n in range(1, 16 + 1): ... string = abjad.math.integer_to_binary_string(n) ... print(f"{n}\t{string}") ... 1 1 2 10 3 11 4 100 5 101 6 110 7 111 8 1000 9 1001 10 1010 11 1011 12 1100 13 1101 14 1110 15 1111 16 10000 """ if n == 0: return "0" result = bin(abs(n)).lstrip("-0b") if n < 0: result = "-" + result return result
[docs] def is_assignable_integer(argument) -> bool: r""" Is true when ``argument`` is equivalent to an integer that can be written without recourse to ties. .. container:: example >>> for n in range(0, 16 + 1): ... print('%s\t%s' % (n, abjad.math.is_assignable_integer(n))) ... 0 False 1 True 2 True 3 True 4 True 5 False 6 True 7 True 8 True 9 False 10 False 11 False 12 True 13 False 14 True 15 True 16 True """ if isinstance(argument, int): if 0 < argument: if "01" not in integer_to_binary_string(argument): return True return False
[docs] def is_integer_equivalent(argument) -> bool: """ Is true when ``argument`` is an integer-equivalent number. .. container:: example >>> abjad.math.is_integer_equivalent(12.0) True >>> abjad.math.is_integer_equivalent('12') True >>> abjad.math.is_integer_equivalent('foo') False """ if isinstance(argument, numbers.Number): return is_integer_equivalent_number(argument) try: int(argument) return True except (TypeError, ValueError): return False
[docs] def is_integer_equivalent_n_tuple(argument, n) -> bool: """ Is true when ``argument`` is a tuple of ``n`` integer-equivalent items. .. container:: example >>> import fractions >>> tuple_ = (2.0, '3', fractions.Fraction(4, 1)) >>> abjad.math.is_integer_equivalent_n_tuple(tuple_, 3) True >>> tuple_ = (2.5, '3', fractions.Fraction(4, 1)) >>> abjad.math.is_integer_equivalent_n_tuple(tuple_, 3) False """ return ( isinstance(argument, tuple) and len(argument) == n and all(is_integer_equivalent(_) for _ in argument) )
[docs] def is_integer_equivalent_number(argument) -> bool: """ Is true when ``argument`` is a number and ``argument`` is equivalent to an integer. .. container:: example >>> abjad.math.is_integer_equivalent_number(12.0) True >>> abjad.math.is_integer_equivalent_number(abjad.Duration(1, 2)) False """ if int(argument) == argument: return True return False
[docs] def is_nonnegative_integer(argument) -> bool: """ Is true when ``argument`` equals a nonnegative integer. .. container:: example >>> abjad.math.is_nonnegative_integer(99) True >>> abjad.math.is_nonnegative_integer(0) True >>> abjad.math.is_nonnegative_integer(-1) False """ if argument == int(argument): if 0 <= argument: return True return False
[docs] def is_nonnegative_integer_equivalent_number(argument) -> bool: """ Is true when ``argument`` is a nonnegative integer-equivalent number. .. container:: example >>> duration = abjad.Duration(4, 2) >>> abjad.math.is_nonnegative_integer_equivalent_number(duration) True """ return is_integer_equivalent_number(argument) and 0 <= argument
[docs] def is_nonnegative_integer_power_of_two(argument) -> bool: """ Is true when ``argument`` is a nonnegative integer power of 2. .. container:: example >>> for n in range(10): ... print(n, abjad.math.is_nonnegative_integer_power_of_two(n)) ... 0 True 1 True 2 True 3 False 4 True 5 False 6 False 7 False 8 True 9 False """ if isinstance(argument, int): return not bool(argument & (argument - 1)) elif isinstance(argument, fractions.Fraction): return is_nonnegative_integer_power_of_two( argument.numerator * argument.denominator ) else: return False
[docs] def is_positive_integer(argument) -> bool: """ Is true when ``argument`` equals a positive integer. .. container:: example >>> abjad.math.is_positive_integer(99) True >>> abjad.math.is_positive_integer(0) False >>> abjad.math.is_positive_integer(-1) False """ if argument == int(argument): if 0 < argument: return True return False
[docs] def is_positive_integer_equivalent_number(argument) -> bool: """ Is true when ``argument`` is a positive integer-equivalent number. .. container:: example >>> abjad.math.is_positive_integer_equivalent_number( ... abjad.Duration(4, 2) ... ) True """ try: return 0 < argument and is_integer_equivalent_number(argument) except TypeError: # Python 3 comparisons with non-numbers return False
[docs] def is_positive_integer_power_of_two(argument) -> bool: r""" Is true when ``argument`` is a positive integer power of 2. .. container:: example >>> for n in range(10): ... print(n, abjad.math.is_positive_integer_power_of_two(n)) ... 0 False 1 True 2 True 3 False 4 True 5 False 6 False 7 False 8 True 9 False """ return 0 < argument and is_nonnegative_integer_power_of_two(argument)
[docs] def least_common_multiple(*integers) -> int: """ Gets least common multiple of positive ``integers``. .. container:: example >>> abjad.math.least_common_multiple(2, 4, 5, 10, 20) 20 >>> abjad.math.least_common_multiple(4, 4) 4 >>> abjad.math.least_common_multiple(4, 5) 20 >>> abjad.math.least_common_multiple(4, 6) 12 >>> abjad.math.least_common_multiple(4, 7) 28 >>> abjad.math.least_common_multiple(4, 8) 8 >>> abjad.math.least_common_multiple(4, 9) 36 >>> abjad.math.least_common_multiple(4, 10) 20 >>> abjad.math.least_common_multiple(4, 11) 44 """ if len(integers) == 1: if not isinstance(integers[0], int): raise TypeError(f"must be integer: {integers[0]!r}.") if not 0 < integers[0]: raise ValueError(f"must be positive: {integers[0]!r}.") return integers[0] current_lcm = _least_common_multiple_helper(*integers[:2]) for remaining_positive_integer in integers[2:]: current_lcm = _least_common_multiple_helper( current_lcm, remaining_positive_integer ) return current_lcm
def _least_common_multiple_helper(m, n): assert isinstance(m, int), repr(m) assert isinstance(n, int), repr(n) factors_m = factors(m) factors_n = factors(n) for x in factors_m: try: factors_n.remove(x) except ValueError: pass result = 1 for x in factors_m + factors_n: result *= x return result
[docs] def partition_integer_by_ratio(n, ratio) -> list[int]: """ Partitions positive integer-equivalent ``n`` by ``ratio``. Returns result with weight equal to absolute value of ``n``. .. container:: example >>> abjad.math.partition_integer_by_ratio(10, (1, 2)) [3, 7] .. container:: example Partitions positive integer-equivalent ``n`` by ``ratio`` with negative parts: >>> abjad.math.partition_integer_by_ratio(10, (1, -2)) [3, -7] .. container:: example Partitions negative integer-equivalent ``n`` by ``ratio``: >>> abjad.math.partition_integer_by_ratio(-10, (1, 2)) [-3, -7] .. container:: example Partitions negative integer-equivalent ``n`` by ``ratio`` with negative parts: >>> abjad.math.partition_integer_by_ratio(-10, (1, -2)) [-3, 7] .. container:: example More examples: >>> abjad.math.partition_integer_by_ratio(10, (1,)) [10] >>> abjad.math.partition_integer_by_ratio(10, (1, 1)) [5, 5] >>> abjad.math.partition_integer_by_ratio(10, (1, -1, -1)) [3, -4, -3] >>> abjad.math.partition_integer_by_ratio(-10, (1, 1, 1, 1)) [-3, -2, -3, -2] >>> abjad.math.partition_integer_by_ratio(-10, (1, 1, 1, 1, 1)) [-2, -2, -2, -2, -2] """ if not is_integer_equivalent_number(n): raise TypeError(f"is not integer-equivalent number: {n!r}.") if not all(is_integer_equivalent_number(_) for _ in ratio): raise ValueError(f"must be integer tuple ratio, not {ratio!r}.") result = [0] parts = [float(abs(n)) * abs(_) / weight(ratio) for _ in ratio] cumulative_parts = cumulative_sums(parts, start=None) for part in cumulative_parts: rounded_part = int(round(part)) - sum(result) if part - round(part) == 0.5: rounded_part += 1 result.append(rounded_part) result = result[1:] if sign(n) == -1: result = [-_ for _ in result] ratio_signs = [sign(_) for _ in ratio] result = [pair[0] * pair[1] for pair in zip(ratio_signs, result)] return result
[docs] def partition_integer_into_canonic_parts( n, decrease_parts_monotonically=True ) -> tuple[int, ...]: """ Partitions integer ``n`` into canonic parts. Returns all parts positive on positive ``n``: .. container:: example >>> for n in range(1, 11): ... print(n, abjad.math.partition_integer_into_canonic_parts(n)) ... 1 (1,) 2 (2,) 3 (3,) 4 (4,) 5 (4, 1) 6 (6,) 7 (7,) 8 (8,) 9 (8, 1) 10 (8, 2) .. container:: example Returns all parts negative on negative ``n``: >>> for n in reversed(range(-20, -10)): ... print(n, abjad.math.partition_integer_into_canonic_parts(n)) ... -11 (-8, -3) -12 (-12,) -13 (-12, -1) -14 (-14,) -15 (-15,) -16 (-16,) -17 (-16, -1) -18 (-16, -2) -19 (-16, -3) -20 (-16, -4) .. container:: example Returns parts that increase monotonically: >>> for n in range(11, 21): ... print(n, abjad.math.partition_integer_into_canonic_parts(n, ... decrease_parts_monotonically=False)) ... 11 (3, 8) 12 (12,) 13 (1, 12) 14 (14,) 15 (15,) 16 (16,) 17 (1, 16) 18 (2, 16) 19 (3, 16) 20 (4, 16) Returns tuple with parts that decrease monotonically. """ assert isinstance(n, int), repr(n) assert isinstance(decrease_parts_monotonically, bool) if n == 0: return (0,) result = [] previous_empty = True binary_n = integer_to_binary_string(abs(n)) binary_length = len(binary_n) for i, character in enumerate(binary_n): if character == "1": place_value = 2 ** (binary_length - i - 1) if previous_empty: result.append(place_value) else: result[-1] += place_value previous_empty = False else: previous_empty = True sign_n = sign(n) if sign(n) == -1: result = [sign_n * _ for _ in result] if decrease_parts_monotonically: return tuple(result) else: return tuple(reversed(result))
[docs] def sign(n) -> int: """ Gets sign of ``n``. .. container:: example Returns -1 on negative ``n``: >>> abjad.math.sign(-96.2) -1 Returns 0 when ``n`` is 0: >>> abjad.math.sign(0) 0 Returns 1 on positive ``n``: >>> abjad.math.sign(abjad.Duration(9, 8)) 1 """ return (0 < n) - (n < 0)
[docs] def weight(argument) -> int: """ Gets weight of ``argument``. .. container:: example >>> abjad.math.weight([-1, -2, 3, 4, 5]) 15 .. container:: example >>> abjad.math.weight([]) 0 Defined equal to sum of the absolute value of items in ``argument``. """ return sum([abs(_) for _ in argument])
[docs] def yield_all_compositions_of_integer(n: int) -> typing.Iterator[tuple[int, ...]]: """ Yields all compositions of positive integer ``n``. .. container:: example >>> for tuple_ in abjad.math.yield_all_compositions_of_integer(5): ... tuple_ ... (5,) (4, 1) (3, 2) (3, 1, 1) (2, 3) (2, 2, 1) (2, 1, 2) (2, 1, 1, 1) (1, 4) (1, 3, 1) (1, 2, 2) (1, 2, 1, 1) (1, 1, 3) (1, 1, 2, 1) (1, 1, 1, 2) (1, 1, 1, 1, 1) Lists parts in descending lex order. Parts sum to ``n``. Finds small values of ``n`` easily. Takes around 4 seconds for ``n`` equal to 17. """ compositions = [] integer = 0 string_length = n while integer < 2 ** (n - 1): binary_string = integer_to_binary_string(integer) binary_string = binary_string.zfill(string_length) digits = [int(_) for _ in list(binary_string)] partition = [] generator = itertools.groupby(digits, lambda _: _) for value, group in generator: partition.append(list(group)) sublengths = [len(part) for part in partition] composition = tuple(sublengths) compositions.append(composition) integer += 1 for composition in reversed(sorted(compositions)): yield composition
[docs] class Infinity: """ Infinity. .. container:: example All numbers compare less than infinity: >>> 9999999 < abjad.Infinity() True >>> 2**38 < abjad.Infinity() True .. container:: example Infinity compares equal to itself: >>> abjad.Infinity() == abjad.Infinity() True .. container:: example Negative infinity compares less than infinity: >>> abjad.NegativeInfinity() < abjad.Infinity() True Initializes as a system singleton at start-up. Available as a built-in after Abjad starts. """ ### CLASS VARIABLES ### __slots__ = ("_value",) ### INTIALIZER ### def __init__(self): self._value = float("infinity") ### SPECIAL METHODS ###
[docs] def __eq__(self, argument) -> bool: """ Compares type. """ return isinstance(argument, type(self))
[docs] def __float__(self): """ Convert infinity to float. Returns float. """ return self._value
[docs] def __ge__(self, argument): """ Is true for all values of ``argument``. Returns true. """ return self._value >= argument
[docs] def __gt__(self, argument): """ Is true for all noninfinite values of ``argument``. Returns true or false. """ return self._value > argument
[docs] def __hash__(self): """ Hashes infinity. """ return hash(self.__class__.__name__ + str(self))
[docs] def __le__(self, argument): """ Is true when ``argument`` is infinite. Returns true or false. """ return self._value <= argument
[docs] def __lt__(self, argument): """ Is true for no values of ``argument``. Returns true or false. """ return self._value < argument
[docs] def __repr__(self) -> str: """ Gets repr. """ return f"{type(self).__name__}()"
[docs] def __sub__(self, argument): """ Subtracts ``argument`` from infinity. Returns infinity or 0 if ``argument`` is also infinity. """ if argument is self: return 0 return self
[docs] class NegativeInfinity(Infinity): """ Negative infinity. .. container:: example All numbers compare greater than negative infinity: >>> abjad.NegativeInfinity() < -9999999 True .. container:: example Negative infinity compares equal to itself: >>> abjad.NegativeInfinity() == abjad.NegativeInfinity() True .. container:: example Negative infinity compares less than infinity: >>> abjad.NegativeInfinity() < abjad.Infinity() True Initializes as a system singleton at start-up. Available as a built-in after Abjad start. """ ### CLASS VARIABLES ### __slots__ = () ### INITIALIZER ### def __init__(self): self._value = float("-infinity")