From b7437624294609c60a9f4573ac9747c72e071fe4 Mon Sep 17 00:00:00 2001 From: unexcellent <> Date: Wed, 11 Dec 2024 14:32:16 +0100 Subject: [PATCH 1/6] refactor: use _indent() function --- generate/generate_boilerplate.py | 42 ++++++++++++++++++++------------ 1 file changed, 26 insertions(+), 16 deletions(-) diff --git a/generate/generate_boilerplate.py b/generate/generate_boilerplate.py index aaf0435..7e7585b 100644 --- a/generate/generate_boilerplate.py +++ b/generate/generate_boilerplate.py @@ -67,26 +67,30 @@ def _generate_properties(current_class: str, units: dict[str, str]) -> list[str] code = [] for unit, factor in units.items(): - code.append(" " * 4 + "@property") - code.append(" " * 4 + f"def {unit}(self) -> float:") - code.append(" " * 8 + f'"""The {current_class.lower()} in {unit.replace("_", " ")}."""') - code.append(" " * 8 + f"return self._base_value / {factor}") + code.append(_indent("@property", 1)) + code.append(_indent(f"def {unit}(self) -> float:", 1)) + code.append(_indent(f'"""The {current_class.lower()} in {unit.replace("_", " ")}."""', 2)) + code.append(_indent(f"return self._base_value / {factor}", 2)) code.append("") return code def _generate_init(units: dict[str, str]) -> list[str]: - code = [" " * 4 + "def __init__(", " " * 8 + "self,", " " * 8 + "_base_value: float = 0.0,"] + code = [ + _indent("def __init__(", 1), + _indent("self,", 2), + _indent("_base_value: float = 0.0,", 2), + ] for unit in units: - code.append(" " * 8 + f"{unit}: float = 0.0,") + code.append(_indent(f"{unit}: float = 0.0,", 2)) - code.append(" " * 4 + ") -> None:") - code.append(" " * 8 + "self._base_value = _base_value") + code.append(_indent(") -> None:", 1)) + code.append(_indent("self._base_value = _base_value", 2)) for unit, factor in units.items(): - code.append(" " * 8 + f"self._base_value += {unit} * {factor}") + code.append(_indent(f"self._base_value += {unit} * {factor}", 2)) code.append("") return code @@ -94,23 +98,29 @@ def _generate_init(units: dict[str, str]) -> list[str]: def _generat_zero_function(current_class: str) -> list[str]: return [ - " " * 4 + "@classmethod", - " " * 4 + f"def zero(cls) -> {current_class}:", - " " * 8 + f'"""Create a {current_class} with a value of zero."""', - " " * 8 + f"return {current_class}()", + _indent("@classmethod", 1), + _indent(f"def zero(cls) -> {current_class}:", 1), + _indent(f'"""Create a {current_class} with a value of zero."""', 2), + _indent(f"return {current_class}()", 2), "", ] def _generat_str_function(current_class: str) -> list[str]: return [ - " " * 4 + f"def __str__(self) -> str:", - " " * 8 + f'"""Display this quantity as a string for printing."""', - " " * 8 + f'return "{current_class}(" + self.BASE_UNIT + "=" + str(self._base_value) + ")"', + _indent(f"def __str__(self) -> str:", 1), + _indent(f'"""Display this quantity as a string for printing."""', 2), + _indent( + f'return "{current_class}(" + self.BASE_UNIT + "=" + str(self._base_value) + ")"', 2 + ), "", ] +def _indent(text: str, number_of_indents: int) -> str: + return " " * 4 * number_of_indents + text + + if __name__ == "__main__": quantities_with_fields = { "Acceleration": { From 2c38b8cc1edf164cc0b56fd1084d74b04fabbd27 Mon Sep 17 00:00:00 2001 From: unexcellent <> Date: Wed, 11 Dec 2024 14:40:51 +0100 Subject: [PATCH 2/6] perf: add Quantitiy.__init__() checks for presence of argument --- benchmarking/bench_quantity.py | 2 +- generate/generate_boilerplate.py | 5 +- quantio/quantities.py | 390 ++++++++++++++++++------------- 3 files changed, 238 insertions(+), 159 deletions(-) diff --git a/benchmarking/bench_quantity.py b/benchmarking/bench_quantity.py index 1c33dbb..7cd9cd2 100644 --- a/benchmarking/bench_quantity.py +++ b/benchmarking/bench_quantity.py @@ -4,7 +4,7 @@ def init_only_base_unit(): - Time(seconds=1) + Time(seconds=1, milliseconds=1) def bench_init(benchmark): diff --git a/generate/generate_boilerplate.py b/generate/generate_boilerplate.py index 7e7585b..76ab1e3 100644 --- a/generate/generate_boilerplate.py +++ b/generate/generate_boilerplate.py @@ -84,13 +84,14 @@ def _generate_init(units: dict[str, str]) -> list[str]: ] for unit in units: - code.append(_indent(f"{unit}: float = 0.0,", 2)) + code.append(_indent(f"{unit}: float | None = None,", 2)) code.append(_indent(") -> None:", 1)) code.append(_indent("self._base_value = _base_value", 2)) for unit, factor in units.items(): - code.append(_indent(f"self._base_value += {unit} * {factor}", 2)) + code.append(_indent(f"if {unit} is not None:", 2)) + code.append(_indent(f"self._base_value += {unit} * {factor}", 3)) code.append("") return code diff --git a/quantio/quantities.py b/quantio/quantities.py index e55badc..251bc85 100644 --- a/quantio/quantities.py +++ b/quantio/quantities.py @@ -64,12 +64,14 @@ def g_force(self) -> float: def __init__( self, _base_value: float = 0.0, - meters_per_square_second: float = 0.0, - g_force: float = 0.0, + meters_per_square_second: float | None = None, + g_force: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += meters_per_square_second * 1 - self._base_value += g_force * (1 / 9.8) + if meters_per_square_second is not None: + self._base_value += meters_per_square_second * 1 + if g_force is not None: + self._base_value += g_force * (1 / 9.8) @classmethod def zero(cls) -> Acceleration: @@ -103,12 +105,14 @@ def radians(self) -> float: def __init__( self, _base_value: float = 0.0, - degrees: float = 0.0, - radians: float = 0.0, + degrees: float | None = None, + radians: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += degrees * (3.141592653589793 / 180) - self._base_value += radians * 1 + if degrees is not None: + self._base_value += degrees * (3.141592653589793 / 180) + if radians is not None: + self._base_value += radians * 1 @classmethod def zero(cls) -> Angle: @@ -142,12 +146,14 @@ def radians_per_second(self) -> float: def __init__( self, _base_value: float = 0.0, - degrees_per_second: float = 0.0, - radians_per_second: float = 0.0, + degrees_per_second: float | None = None, + radians_per_second: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += degrees_per_second * (3.141592653589793 / 180) - self._base_value += radians_per_second * 1 + if degrees_per_second is not None: + self._base_value += degrees_per_second * (3.141592653589793 / 180) + if radians_per_second is not None: + self._base_value += radians_per_second * 1 @classmethod def zero(cls) -> AngularVelocity: @@ -211,24 +217,32 @@ def square_micrometers(self) -> float: def __init__( self, _base_value: float = 0.0, - square_miles: float = 0.0, - square_kilometers: float = 0.0, - square_meters: float = 0.0, - square_feet: float = 0.0, - square_inches: float = 0.0, - square_centimeters: float = 0.0, - square_millimeters: float = 0.0, - square_micrometers: float = 0.0, + square_miles: float | None = None, + square_kilometers: float | None = None, + square_meters: float | None = None, + square_feet: float | None = None, + square_inches: float | None = None, + square_centimeters: float | None = None, + square_millimeters: float | None = None, + square_micrometers: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += square_miles * 1609.34**2 - self._base_value += square_kilometers * 10 ** (3 * 2) - self._base_value += square_meters * 1 - self._base_value += square_feet * 0.3048**2 - self._base_value += square_inches * 0.0254**2 - self._base_value += square_centimeters * 10 ** (-2 * 2) - self._base_value += square_millimeters * 10 ** (-3 * 2) - self._base_value += square_micrometers * 10 ** (-6 * 2) + if square_miles is not None: + self._base_value += square_miles * 1609.34**2 + if square_kilometers is not None: + self._base_value += square_kilometers * 10 ** (3 * 2) + if square_meters is not None: + self._base_value += square_meters * 1 + if square_feet is not None: + self._base_value += square_feet * 0.3048**2 + if square_inches is not None: + self._base_value += square_inches * 0.0254**2 + if square_centimeters is not None: + self._base_value += square_centimeters * 10 ** (-2 * 2) + if square_millimeters is not None: + self._base_value += square_millimeters * 10 ** (-3 * 2) + if square_micrometers is not None: + self._base_value += square_micrometers * 10 ** (-6 * 2) @classmethod def zero(cls) -> Area: @@ -272,16 +286,20 @@ def grams_per_milliliter(self) -> float: def __init__( self, _base_value: float = 0.0, - grams_per_cubic_meter: float = 0.0, - kilograms_per_cubic_meter: float = 0.0, - kilograms_per_liter: float = 0.0, - grams_per_milliliter: float = 0.0, + grams_per_cubic_meter: float | None = None, + kilograms_per_cubic_meter: float | None = None, + kilograms_per_liter: float | None = None, + grams_per_milliliter: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += grams_per_cubic_meter * 10**3 - self._base_value += kilograms_per_cubic_meter * 1 - self._base_value += kilograms_per_liter * 10**-3 - self._base_value += grams_per_milliliter * 10**-3 + if grams_per_cubic_meter is not None: + self._base_value += grams_per_cubic_meter * 10**3 + if kilograms_per_cubic_meter is not None: + self._base_value += kilograms_per_cubic_meter * 1 + if kilograms_per_liter is not None: + self._base_value += kilograms_per_liter * 10**-3 + if grams_per_milliliter is not None: + self._base_value += grams_per_milliliter * 10**-3 @classmethod def zero(cls) -> Density: @@ -330,18 +348,23 @@ def milliohm(self) -> float: def __init__( self, _base_value: float = 0.0, - gigaohm: float = 0.0, - megaohm: float = 0.0, - kiloohm: float = 0.0, - ohm: float = 0.0, - milliohm: float = 0.0, + gigaohm: float | None = None, + megaohm: float | None = None, + kiloohm: float | None = None, + ohm: float | None = None, + milliohm: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += gigaohm * 10**9 - self._base_value += megaohm * 10**6 - self._base_value += kiloohm * 10**3 - self._base_value += ohm * 1 - self._base_value += milliohm * 10**-3 + if gigaohm is not None: + self._base_value += gigaohm * 10**9 + if megaohm is not None: + self._base_value += megaohm * 10**6 + if kiloohm is not None: + self._base_value += kiloohm * 10**3 + if ohm is not None: + self._base_value += ohm * 1 + if milliohm is not None: + self._base_value += milliohm * 10**-3 @classmethod def zero(cls) -> ElectricalResistance: @@ -390,18 +413,23 @@ def milliamperes(self) -> float: def __init__( self, _base_value: float = 0.0, - gigaamperes: float = 0.0, - megaamperes: float = 0.0, - kiloamperes: float = 0.0, - amperes: float = 0.0, - milliamperes: float = 0.0, + gigaamperes: float | None = None, + megaamperes: float | None = None, + kiloamperes: float | None = None, + amperes: float | None = None, + milliamperes: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += gigaamperes * 10**9 - self._base_value += megaamperes * 10**6 - self._base_value += kiloamperes * 10**3 - self._base_value += amperes * 1 - self._base_value += milliamperes * 10**-3 + if gigaamperes is not None: + self._base_value += gigaamperes * 10**9 + if megaamperes is not None: + self._base_value += megaamperes * 10**6 + if kiloamperes is not None: + self._base_value += kiloamperes * 10**3 + if amperes is not None: + self._base_value += amperes * 1 + if milliamperes is not None: + self._base_value += milliamperes * 10**-3 @classmethod def zero(cls) -> ElectricCurrent: @@ -450,18 +478,23 @@ def millijoules(self) -> float: def __init__( self, _base_value: float = 0.0, - gigajoules: float = 0.0, - megajoules: float = 0.0, - kilojoules: float = 0.0, - joules: float = 0.0, - millijoules: float = 0.0, + gigajoules: float | None = None, + megajoules: float | None = None, + kilojoules: float | None = None, + joules: float | None = None, + millijoules: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += gigajoules * 10**9 - self._base_value += megajoules * 10**6 - self._base_value += kilojoules * 10**3 - self._base_value += joules * 1 - self._base_value += millijoules * 10**-3 + if gigajoules is not None: + self._base_value += gigajoules * 10**9 + if megajoules is not None: + self._base_value += megajoules * 10**6 + if kilojoules is not None: + self._base_value += kilojoules * 10**3 + if joules is not None: + self._base_value += joules * 1 + if millijoules is not None: + self._base_value += millijoules * 10**-3 @classmethod def zero(cls) -> Energy: @@ -505,16 +538,20 @@ def hertz(self) -> float: def __init__( self, _base_value: float = 0.0, - gigahertz: float = 0.0, - megahertz: float = 0.0, - kilohertz: float = 0.0, - hertz: float = 0.0, + gigahertz: float | None = None, + megahertz: float | None = None, + kilohertz: float | None = None, + hertz: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += gigahertz * 10**9 - self._base_value += megahertz * 10**6 - self._base_value += kilohertz * 10**3 - self._base_value += hertz * 1 + if gigahertz is not None: + self._base_value += gigahertz * 10**9 + if megahertz is not None: + self._base_value += megahertz * 10**6 + if kilohertz is not None: + self._base_value += kilohertz * 10**3 + if hertz is not None: + self._base_value += hertz * 1 @classmethod def zero(cls) -> Frequency: @@ -578,24 +615,32 @@ def micrometers(self) -> float: def __init__( self, _base_value: float = 0.0, - miles: float = 0.0, - kilometers: float = 0.0, - meters: float = 0.0, - feet: float = 0.0, - inches: float = 0.0, - centimeters: float = 0.0, - millimeters: float = 0.0, - micrometers: float = 0.0, + miles: float | None = None, + kilometers: float | None = None, + meters: float | None = None, + feet: float | None = None, + inches: float | None = None, + centimeters: float | None = None, + millimeters: float | None = None, + micrometers: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += miles * 1609.34 - self._base_value += kilometers * 10**3 - self._base_value += meters * 1 - self._base_value += feet * 0.3048 - self._base_value += inches * 0.0254 - self._base_value += centimeters * 10**-2 - self._base_value += millimeters * 10**-3 - self._base_value += micrometers * 10**-6 + if miles is not None: + self._base_value += miles * 1609.34 + if kilometers is not None: + self._base_value += kilometers * 10**3 + if meters is not None: + self._base_value += meters * 1 + if feet is not None: + self._base_value += feet * 0.3048 + if inches is not None: + self._base_value += inches * 0.0254 + if centimeters is not None: + self._base_value += centimeters * 10**-2 + if millimeters is not None: + self._base_value += millimeters * 10**-3 + if micrometers is not None: + self._base_value += micrometers * 10**-6 @classmethod def zero(cls) -> Length: @@ -654,22 +699,29 @@ def micrograms(self) -> float: def __init__( self, _base_value: float = 0.0, - tonnes: float = 0.0, - kilograms: float = 0.0, - pounds: float = 0.0, - ounces: float = 0.0, - grams: float = 0.0, - milligrams: float = 0.0, - micrograms: float = 0.0, + tonnes: float | None = None, + kilograms: float | None = None, + pounds: float | None = None, + ounces: float | None = None, + grams: float | None = None, + milligrams: float | None = None, + micrograms: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += tonnes * 10**3 - self._base_value += kilograms * 1 - self._base_value += pounds * (1 / 2.20462) - self._base_value += ounces * (1 / 35.27396) - self._base_value += grams * 10**-3 - self._base_value += milligrams * 10**-6 - self._base_value += micrograms * 10**-9 + if tonnes is not None: + self._base_value += tonnes * 10**3 + if kilograms is not None: + self._base_value += kilograms * 1 + if pounds is not None: + self._base_value += pounds * (1 / 2.20462) + if ounces is not None: + self._base_value += ounces * (1 / 35.27396) + if grams is not None: + self._base_value += grams * 10**-3 + if milligrams is not None: + self._base_value += milligrams * 10**-6 + if micrograms is not None: + self._base_value += micrograms * 10**-9 @classmethod def zero(cls) -> Mass: @@ -718,18 +770,23 @@ def milliwatts(self) -> float: def __init__( self, _base_value: float = 0.0, - gigawatts: float = 0.0, - megawatts: float = 0.0, - kilowatts: float = 0.0, - watts: float = 0.0, - milliwatts: float = 0.0, + gigawatts: float | None = None, + megawatts: float | None = None, + kilowatts: float | None = None, + watts: float | None = None, + milliwatts: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += gigawatts * 10**9 - self._base_value += megawatts * 10**6 - self._base_value += kilowatts * 10**3 - self._base_value += watts * 1 - self._base_value += milliwatts * 10**-3 + if gigawatts is not None: + self._base_value += gigawatts * 10**9 + if megawatts is not None: + self._base_value += megawatts * 10**6 + if kilowatts is not None: + self._base_value += kilowatts * 10**3 + if watts is not None: + self._base_value += watts * 1 + if milliwatts is not None: + self._base_value += milliwatts * 10**-3 @classmethod def zero(cls) -> Power: @@ -788,22 +845,29 @@ def millipascal(self) -> float: def __init__( self, _base_value: float = 0.0, - terapascal: float = 0.0, - gigapascal: float = 0.0, - megapascal: float = 0.0, - kilopascal: float = 0.0, - bar: float = 0.0, - pascal: float = 0.0, - millipascal: float = 0.0, + terapascal: float | None = None, + gigapascal: float | None = None, + megapascal: float | None = None, + kilopascal: float | None = None, + bar: float | None = None, + pascal: float | None = None, + millipascal: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += terapascal * 10**9 - self._base_value += gigapascal * 10**6 - self._base_value += megapascal * 10**3 - self._base_value += kilopascal * 1 - self._base_value += bar * 10**-2 - self._base_value += pascal * 10**-3 - self._base_value += millipascal * 10**-3 + if terapascal is not None: + self._base_value += terapascal * 10**9 + if gigapascal is not None: + self._base_value += gigapascal * 10**6 + if megapascal is not None: + self._base_value += megapascal * 10**3 + if kilopascal is not None: + self._base_value += kilopascal * 1 + if bar is not None: + self._base_value += bar * 10**-2 + if pascal is not None: + self._base_value += pascal * 10**-3 + if millipascal is not None: + self._base_value += millipascal * 10**-3 @classmethod def zero(cls) -> Pressure: @@ -857,20 +921,26 @@ def nanoseconds(self) -> float: def __init__( self, _base_value: float = 0.0, - hours: float = 0.0, - minutes: float = 0.0, - seconds: float = 0.0, - milliseconds: float = 0.0, - microseconds: float = 0.0, - nanoseconds: float = 0.0, + hours: float | None = None, + minutes: float | None = None, + seconds: float | None = None, + milliseconds: float | None = None, + microseconds: float | None = None, + nanoseconds: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += hours * 60 * 60 - self._base_value += minutes * 60 - self._base_value += seconds * 1 - self._base_value += milliseconds * 10**-3 - self._base_value += microseconds * 10**-6 - self._base_value += nanoseconds * 10**-9 + if hours is not None: + self._base_value += hours * 60 * 60 + if minutes is not None: + self._base_value += minutes * 60 + if seconds is not None: + self._base_value += seconds * 1 + if milliseconds is not None: + self._base_value += milliseconds * 10**-3 + if microseconds is not None: + self._base_value += microseconds * 10**-6 + if nanoseconds is not None: + self._base_value += nanoseconds * 10**-9 @classmethod def zero(cls) -> Time: @@ -909,14 +979,17 @@ def miles_per_hour(self) -> float: def __init__( self, _base_value: float = 0.0, - meters_per_second: float = 0.0, - kilometers_per_hour: float = 0.0, - miles_per_hour: float = 0.0, + meters_per_second: float | None = None, + kilometers_per_hour: float | None = None, + miles_per_hour: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += meters_per_second * 1 - self._base_value += kilometers_per_hour * (1 / 3.6) - self._base_value += miles_per_hour * (1 / 2.23694) + if meters_per_second is not None: + self._base_value += meters_per_second * 1 + if kilometers_per_hour is not None: + self._base_value += kilometers_per_hour * (1 / 3.6) + if miles_per_hour is not None: + self._base_value += miles_per_hour * (1 / 2.23694) @classmethod def zero(cls) -> Velocity: @@ -965,18 +1038,23 @@ def millivolts(self) -> float: def __init__( self, _base_value: float = 0.0, - gigavolts: float = 0.0, - megavolts: float = 0.0, - kilovolts: float = 0.0, - volts: float = 0.0, - millivolts: float = 0.0, + gigavolts: float | None = None, + megavolts: float | None = None, + kilovolts: float | None = None, + volts: float | None = None, + millivolts: float | None = None, ) -> None: self._base_value = _base_value - self._base_value += gigavolts * 10**9 - self._base_value += megavolts * 10**6 - self._base_value += kilovolts * 10**3 - self._base_value += volts * 1 - self._base_value += millivolts * 10**-3 + if gigavolts is not None: + self._base_value += gigavolts * 10**9 + if megavolts is not None: + self._base_value += megavolts * 10**6 + if kilovolts is not None: + self._base_value += kilovolts * 10**3 + if volts is not None: + self._base_value += volts * 1 + if millivolts is not None: + self._base_value += millivolts * 10**-3 @classmethod def zero(cls) -> Voltage: From 73263772ca4a811be5e448c45f3b3dbd4d760723 Mon Sep 17 00:00:00 2001 From: unexcellent <> Date: Wed, 11 Dec 2024 14:43:34 +0100 Subject: [PATCH 3/6] test: refactor Quantitiy init bench --- benchmarking/bench_quantity.py | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/benchmarking/bench_quantity.py b/benchmarking/bench_quantity.py index 7cd9cd2..7ac1c5b 100644 --- a/benchmarking/bench_quantity.py +++ b/benchmarking/bench_quantity.py @@ -3,12 +3,9 @@ from quantio import Time -def init_only_base_unit(): - Time(seconds=1, milliseconds=1) - - def bench_init(benchmark): - benchmark.pedantic(init_only_base_unit, iterations=100, rounds=100) + f = lambda: Time(seconds=1, milliseconds=1) + benchmark.pedantic(f, iterations=100, rounds=100) if __name__ == "__main__": From 5213d5c240a172f43937144ad6b55ddf57ffc64b Mon Sep 17 00:00:00 2001 From: unexcellent <> Date: Wed, 11 Dec 2024 14:50:06 +0100 Subject: [PATCH 4/6] test: add vector benchmark functions --- benchmarking/bench_vector.py | 44 ++++++++++++++++++++++++++++++++++++ 1 file changed, 44 insertions(+) create mode 100644 benchmarking/bench_vector.py diff --git a/benchmarking/bench_vector.py b/benchmarking/bench_vector.py new file mode 100644 index 0000000..bd3920d --- /dev/null +++ b/benchmarking/bench_vector.py @@ -0,0 +1,44 @@ +import pytest +import numpy as np + +from quantio import Length, Vector + + +def bench_from_numpy__base_unit(benchmark): + f = lambda: Vector.from_numpy(np.ones(100), Length, "meters") + benchmark.pedantic(f, iterations=100, rounds=100) + + +def bench_from_numpy__other_unit(benchmark): + f = lambda: Vector.from_numpy(np.ones(100), Length, "centimeters") + benchmark.pedantic(f, iterations=100, rounds=100) + + +def bench_to_numpy__base_unit(benchmark): + vector = Vector.from_numpy(np.ones(100), Length, "meters") + f = lambda: vector.to_numpy("meters") + benchmark.pedantic(f, iterations=100, rounds=100) + + +def bench_to_numpy__other_unit(benchmark): + vector = Vector.from_numpy(np.ones(100), Length, "meters") + f = lambda: vector.to_numpy("centimeters") + benchmark.pedantic(f, iterations=100, rounds=100) + + +def bench_add(benchmark): + vector1 = Vector.from_numpy(np.ones(100), Length, "meters") + vector2 = Vector.from_numpy(np.ones(100), Length, "meters") + f = lambda: vector1 + vector2 + benchmark.pedantic(f, iterations=100, rounds=100) + + +def bench_sub(benchmark): + vector1 = Vector.from_numpy(np.ones(100), Length, "meters") + vector2 = Vector.from_numpy(np.ones(100), Length, "meters") + f = lambda: vector1 - vector2 + benchmark.pedantic(f, iterations=100, rounds=100) + + +if __name__ == "__main__": + pytest.main([__file__, "-v"]) From 3730c1f5d11dd62e0be16e9420cb792b521402f5 Mon Sep 17 00:00:00 2001 From: unexcellent <> Date: Wed, 11 Dec 2024 14:53:45 +0100 Subject: [PATCH 5/6] perf: make Vector.to_numpy() faster by not calling Vector.elements --- quantio/vector.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/quantio/vector.py b/quantio/vector.py index 953dde1..dca0cc8 100644 --- a/quantio/vector.py +++ b/quantio/vector.py @@ -83,7 +83,7 @@ def from_numpy( def to_numpy(self, unit: str | None = None) -> np.ndarray[float]: """Convert this vector into a numpy array of floats.""" - if not isinstance(self.elements[0], Quantity): + if not isinstance(self[0], Quantity): return self._elements if unit is None: @@ -92,7 +92,8 @@ def to_numpy(self, unit: str | None = None) -> np.ndarray[float]: if unit == self._quantitiy.BASE_UNIT: # type: ignore return self._elements - return np.array([getattr(element, unit) for element in self.elements]) + conversion_factor = getattr(self._quantitiy(1), unit) + return self._elements * conversion_factor def sum(self) -> T: """Return a sum of all elements of this array.""" From 3679c64473922a75131519c5bfda7520d6a32856 Mon Sep 17 00:00:00 2001 From: unexcellent <> Date: Wed, 11 Dec 2024 15:00:56 +0100 Subject: [PATCH 6/6] perf: make Vector.__add__() and Vector.__sub__() faster by not calling Vector.elements --- quantio/vector.py | 20 ++++++++++++-------- test/test_vector.py | 2 +- 2 files changed, 13 insertions(+), 9 deletions(-) diff --git a/quantio/vector.py b/quantio/vector.py index dca0cc8..a711f47 100644 --- a/quantio/vector.py +++ b/quantio/vector.py @@ -66,11 +66,9 @@ def tile(cls, element: T | Vector[T] | list[T], length: int) -> Vector[T]: return Vector(np.tile(element, length)) @classmethod - def from_numpy( - cls, array: np.ndarray, element_class: type[Quantity], unit: str - ) -> Vector[Quantity]: + def from_numpy(cls, array: np.ndarray, element_class: type[Quantity], unit: str) -> Vector[T]: """Construct a quantity vector from a numpy array.""" - vector: Vector[Quantity] = Vector([0]) + vector: Vector[T] = Vector([0]) if unit == element_class.BASE_UNIT: vector._elements = array @@ -119,15 +117,21 @@ def __add__(self, other: Vector[T] | np.ndarray) -> Vector[T]: """Add another vector to this one.""" if not isinstance(other[0], self._quantitiy): raise CanNotAddTypesError(self[0].__class__.__name__, other[0].__class__.__name__) - other_elements = other.elements if isinstance(other, Vector) else np.array(other) - return Vector[T](self.elements + other_elements) + return Vector[T].from_numpy( + self._elements + other._elements, + self._quantitiy, + self._quantitiy.BASE_UNIT, # type: ignore[attr-defined] + ) def __sub__(self, other: Vector[T] | np.ndarray) -> Vector[T]: """Subtract another vector from this one.""" if not isinstance(other[0], self._quantitiy): raise CanNotSubtractTypesError(self[0].__class__.__name__, other[0].__class__.__name__) - other_elements = other.elements if isinstance(other, Vector) else np.array(other) - return Vector[T](self.elements - other_elements) + return Vector[T].from_numpy( + self._elements - other._elements, + self._quantitiy, + self._quantitiy.BASE_UNIT, # type: ignore[attr-defined] + ) def __mul__(self, other: Vector | np.ndarray | float) -> np.ndarray: """Multiply this vector with either another vector or a scalar.""" diff --git a/test/test_vector.py b/test/test_vector.py index 5e89c97..1bc3a20 100644 --- a/test/test_vector.py +++ b/test/test_vector.py @@ -245,7 +245,7 @@ def test_sum__quantity(): def test_from_numpy__quantity(): array = np.array([0, 1, 2]) - actual = Vector.from_numpy(array, Length, "meters") + actual = Vector[Length].from_numpy(array, Length, "meters") assert actual == Vector([Length(meters=0), Length(meters=1), Length(meters=2)])