Defining Aircraft ‘Speed’#

Measuring translations and velocities for a ground-based vehicle (your car) is relatively simple as the vehicle is moving with respect to an inertially-fixed reference frame (the Earth), that it is always connected to. To determine how long it will take to get somewhere, distance/time/velocity calculations are simple, and the engine performance is chiefly a function of the vehicle speed.

By comparison, an aircraft is moving with respect to the inertially-fixed reference frame, but it is not connected to it - thus is it difficult to make measurements between the two. Furthermore, aerodynamic performance (how much lift and drag is being produced) is a function of the aircraft’s orientation and translation with respect to the incident wind, and not to the Earth. Similarly, the powerplant performance will be a function of the incident wind and not the groundspeed.

This gives us our first set of ‘speeds’, and the relationship between them:

\[\begin{aligned} V_{gnd} &= V + V_{wind}\label{eq:Vgnd1} \end{aligned}\]

where

\[\begin{split}\begin{aligned} &V_{gnd} : \text{Ground Speed}\nonumber\\ &V : \text{True Airspeed (TAS)}\nonumber\\ &V_{wind} : \text{Wind Speed}\nonumber\end{aligned}\end{split}\]

where we can simply add the two together via vector addition. In the two figures below, the two situations will require the exact same amount of thrust, and the same configuration of control surfaces - but the there is a groundspeed difference of 100kn.

../_images/HeadWind.png

Fig. 14 Aircraft With Headwind#

../_images/TailWind.png

Fig. 15 Aircraft With Tailwind#

This is why LHR-ORD takes typically 8h50m, but ORD-LHR takes 7h30m, yet in each case, the aircraft is flying at typical cruise velocity, requiring the same amount of total thrust.

Airspeed Measurement#

Clearly aerospace engineers require a means of determining the true airspeed, in order to determine aircraft powerplant/aerodynamic performance, and in order to calculate groundspeed and thus facilitate speed/time calculations.

Equation (1) may be developed from the isentropic flow equations, and allows us to determine true airspeed as a function of density and pressure:

(1)#\[V =\sqrt{7\cdot\frac{p}{\rho}\cdot\left\{\left(\frac{\Delta p}{p}+1\right)^\frac{\gamma-1}{\gamma}-1\right\}}\]

where

\[ \begin{align}\begin{aligned}\rho = \text{ Local air density}\\p = \text{ Local static pressure}\\\gamma = \text{ Specific heat ratio = 1.4 for diatomic gases}\\\Delta p = \text{ Dynamic pressure/pressure difference} : p_o-p\\p_o = \text{ Stagnation/total pressure} p_o = \tfrac{1}{2}\rho V^2 + p\end{aligned}\end{align} \]

Equation (1) is often simply given in textbooks with no derivation or explanation. You are encouraged to understand the derivation, but I won’t test you on it.

Equation (1) allows us to determine the velocity with respect to a volume of air, via measurement of three quantities - density \(\rho\), static pressure \(p\), and impact pressure \(\Delta p\). ‘Impact pressure’ is a term that is probably new to you, and refers to the difference between total and static pressure - and, for an incompressible fluid, is the same as dynamic pressure, but not for a compressible fluid.

We have the relationship \(V(p, \rho, \Delta p)\), but the first two quantities are relatively difficult to measure, as they are absolute quantities. Hence, transducers for measurement of absolute pressure and density would require regular calibration, and would add complexity to a measurement system. The reasons for this are complex, and will be elaborated upon in an instrumentation/measurement course - suffice to say that, in general, all transducers are relative transducers, and thus measurement of any absolute quantity require a datum against which they can be compared, calibrated for changes in non-measured quantities, such as temperature.

By contrast, \(\Delta p\) is a relative quantity. Whilst a transducer for measurement will still require calibration, measurement of \(\Delta p\) is relatively easy - for this purpose, we use a pitot-static device. Originally these were analogue devices, measuring the pressure difference between two sides of a pressure chamber via a mechanical diaphragm, calibrated to provide an accurate reading of \(\Delta p\):

../_images/PitotStatic.png

Fig. 16 Aircraft Pitot-Static Measurement Device#

Calibrated Airspeed#

