2Do a quick, sequential, numerical (not symbolic) exploration of some electronic component values to
3propose solutions that use standard, inexpensive parts.
7from bisect
import bisect_left
8from itertools
import islice
9from math
import log10, floor
25E6 = (1.0, 1.5, 2.2, 3.3, 4.7, 6.8)
27E12 = (1.0, 1.2, 1.5, 1.8, 2.2, 2.7, 3.3, 3.9, 4.7, 5.6, 6.8, 8.2)
30 1.0, 1.1, 1.2, 1.3, 1.5, 1.6, 1.8, 2.0, 2.2, 2.4, 2.7, 3.0,
31 3.3, 3.6, 3.9, 4.3, 4.7, 5.1, 5.6, 6.2, 6.8, 7.5, 8.2, 9.1,
35 1.00, 1.05, 1.10, 1.15, 1.21, 1.27, 1.33, 1.40, 1.47, 1.54, 1.62, 1.69,
36 1.78, 1.87, 1.96, 2.05, 2.15, 2.26, 2.37, 2.49, 2.61, 2.74, 2.87, 3.01,
37 3.16, 3.32, 3.48, 3.65, 3.83, 4.02, 4.22, 4.42, 4.64, 4.87, 5.11, 5.36,
38 5.62, 5.90, 6.19, 6.49, 6.81, 7.15, 7.50, 7.87, 8.25, 8.66, 9.09, 9.53,
42 1.00, 1.02, 1.05, 1.07, 1.10, 1.13, 1.15, 1.18, 1.21, 1.24, 1.27, 1.30,
43 1.33, 1.37, 1.40, 1.43, 1.47, 1.50, 1.54, 1.58, 1.62, 1.65, 1.69, 1.74,
44 1.78, 1.82, 1.87, 1.91, 1.96, 2.00, 2.05, 2.10, 2.15, 2.21, 2.26, 2.32,
45 2.37, 2.43, 2.49, 2.55, 2.61, 2.67, 2.74, 2.80, 2.87, 2.94, 3.01, 3.09,
46 3.16, 3.24, 3.32, 3.40, 3.48, 3.57, 3.65, 3.74, 3.83, 3.92, 4.02, 4.12,
47 4.22, 4.32, 4.42, 4.53, 4.64, 4.75, 4.87, 4.99, 5.11, 5.23, 5.36, 5.49,
48 5.62, 5.76, 5.90, 6.04, 6.19, 6.34, 6.49, 6.65, 6.81, 6.98, 7.15, 7.32,
49 7.50, 7.68, 7.87, 8.06, 8.25, 8.45, 8.66, 8.87, 9.09, 9.31, 9.53, 9.76,
55 Run bisect, but use one index before the return value of `bisect_left`
56 @param a The sorted haystack
58 @return The index of the array element that equals
or is lesser than `x`
62 (i < len(a)
and a[i] > x)
or
63 (i >= len(a)
and a[i % len(a)]*10 > x)
69def approximate(x: float, series: Sequence[float]) -> tuple[Optional[int], float]:
71 Approximate a value by using the given series.
72 @param x Any positive value
73 @param series Any of E3 through E96
74 @return An integer index into the series
for the element lesser than
or equal to the value
's
75 mantissa, and the value
's decade - a power of ten
78 return None, float(
'inf')
80 decade = 10**floor(log10(x))
83 if index >= len(series):
88def fmt_eng(x: float, unit: str, sig: int = 2) -> str:
90 Format a number in engineering (SI) notation
92 @param unit The quantity unit (Hz, A, etc.)
93 @param sig Number of significant digits to show
94 @return The formatted string
98 elif x == float(
'inf'):
101 p = floor(log10(abs(x)))
102 e = int(floor(p / 3))
103 digs = max(0, sig - p%3 - 1)
104 mantissa = x / 10**(3*e)
110 prefix =
' kMGTPEZY'[e]
112 prefix =
'mμnpfazy'[-e-1]
114 raise IndexError(f
'Number out of SI range: {x:.1e}')
116 fmt =
'{:.%df} {:}{:}' % digs
117 return fmt.format(mantissa, prefix, unit)
122 A callable with any number of floating-point arguments, returning a float
130 A value associated with a component - to track approximated values
135 component:
'Component',
136 decade: Optional[float] =
None,
137 index: Optional[int] =
None,
138 exact: Optional[float] =
None,
142 - exact - approximated value will be calculated
143 - exact, index, decade - approximated value = series[index]*decade
144 - index, decade - approximated value = series[index]*decade;
147 @param decade The quantity
's power-of-ten
148 @param index The integer index into the series
for the quantity
's mantissa
149 @param exact The exact quantity,
if known
155 assert decade
is None
156 assert exact
is not None
160 assert decade
is not None
161 self.index, self.
decade = index, decade
163 if self.
decade == float(
'inf'):
166 self.
approx = component.series[self.index] * self.
decade
168 if index
is not None:
180 @return If this approximated value
is below its exact value, then the next-highest E24
181 value; otherwise
None
186 index, decade = self.index + 1, self.
decade
191 index=index, decade=decade)
198 if self.
error**2 < other.error**2:
211 A component, without knowledge of its value - only bounds and defining formula
219 series: Sequence[float] = E24,
220 calculate: Optional[CalculateCall] =
None,
222 maximum: Optional[float] =
None,
223 use_for_err: bool =
True,
226 @param prefix i.e. R, C
or L
227 @param suffix Typically a number, i.e. the
"2" in R2
228 @param unit i.e. Hz, A, F, ...
229 @param series One of E3 through E96
230 @param calculate A callable that will be given all values of previous components
in the
231 calculation sequence. These values are floats,
and the
return must be a
232 float. If this callable
is None, the component will be interpreted
as a
234 @param minimum Min allowable value; the
return of calculate will be checked against this
and
235 failures will be silently dropped. Must be at least zero,
or greater than
236 zero
if calculate
is not None.
237 @param maximum Max allowable value; the
return of calculate will be checked against this
and
238 failures will be silently dropped.
239 @param use_for_err If
True, error
from this component
's ideal to approximated value will
240 influence the solution rank.
243 self.prefix, self.suffix, self.unit, self.series,
244 self.calculate, self.min, self.max, self.use_for_err,
246 prefix, suffix, unit, series, calculate, minimum, maximum,
251 raise ValueError(
'Minimum must be non-negative')
252 if maximum
is not None and maximum < minimum:
253 raise ValueError(
'Invalid maximum value')
261 self.digits: int = 3
if len(series) > 24
else 2
263 self.fmt_field: Callable[[str], str] = (
'{:>%d}' % (4 + self.digits)).format
270 return f
'{self.prefix}{self.suffix}'
272 def _calculate_values(
273 self, prev: Sequence[ComponentValue]
274 ) -> Iterable[ComponentValue]:
278 from_exact_val = self.calculate(*(p.exact
for p
in prev))
279 if from_exact_val <= 0:
284 other = from_exact.get_other()
290 from_approx_val = self.calculate(*(p.approx
for p
in prev))
291 if from_approx_val > 0:
293 if from_approx.exact != from_exact.exact:
295 other = from_approx.get_other()
301 self.min <= v.exact
and
302 (self.max
is None or self.max >= v.exact)
306 def _all_values(self) -> Iterable[Tuple[int, float]]:
308 for index
in range(self.start_index, len(self.series)):
312 for index
in range(len(self.series)):
316 self, prev: Sequence[ComponentValue],
317 ) -> Iterable[ComponentValue]:
320 if value.approx > self.max:
329 series: Sequence[float] = E24,
330 calculate: Optional[CalculateCall] =
None,
332 maximum: Optional[float] =
None,
333 use_for_err: bool =
False,
335 super().
__init__(
'R', suffix,
'Ω', series, calculate, minimum, maximum, use_for_err)
342 series: Sequence[float] = E24,
343 calculate: Optional[CalculateCall] =
None,
345 maximum: Optional[float] =
None,
346 use_for_err: bool =
True,
348 super().
__init__(
'C', suffix,
'F', series, calculate, minimum, maximum, use_for_err)
353 A calculated parameter - potentially but not necessarily a circuit output - to be calculated
and
354 checked
for error
in the solution ranking process.
358 self, name: str, unit: str, expected: float, calculate: CalculateCall,
361 @param name i.e. Vout
362 @param unit i.e. V, A, Hz...
363 @param expected The value that this parameter would assume under ideal circumstances
364 @param calculate A callable accepting a sequence of floats - one per component,
in the same
365 order
as they were passed to the Solver constructor; returning a float.
368 name, unit, expected, calculate,
371 def error(self, value: float) -> float:
373 @return Absolute error, since the expected value might be 0
375 return value - self.expected
383 Basic recursive solver class that does a brute-force search through some component values.
388 components: Sequence[Component],
389 outputs: Sequence[Output],
390 threshold: Optional[float] = 1e-3,
393 @param components A sequence of Component instances. The order of this sequence determines
394 the order of parameters passed to Output.calculate
and
396 @param outputs A sequence of Output instances - can be empty.
397 @param threshold Maximum error above which solutions will be discarded.
399 self.components, self.outputs = components, outputs
400 self.candidates: List[Tuple[
403 Sequence[ComponentValue],
405 self.approx_seen: Set[Tuple[float, ...]] = set()
408 def _recurse(self, values: List[Optional[ComponentValue]], index: int = 0) ->
None:
409 if index >= len(self.components):
412 comp = self.components[index]
413 for v
in comp.values(values[:index]):
419 Recurse through all of the components, doing a brute-force search. Results are stored in
420 self.candidates
and sorted
in order of increasing error.
422 values = [None]*len(self.components)
424 self.candidates.sort(key=
lambda v: v[0])
426 def _evaluate(self, values: Sequence[ComponentValue]) ->
None:
427 approx = tuple(v.approx
for v
in values)
428 if approx
in self.approx_seen:
437 for o, v
in zip(self.
outputs, outputs)
440 for c, v
in zip(self.components, values)
444 self.candidates.append((err, outputs, tuple(values)))
445 self.approx_seen.add(approx)
447 def print(self, top: int = 10) ->
None:
449 Print a table of all component values, output values and output error.
450 @param top Row limit.
454 comp.fmt_field(comp.name)
455 for comp
in self.components
458 f
'{output.name:>10} {"Err":>8}'
462 for err, outputs, values
in islice(self.candidates, top):
464 value.component.fmt_field(str(value))
468 f
'{fmt_eng(value, output.unit, 4):>10} '
469 f
'{output.error(value):>8.1e}'
470 for value, output
in zip(outputs, self.
outputs)
A callable with any number of floating-point arguments, returning a float.
float __call__(self, *float args)
def __init__(self, str suffix, Sequence[float] series=E24, Optional[CalculateCall] calculate=None, float minimum=0, Optional[float] maximum=None, bool use_for_err=True)
A value associated with a component - to track approximated values.
None __init__(self, 'Component' component, Optional[float] decade=None, Optional[int] index=None, Optional[float] exact=None)
Valid combinations:
Optional[ 'ComponentValue'] get_other(self)
'ComponentValue' get_best(self)
A component, without knowledge of its value - only bounds and defining formula.
Iterable[Tuple[int, float]] _all_values(self)
Iterable[ComponentValue] _iter_values(self, Sequence[ComponentValue] prev)
None __init__(self, str prefix, str suffix, str unit, Sequence[float] series=E24, Optional[CalculateCall] calculate=None, float minimum=0, Optional[float] maximum=None, bool use_for_err=True)
Iterable[ComponentValue] _calculate_values(self, Sequence[ComponentValue] prev)
A calculated parameter - potentially but not necessarily a circuit output - to be calculated and chec...
None __init__(self, str name, str unit, float expected, CalculateCall calculate)
float error(self, float value)
None __init__(self, str suffix, Sequence[float] series=E24, Optional[CalculateCall] calculate=None, float minimum=0, Optional[float] maximum=None, bool use_for_err=False)
Basic recursive solver class that does a brute-force search through some component values.
None print(self, int top=10)
Print a table of all component values, output values and output error.
None _recurse(self, List[Optional[ComponentValue]] values, int index=0)
None __init__(self, Sequence[Component] components, Sequence[Output] outputs, Optional[float] threshold=1e-3)
None _evaluate(self, Sequence[ComponentValue] values)
None solve(self)
Recurse through all of the components, doing a brute-force search.
int bisect_lower(Sequence[float] a, float x)
Run bisect, but use one index before the return value of bisect_left
tuple[Optional[int], float] approximate(float x, Sequence[float] series)
Approximate a value by using the given series.