{ } Raw JSON

bundles / scipy 1.17.1 / scipy / signal / _filter_design / bilinear

function

scipy.signal._filter_design:bilinear

source: /scipy/signal/_filter_design.py :2336

Signature

def   bilinear ( b a fs = 1.0 )

Summary

Calculate a digital IIR filter from an analog transfer function by utilizing the bilinear transform.

Parameters

b : array_like

Coefficients of the numerator polynomial of the analog transfer function in form of a complex- or real-valued 1d array.

a : array_like

Coefficients of the denominator polynomial of the analog transfer function in form of a complex- or real-valued 1d array.

fs : float

Sample rate, as ordinary frequency (e.g., hertz). No pre-warping is done in this function.

Returns

beta : ndarray

Coefficients of the numerator polynomial of the digital transfer function in form of a complex- or real-valued 1d array.

alpha : ndarray

Coefficients of the denominator polynomial of the digital transfer function in form of a complex- or real-valued 1d array.

Notes

The parameters and are 1d arrays of length and . They define the analog transfer function

The bilinear transform [1] is applied by substituting

into , with being the sampling rate. This results in the digital transfer function in the -domain

This expression can be simplified by multiplying numerator and denominator by , with . This allows to be reformulated as

This is the equation implemented to perform the bilinear transform. Note that for large , or can cause a numeric overflow for sufficiently large or .

Array API Standard Support

bilinear has experimental support for Python Array API Standard compatible backends in addition to NumPy. Please consider testing these features by setting an environment variable SCIPY_ARRAY_API=1 and providing CuPy, PyTorch, JAX, or Dask arrays as array arguments. The following combinations of backend and device (or other capability) are supported.

====================  ====================  ====================
Library               CPU                   GPU
====================  ====================  ====================
NumPy                 ✅                     n/a                 
CuPy                  n/a                   ✅                   
PyTorch               ✅                     ⛔                   
JAX                   ⚠️ no JIT
Dask                  ⚠️ computes graph     n/a                 
====================  ====================  ====================

CuPy does not accept complex inputs.

See dev-arrayapi for more information.

Examples

The following example shows the frequency response of an analog bandpass filter and the corresponding digital filter derived by utilitzing the bilinear transform:
from scipy import signal
import matplotlib.pyplot as plt
import numpy as np
fs = 100  # sampling frequency
om_c = 2 * np.pi * np.array([7, 13])  # corner frequencies
bb_s, aa_s = signal.butter(4, om_c, btype='bandpass', analog=True, output='ba')
bb_z, aa_z = signal.bilinear(bb_s, aa_s, fs)
w_z, H_z = signal.freqz(bb_z, aa_z)  # frequency response of digitial filter
w_s, H_s = signal.freqs(bb_s, aa_s, worN=w_z*fs)  # analog filter response
f_z, f_s = w_z * fs / (2*np.pi), w_s / (2*np.pi)
Hz_dB, Hs_dB = (20*np.log10(np.abs(H_).clip(1e-10)) for H_ in (H_z, H_s))
fg0, ax0 = plt.subplots()
ax0.set_title("Frequency Response of 4-th order Bandpass Filter")
ax0.set(xlabel='Frequency $f$ in Hertz', ylabel='Magnitude in dB',
        xlim=[f_z[1], fs/2], ylim=[-200, 2])
ax0.semilogx(f_z, Hz_dB, alpha=.5, label=r'$|H_z(e^{j 2 \pi f})|$')
ax0.semilogx(f_s, Hs_dB, alpha=.5, label=r'$|H_s(j 2 \pi f)|$')
ax0.legend()
ax0.grid(which='both', axis='x')
ax0.grid(which='major', axis='y')
plt.show()
fig-7d5cb9c14b82590e.png
The difference in the higher frequencies shown in the plot is caused by an effect called "frequency warping". [1]_ describes a method called "pre-warping" to reduce those deviations.

See also

bilinear_zpk
lp2bp
lp2bs
lp2hp
lp2lp

Aliases

  • scipy.signal.bilinear

Referenced by

This package