To avoid having to measure \(p\) and \(\rho\), rather than using the pitot-static to measure \(\Delta p\), aircraft Engineers calibrated their devices to provide the correct value of true airspeed at sea-level ISA density and pressure. This provides us with calibrated airspeed, \(V_c\)/CAS, which is only the same as true airspeed at sea-level ISA conditions. \(V_c\) can be calculated via Equation (2):

(2)#\[ V_c=\sqrt{7\cdot\frac{p_{SL}}{\rho_{SL}}\cdot\left\{\left(\frac{\Delta p}{p_{SL}}+1\right)^\frac{\gamma-1}{\gamma}-1\right\}}\]

Indicated Airspeed#

In reality, such a pitot-static device will measure the calibrated airspeed at a point on the aircraft surface, where the flow has already been disturbed by the aircraft. The device is NOT measuring the freestream velicity - thus there is a position error, based on where the pitot-static is on the aircraft. During flight testing, these inaccuracies can be quantified, and a difference between what is indicated by the device, indicated airspeed/IAS, and calibrated airspeed is determined - \(\Delta V_p\), the position error.

(3)#\[V_c = V_I + \Delta V_p\]

At this stage - we have three airspeeds; Indicated (what is shown on the measurement device, for ISA SL conditions), Calibrated (what should be shown on the measurement device, for ISA SL conditions), and True Airspeed. We don’t have a means to convert Calibrated to True yet, so we adopt a two-step approach.

Equivalent Airspeed#

The first correction is for the actual pressure at a given altitude. This gives Equivalent Airspeed, EAS.

(4)#\[ V_e=\sqrt{7\cdot\frac{p}{\rho_{SL}}\cdot\left\{\left(\frac{\Delta p}{p}+1\right)^\frac{\gamma-1}{\gamma}-1\right\}}\]

where we can see that the sea-level pressure has been replaced with the actual pressure. In practice, this correction is applied as a multiplier between CAS and EAS:

\[V_e = f\cdot V_c\]

where

\[ f = \frac{V_e}{V_c} \]
\[ = \frac{\sqrt{7\cdot\frac{p}{\rho_{SL}}\cdot\left\{\left(\frac{\Delta p}{p}+1\right)^\frac{\gamma-1}{\gamma}-1\right\}}}{\sqrt{7\cdot\frac{p_{SL}}{\rho_{SL}}\cdot\left\{\left(\frac{\Delta p}{p_{SL}}+1\right)^\frac{\gamma-1}{\gamma}-1\right\}}}\]
\[= f(\Delta p, p)\]

Since \(f\) is only dependent on the \(\Delta p\) the aircraft speed, and the \(p\) the aircraft altitude, it can be calculated and tabulated - see the table below for the pressure correction factor:

h, ft

Calibrated Airspeed, kn

100

125

150

175

200

225

250

275

300

5000

0.999

0.999

0.999

0.998

0.998

0.997

0.997

0.996

0.995

10000

0.999

0.998

0.997

0.996

0.995

0.994

0.992

0.991

0.989

15000

0.998

0.997

0.995

0.994

0.992

0.990

0.987

0.985

0.982

20000

0.997

0.995

0.993

0.990

0.987

0.984

0.981

0.977

0.973

25000

0.995

0.993

0.990

0.986

0.982

0.978

0.973

0.968

0.963

30000

0.993

0.990

0.986

0.981

0.975

0.970

0.963

0.957

0.950

35000

0.991

0.986

0.981

0.974

0.967

0.959

0.951

0.943

0.934

40000

0.988

0.982

0.974

0.966

0.957

0.947

0.937

0.926

0.916

45000

0.984

0.976

0.966

0.956

0.944

0.932

0.920

0.907

0.895

50000

0.979

0.969

0.957

0.944

0.930

0.915

0.901

0.886

0.871

Intermediate values are found via linear interpolation - you should be comfortable with doing this by hand, or by writing code. An example code is provided below that provides two-dimensional interpolation for any values - you can use this function to find out the pressure correction factor - use the syntax f_correction(VC, h) (both of these have default values, so you don’t need to provide either).

Hide code cell source
import numpy as np
import scipy.interpolate
from scipy.interpolate import interp2d
presscorr = np.array([[5000,0.999,0.999,0.999,0.998,0.998,0.997,0.997,0.996,0.995],
    [10000,0.999,0.998,0.997,0.996,0.995,0.994,0.992,0.991,0.989],
    [15000,0.998,0.997,0.995,0.994,0.992,0.99,0.987,0.985,0.982],
    [20000,0.997,0.995,0.993,0.99,0.987,0.984,0.981,0.977,0.973],
    [25000,0.995,0.993,0.99,0.986,0.982,0.978,0.973,0.968,0.963],
    [30000,0.993,0.99,0.986,0.981,0.975,0.97,0.963,0.957,0.95],
    [35000,0.991,0.986,0.981,0.974,0.967,0.959,0.951,0.943,0.934],
    [40000,0.988,0.982,0.974,0.966,0.957,0.947,0.937,0.926,0.916],
    [45000,0.984,0.976,0.966,0.956,0.944,0.932,0.92,0.907,0.895],
    [50000,0.979,0.969,0.957,0.944,0.93,0.915,0.901,0.886,0.871]])

h = presscorr[:, 0]
VC = np.array([100, 125, 150, 175, 200, 225, 250, 275, 300])
table = presscorr[:, 1:]



f_corr = interp2d(VC, h, table, kind='linear')

def f_correction(VC=125, h=45000, ):
    f = f_corr(VC, h)[0]
    return f
---------------------------------------------------------------------------
NotImplementedError                       Traceback (most recent call last)
Cell In[1], line 21
     16 VC = np.array([100, 125, 150, 175, 200, 225, 250, 275, 300])
     17 table = presscorr[:, 1:]
---> 21 f_corr = interp2d(VC, h, table, kind='linear')
     23 def f_correction(VC=125, h=45000, ):
     24     f = f_corr(VC, h)[0]

File ~/PycharmProjects/Aircraft-Flight-Mechanics/venv/lib/python3.12/site-packages/scipy/interpolate/_interpolate.py:129, in interp2d.__init__(self, x, y, z, kind, copy, bounds_error, fill_value)
    127 def __init__(self, x, y, z, kind='linear', copy=True, bounds_error=False,
    128              fill_value=None):
--> 129     raise NotImplementedError(err_mesg)

NotImplementedError: `interp2d` has been removed in SciPy 1.14.0.

For legacy code, nearly bug-for-bug compatible replacements are
`RectBivariateSpline` on regular grids, and `bisplrep`/`bisplev` for
scattered 2D data.

In new code, for regular grids use `RegularGridInterpolator` instead.
For scattered data, prefer `LinearNDInterpolator` or
`CloughTocher2DInterpolator`.

For more details see
https://scipy.github.io/devdocs/tutorial/interpolate/interp_transition_guide.html
VC = 182
h = 39500
f = f_correction(VC, h)
print(f"For a CAS of {VC}, at an altitude of {h} feet, the pressure correction factor is {f}")
For a CAS of 182, at an altitude of 39500 feet, the pressure correction factor is 0.964336

True Airspeed#

Finally, True Airspeed may be calculated from Equivalent Airspeed, by accounting for the actual density at the correct altitude:

\[V = V_e\cdot\sqrt{\frac{\rho_{SL}}{\rho}}\]
\[V = V_e\cdot\sqrt{\frac{1}{\sigma}}\]
\[\sigma = \frac{\rho}{\rho_{SL}}\]

Where we may get the density at different altitudes from standard tables of the atmospheric properties. These are available in the original PDF notes. What we’ll use more regularly is either atmosisa in MATLAB or ambiance in Python.

If you can’t get the following to work - then you might need to install ambiance. Try pip install ambiance in your terminal window. See here for more instructions: https://pypi.org/project/ambiance/

from ambiance import Atmosphere

sealevel = Atmosphere(0)

h = 12000

altmospere = Atmosphere(h)

print(f"At sea level, the pressure is {sealevel.pressure[0]:6.3f}Pa,\
 the density is {sealevel.density[0]:1.3f}kg/m^3,\
 the tempertaure is {sealevel.temperature[0]-273.15:1.1f}C, \
 and the viscosity is {sealevel.kinematic_viscosity[0]:1.3e}kg/s")

print("")

print(f"At an altitude of {h}m, the pressure is {altmospere.pressure[0]:6.3f}Pa,\
the density is {altmospere.density[0]:1.3f}kg/m^3,\
the tempertaure is {altmospere.temperature[0]-273.15:1.1f}C,\
and the viscosity is {altmospere.kinematic_viscosity[0]:1.3e}kg/s")
At sea level, the pressure is 101325.000Pa, the density is 1.225kg/m^3, the tempertaure is 15.0C,  and the viscosity is 1.461e-05kg/s

At an altitude of 12000m, the pressure is 19399.392Pa,the density is 0.312kg/m^3,the tempertaure is -56.5C,and the viscosity is 4.557e-05kg/s

Using ambiance to define the density, we can create a function, sigma_density (it’s a good idea to make function handles descriptive, and there tend to be lots of sigma functions that mean important things in programming languages). This function will return the density ratio for a given altitude provided in metres or feet - bearing in mind than the units for ambiance are metres.

from ambiance import Atmosphere

def sigma_density(h=10000, alt_units='m'):
    if alt_units == 'm':
        factor = 1
    elif alt_units == 'km':
        factor = 1000
    elif alt_units == 'ft':
        factor = 0.3048
    elif alt_units == 'miles':
        factor = 1609.34
    
    sea_level = Atmosphere(0)
    atmospheric_properties = Atmosphere(h*factor)
    
    sigma = atmospheric_properties.density / sea_level.density
    return sigma
    
h_in = 39500

h_unit = 'ft'

print(f"At {h_in}{h_unit}, the density ratio is {sigma_density(h_in, h_unit)[0]:1.6f}")

atm = Atmosphere(39000*.3048)
print(atm.temperature)
At 39500ft, the density ratio is 0.253064
[216.65]
Hide code cell source
# Get four different atmospheres
atmo_sl = Atmosphere(0*.3048)
atmo_35kft = Atmosphere(35000*.3048)
atmo_40kft = Atmosphere(40000*.3048)
atmo_395kft = Atmosphere(39500*.3048)


# Conversions
kg_to_slugs = 1/14.59390
m_to_feet = 3.28084

# Convert the densities
dens_imperial_sl = atmo_sl.density * kg_to_slugs / m_to_feet**3
dens_imperial_35kft = atmo_35kft.density * kg_to_slugs / m_to_feet**3
dens_imperial_40kft = atmo_40kft.density * kg_to_slugs / m_to_feet**3
dens_imperial_395kft = atmo_395kft.density * kg_to_slugs / m_to_feet**3

print(f"The density at 35,000ft from the table is {7.38*1e-4:1.4e}slugs/ft**3 and from ICAO it is {dens_imperial_35kft[0]:1.4e}slugs/ft**3")
print(f"The density at 40,000ft from the table is {5.87*1e-4:1.4e}slugs/ft**3 and from ICAO it is {dens_imperial_40kft[0]:1.4e}slugs/ft**3")
print("")

# Interpolate the densities
dens_interpolated_39500_ambiance = dens_imperial_35kft + 4500/5000 * (dens_imperial_40kft - dens_imperial_35kft)
dens_interpolated_39500_table = 7.38e-4 + 4500/5000 * (5.87e-4 - 7.38e-4)

print(f"The density at 39,500ft from ICAO is {dens_imperial_395kft[0]:1.4e}slugs/ft**3 and interpolated between 35000 and 40000 ICAO it is {dens_interpolated_39500_ambiance[0]:1.4e}slugs/ft**3")

print("")
# Get sigmas
sigma_ambiance_direct = (atmo_395kft.density/atmo_sl.density)[0]
sigma_table = dens_interpolated_39500_table/(23.77*1e-4)
sigma_ambiance = dens_interpolated_39500_ambiance/dens_imperial_sl


print(f"Sigma from the table is {sigma_table:1.4f}, \n from ambiance (at 39000) is is {sigma_ambiance_direct:1.4f}, \n and from ambiance (interpolated between 35000 and 40000) is is {sigma_ambiance[0]:1.4f}")
Hide code cell output
The density at 35,000ft from the table is 7.3800e-04slugs/ft**3 and from ICAO it is 7.3821e-04slugs/ft**3
The density at 40,000ft from the table is 5.8700e-04slugs/ft**3 and from ICAO it is 5.8728e-04slugs/ft**3

The density at 39,500ft from ICAO is 6.0151e-04slugs/ft**3 and interpolated between 35000 and 40000 ICAO it is 6.0237e-04slugs/ft**3

Sigma from the table is 0.2533, 
 from ambiance (at 39000) is is 0.2531, 
 and from ambiance (interpolated between 35000 and 40000) is is 0.2534

The inputs can also be numpy arrays

alts = np.arange(1, 10)
sigmas = sigma_density(alts, 'miles')

for alt, sigma in zip(alts, sigmas):
    if alt == 1:
        print(f"At an altitude of {alt} mile, the density ratio is {sigma}")
    else:
        print(f"At an altitude of {alt} miles, the density ratio is {sigma}")
         
# Note that the syntax to write mile/miles for single vs. any other quatity is completely unnecessary but nice
At an altitude of 1 mile, the density ratio is 0.8544179176392871
At an altitude of 2 miles, the density ratio is 0.7256946830480345
At an altitude of 3 miles, the density ratio is 0.6124148561281167
At an altitude of 4 miles, the density ratio is 0.5132337621724308
At an altitude of 5 miles, the density ratio is 0.42687647082233393
At an altitude of 6 miles, the density ratio is 0.3521367607463307
At an altitude of 7 miles, the density ratio is 0.28579654351527745
At an altitude of 8 miles, the density ratio is 0.22195323343439954
At an altitude of 9 miles, the density ratio is 0.17239372263976577

Approximate density correction#

For altitudes below about 16km, the approximation can be used

\[\sigma=\frac{20-H}{20+H}\]

where \(H\) is the altitude in \(km\).

The error in this approximation is easily shown using the functions created above:

Hide code cell source
def sigma_approx(H):
    sigma = (20 - H ) / (20 + H)
    return sigma

Hs = np.linspace(0, 30e3, int(30e3))

sigma_exact = sigma_density(Hs)
sigma_approximation = sigma_approx(Hs/1e3)

import matplotlib.pyplot as plt

plt.figure()
plt.plot(Hs/1e3, sigma_exact, '-', label="Exact")
plt.plot(Hs/1e3, sigma_approximation, '--', label="Approximation")
plt.legend();
plt.title('Exact and Approximate Sigma Corrections')
plt.xlabel('Altitude in km')
plt.ylabel('$\\sqrt{\\frac{\\rho}{\\rho_{sl}}}$');
../_images/6f4f766279cd860adec1a9411e3589747bf2a9964e83aad0cd76ca9e9ba37348.png

Why use EAS?#

Equivalent airspeed is actually very useful (though pilots tend not to actually use it, annoyingly). We tend to define \(q_\infty\) as \(\tfrac{1}{2}\rho_{SL}V_e^2\), which means that aerodynamic coefficients remain the same for a given \(V_e\) and \(\alpha/\beta\).

It is useful to see that the Equivalent Air Speed (EAS) for a given flight condition, \(V_e\), is the speed which if flown at standard sea level density (\(\rho_{sl} \sim 1.225 kg\cdot m^{-3}\)) would give the same aerodynamic loads, for the same aerodynamic configuration, constant \(C_L\).

This is useful for simplicity of calculations, but is also useful for ease of flight - stall will always occur at the same angle, for the same EAS, regardless of flight altitude. Structural limits are always defined in EAS for the same reason, as the loads are constant at any altitude for the same EAS.

Summary of Corrections#

We can remember the order of corrections by the mnemonic “ICE-T”, with the relative magnitudes of the velocities given by the shape that loosely looks like a square root -

../_images/IceT.png

Fig. 17 Airspeed Corrections#

Problems:#

Below are a range of numerical and theoretical questions. Some of the numerical questions have random numbers that change each time the notes get updated - so you’re not going crazy if you thought it was different last time.

Try and go through the questions on your own before looking at the solutions - questions you’ll face in tests/exams will be similar but not exactly the same, so you need to be able to understand the why of the solution rather than just following an algorithm.

Problem 1.1 - Conversion between airspeeds#

An aircraft is flying at an altitude of 35000ft, with a calibrated airspeed of 195kn, with a 10kn headwind:

a) How long will it take to cover 100 miles?

b) How long will it take to cover 200km?

c) If instead of \(V_C\) = 195kn, you have \(V_I\) = 195kn, with a position error of \(\Delta V_P=+2kn\), what do the above answers change to?

Try and tackle the problem yourself before you see the solution below. The numbers in this problem will change each time the notes are updated.

Problem 1.2 - Theory#

a) When converting between different airspeeds, how can you ‘sense check’ your numbers?

b) What is the significance of “equivalent airspeed”, and why is it preferable to use when defining limiting speeds for aircraft

c) In what cases could you find that your true airspeed is less than your calibrated airspeed?

d) For what conditions is EAS equivalent to TAS?