Energy Balances II
Contents
Energy Balances II#
This lecture begins with example problems using Energy Balances.
import matplotlib.pyplot as plt
import numpy as np
import scipy.optimize as opt
from scipy.integrate import solve_ivp
from scipy.interpolate import interp1d
Brief Review of the Need For Energy Balances#
Until now, we’ve always considered isothermal reactors. We have often considered the effect of changing temperature on a rate constant, but as far as the reactor is concerned, we have always assumed that the entire reactor exists at a single temperature that is equal to the temperature at the start of the run (in a batch reactor) or the temperatuer of the feed (in a flow reactor). These would be isothermal reactors.
We know after courses in mass and energy balances and thermodynamics that most reactions are not thermoneutral. This means that most reactions have a non-zero heat of reaction (\(\Delta H\)), so they will either release energy to or absorb energy from the environment. As this happens, we should anticipate that the temperature inside of the reactor may change considerably. From a strict chemistry standpoint, this is important for two reasons:
- Rate constants have an exponential dependence on temperature 
- Equilibrium constants have an exponential dependence on temperature 
In other words, changing the reaction temperature over the course of reaction progress can dramatically impact how favorable reaction is (thermodynamics, equilibrium constant) and how fast the reaction occurs (kinetics, rate constant). So it is extremely important that we understand how to anticipate changes in temperature as a function of reaction progress, reaction time, and/or reactor size. We do this by writing energy balances on the reactor of interest. We can use the same basic concept that we did for material balances – we envision a shell balance on the reactor, and we consider the energy entering, leaving, and accumulating in that control volume.
If you are interested in the derivation of the reactor energy balances starting from a shell balance, please see Lecture 39, where the steps are outlined in detail. It is rare that we will need to go back to the starting point in general, and in all problems we will consider this semester, we will only work with the following, useful forms of the energy balance.
Energy Balances for Ideal Reactor Archetypes in 587#
Batch Reactor#
The material balance for species j in a well-mixed batch reactor is:
In cases where we have either an incompressible fluid or our reactor operates at constant pressure, the energy balance on a batch reactor is:
A few notation conventions: the bar over a property here means it is an intensive molar property for a species and it has units of “per mole”. \(\dot{Q}\) is the rate of heat exhange with the surroundings. It is given by:
Where \(U\) is an overall heat transfer coefficient, \(A\) is the heat exchange area, and \(T_a\) is the temperature of the heat exchange fluid.
CSTR#
The material balance for species j in a CSTR is:
If we have an incompressible fluid or a reactor operating at constant pressure, the energy balance on a CSTR is:
The rate of heat exchange is the same as in a batch reactor:
PFR#
The material balance for species j in a PFR is:
If we have an ideal gas or a reactor operating without a pressure drop, the energy balance for a PFR is:
For a PFR, we express the rate of heat transfer per unit volume, and it has a symbol \(\dot{q}\):
\(U\) is an overall heat transfer coefficient as usual, and \(a\) is the ratio of surface area to volume for the reactor. For a tube, this is usually given as:
Example Problem 01#
The exothermic elementary liquid-phase reaction (Rawlings and Ekerdt, E6.1):
is carried out in a batch reactor equipped with a cooling coil such that you can vary conditions between isothermal and adiabatic. The reactor is initially charged with equal concentrations of A and B at 27\(^\circ\)C (300K), and both of their concentrations are 2.0 mol/L. The reaction is irreversible, and it is first order with respect to both A and B.
The rate constant can be computed from the following equation:
where: \(k_0 = 0.01725 \, \textrm{L} \, \textrm{mol}^{-1} \, \textrm{min}^{-1} \textrm{at 300K and} \frac{E_A}{R} = 2660 \textrm{K}\)
You additionally have the following data:
If the reactor operates isothermally at 300K, how long does it take to achieve 95% conversion of species A? Plot XA and rate vs time for this solution.
Solution to Example Problem 01#
This is a problem we are already familiar with. We’d like to solve for the time required to achieve a specific conversion in a well-mixed batch reactor. There are many paths we can take, but they all involve writing material balances. My approach here will be to write balances on all species and integrate the coupled system of ODEs numerically using solve_ivp.
We can express production rates in terms of the reaction rate:
This reaction follows elementary kinetics, so the rate law is:
We are solving ODEs in the time domain with 3 state (dependent) variables: NA, NB, and NC. I can express my concentrations in terms of these quantities:
And we know the rate constant for this system. Since it is isothermal at 300K, the rate constant is always equal to the rate constant at 300K:
That is everything we need to solve using an ODE solver; see cell below.
def isothermal(t, var):
    NA = var[0]
    NB = var[1]
    NC = var[2]
    V  = 1200 #L
    
    CA = NA/V
    CB = NB/V
    CC = NC/V
    
    k0 = 0.01725 #L/mol/min
    
    r  = k0*CA*CB
    RA = -r
    RB = -r
    RC =  r
    
    D1 = RA*V
    D2 = RB*V
    D3 = RC*V
    return [D1, D2, D3]
##problem statement information
k0    = 0.01725 #L/mol/min
V0    = 1200
CA0   = 2.0 #mol/L
CB0   = 2.0 #mol/L
CC0   = 0.0
NA0   = CA0*V0
NB0   = CB0*V0
NC0   = CC0*V0
#set up IVP solver, solve problem
N0    = [NA0, NB0, NC0]
tspan = (0.0, 1000.0)
ansa  = solve_ivp(isothermal, tspan, N0, atol = 1e-10, rtol = 1e-10)
#Workup solution to get requested quantities
t     = ansa.t
NA    = ansa.y[0,:]
NB    = ansa.y[1,:]
CA    = NA/V0
CB    = NB/V0
XA    = (NA0 - NA)/NA0
r     = k0*CA*CB
T_iso = np.zeros(len(t)) + 300
k     = np.zeros(len(t)) + k0
itpa  = interp1d(XA, t, kind = 'cubic')
tsola = itpa(0.95)
print(f'For isothermal operation at 300K, it will take {tsola:3.0f} minutes to achieve 95% conversion')
##Plot of conversion vs. time
plt.figure(1, figsize = (5, 5))
plt.plot(t, XA)
plt.xlabel('time (min)', fontsize = 12)
plt.ylabel('XA', fontsize = 12)
plt.xlim(0, max(tspan))
plt.ylim(0, 1)
plt.show()
##Create plot with 2 yaxes to plot Conversion and Temperature vs. time
fig1, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
conv = ax1.plot(t, XA, color = 'black', label = 'XA')
temp = ax2.plot(t, T_iso, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, 1)
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('XA', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
##Create a plot with two y axes to show rate and temperature vs. time
fig2, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
rate = ax1.plot(t, r, color = 'black', label = 'rate')
temp = ax2.plot(t, T_iso, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, max(r))
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('rate (mol/L/min)', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
##Create a plot with two y axes to show rate constant and temperature vs. time
fig3, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
cons = ax1.plot(t, k, color = 'black', label = 'k')
temp = ax2.plot(t, T_iso, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, max(r))
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('rate constant (L/mol/min)', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
For isothermal operation at 300K, it will take 551 minutes to achieve 95% conversion
 
 
 
 
Example Problem 02#
If the reactor operates adiabatically, what is the maximum temperature that the reactor can reach during operation?
Solution to Example Problem 02#
We’ll start this with the energy balance on the batch reactor:
First, some simplifications that are specific to this problem. It is adiabatic, so we know there is no heat exchange with the surroundings, and \(\dot{Q} = 0\). Second, there is only one reaction involved so the summation term involving heats of reaction involves only one term.
We’ll expand the summation of species on the left hand side:
And I’ll go ahead and move that summation to the right hand side; this is our starting point for this particular problem:
We could actually solve this as is by combining it with material balances on A, B, and C (see Example Problem 03 below), but the problem statement does not ask us anything about how long it takes to achieve a maximum temperature increase, it only asks what the maximum temperature increase is. This means we don’t really need to integrate the problem in the time domain, we are really only interested in the increase in temperature as a function of reaction progress. Clearly, the further this exothermic reaction goes toward completion (\(X_A = 1\)), the higher the temperature will increase. The relationship between temperature and conversion in an adiabatic process is usually called the “adiabatic temperature change” for a reaction, and it is a very important quantity for us to be able to estimate. To facilitate an analytical solution, we’d like to reduce this to a single ODE instead of having to solve multiple, coupled ODEs. We do this by substituting a material balance into the energy balance.
Specifically, if we write a balance on species A, we find:
We can express \(N_A\) as a function of fractional conversion:
Substituting this into the material balance above, we get:
We see that the rV product appears in our energy balance, so we’ll substitue this into the EB to get:
We can multiply both sides by \(dt\), which converts this from an ODE that describes how temperature changes with time to one that describes how temperature changes with fractional conversion of A:
That almost looks like an equation we can solve, but it isn’t 100% clear that it is a separable ODE just yet. Specifically, I know for sure that \(N_A\), \(N_B\), and \(N_C\) depend on \(X_A\), so we need to specify that dependence before we can integrate the right hand side. Furthermore, I know that heat capacities and heats of reaction will generally change with temperature, so I need to figure out those functional dependencies and determine whether or not they need to be integrated with respect to temperature and/or conversion.
We’ll do this term by term.
First, in the problem statement, we are given heat capacities for each species. They are all clearly constant. They are temperature independent, so we can be confident that each heat capacity in the above ODE is a constant.
Now we address the heat of reaction, which we always expect to change with temperature.
Whether or not the heat of reaction changes with temperature depends on the value of \(\Delta C_p\) for that reaction. We define the \(\Delta C_p\) for reaction \(i\) as:
For this problem:
Substituting numbers:
Thus, for this specific case:
And:
Next, we’ll look at the summation of molar quantities multiplied by heat capacities:
We’ll express everything on the right hand side as a function of fractional conversion; this is based on a mole table, from which we get:
Substituting these into that summation term:
We can simplify that using summation notation:
And the summation on the right hand side is just the \(\Delta C_p\) for this reaction, so:
For this specific reaction, we have proven that \(\Delta C_p = 0\), so:
In other words, this summation on the left hand side is dependent only on the initial numbers of moles in the system, and it is not a function of fractional conversion:
With all of this, we return to our ODE and make appropriate substitutions:
All of the terms before \(dX_A\) on the right hand side are constants. They depend neither on \(X_A\) or \(T\), so we can easily solve the integral:
Because everything on the right hand side is constant, this becomes:
And the integral evaluation is straightforward:
The maximum temperature increase occurs when the reaction reaches completion at \(X_A = 1\); plugging this and values from the problem statment into the solution, we find that:
Example Problem 03#
If the reactor operates adiabatically, how long does it take to reach 95% conversion? Plot \(X_A\), \(T\), and rate vs time for this case. Compare your result to isothermal operation at 300K
Solution to Example Problem 03#
Here, we have to work in the time domain; the problem asks how long it will take to achieve a certain conversion. We approach this almost identically to part A. We start by writing material balances:
We can express production rates in terms of the reaction rate:
This reaction follows elementary kinetics, so the rate law is:
We are solving ODEs in the time domain with 3 state (dependent) variables: NA, NB, and NC. I can express my concentrations in terms of these quantities:
The only difference between this and part A is that the reaction temperature is changing (since the reactor is adiabatic). For that reason, the rate constant is changing as a function of time. It is no longer fixed, so we need a framework to calculate it as reaction time changes. We have an expression for the rate constant given in the problem statement:
Almost everything is given in the problem statement. We know EA/R, \(k_0\), and \(T_0\). The only thing we don’t know yet is the reaction temperature. It is changing as a function of time, so we need to write the energy balance to account for this.
We start by setting \(\dot{Q} = 0\) since this reaction is adiabatic. We also recognize that we only have a single reaction. Finally, this system involves 3 species: A, B, and C, so we can expand the summation term on the left hand side. This balance simplifies to:
That may look a little new, but it’s fairly straightforward. It’s a fourth time-domain differential equation that is coupled to the balances on A, B, and C. We can solve all four ODEs simultaneously with solve_ivp as usual. All of the terms on the right hand side are either constant (\(\Delta H\), \(C_{p,j}\), \(V\)), or they are being tracked by our ODE solver (\(N_A\), \(N_B\), and \(N_C\)).
We have everything we need to solve the ODE system. See below.
def adiabatic(t, var):
    NA = var[0]
    NB = var[1]
    NC = var[2]
    T  = var[3]
    
    V   = 1200 #L
    EAR = 2660 #K 
    DH  = -10*1000 #cal/mol
    CPA = 20 #cal/mol/K
    CPB = 20 #cal/mol/K
    CPC = 40 #cal/mol/K
    
    CA = NA/V
    CB = NB/V
    CC = NC/V
    
    k0 = 0.01725 #L/mol/min
    k  = k0*np.exp(-EAR*(1/T - 1/T0))
    
    r  =  k*CA*CB
    RA = -r
    RB = -r
    RC =  r
    
    Qdot = 0
    
    D1 = RA*V
    D2 = RB*V
    D3 = RC*V
    DT = (-DH*r*V + Qdot)/(NA*CPA + NB*CPB + NC*CPC)
    return [D1, D2, D3, DT]
##problem statement information
k0    = 0.01725 #L/mol/min
EAR   = 2660 #K
V0    = 1200
CA0   = 2.0 #mol/L
CB0   = 2.0 #mol/L
CC0   = 0.0
NA0   = CA0*V0
NB0   = CB0*V0
NC0   = CC0*V0
T0    = 300 #K
#set up IVP solver, solve problem
var0    = [NA0, NB0, NC0, T0]
tspan = (0.0, 50.0)
ansb  = solve_ivp(adiabatic, tspan, var0, atol = 1e-10, rtol = 1e-10)
#Workup solution to get requested quantities
t     = ansb.t
NA    = ansb.y[0,:]
NB    = ansb.y[1,:]
T     = ansb.y[3,:]
CA    = NA/V0
CB    = NB/V0
XA    = (NA0 - NA)/NA0
k     = k0*np.exp(-EAR*(1/T - 1/T0))
r     = k*CA*CB
itpb  = interp1d(XA, t, kind = 'cubic')
tsolb = itpb(0.95)
print(f'For adiabatic operation, it will take {tsolb:3.0f} minutes to achieve 95% conversion')
##Plot of conversion vs. time
plt.figure(1, figsize = (5, 5))
plt.plot(t, XA)
plt.xlabel('time (min)', fontsize = 12)
plt.ylabel('XA', fontsize = 12)
plt.xlim(0, max(tspan))
plt.ylim(0, 1)
plt.show()
##Create plot with 2 yaxes to plot Conversion and Temperature vs. time
fig1, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
conv = ax1.plot(t, XA, color = 'black', label = 'XA')
temp = ax2.plot(t, T, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, 1)
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('XA', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
##Create a plot with two y axes to show rate and temperature vs. time
fig2, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
rate = ax1.plot(t, r, color = 'black', label = 'rate')
temp = ax2.plot(t, T, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, max(r))
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('rate (mol/L/min)', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
##Create a plot with two y axes to show rate constant and temperature vs. time
fig3, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
cons = ax1.plot(t, k, color = 'black', label = 'k')
temp = ax2.plot(t, T, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, max(k))
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('rate constant (L/mol/min)', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
For adiabatic operation, it will take  20 minutes to achieve 95% conversion
 
 
 
 
Example Problem 04#
Assume the reactor operates nonadiabatically with heat exchange. For this case, UA = 12 kcal/min/K and the temperature of the heat transfer fluid is constant at 300K. How long does it take to achieve 95% conversion? Plot XA, T, and rate vs. time in this reactor; consider the impacts of changing heat transfer rates by adjusting the temperature of the heat transfer fluid or the value of UA (heat transfer coefficient times interfacial area)
Solution to Example Problem 04#
def heatex(t, var):
    NA = var[0]
    NB = var[1]
    NC = var[2]
    T  = var[3]
    
    V   = 1200 #L
    EAR = 2660 #K 
    DH  = -10*1000 #cal/mol
    CPA = 20 #cal/mol/K
    CPB = 20 #cal/mol/K
    CPC = 40 #cal/mol/K
    UA  = 12*1000 #cal/min/K
    Ta  = 300  #K
    
    CA = NA/V
    CB = NB/V
    CC = NC/V
    
    k0 = 0.01725 #L/mol/min
    k  = k0*np.exp(-EAR*(1/T - 1/T0))
    
    r  =  k*CA*CB
    RA = -r
    RB = -r
    RC =  r
    
    Qdot = UA*(Ta - T)
    
    D1 = RA*V
    D2 = RB*V
    D3 = RC*V
    DT = (-DH*r*V + Qdot)/(NA*CPA + NB*CPB + NC*CPC)
    return [D1, D2, D3, DT]
##problem statement information
k0    = 0.01725 #L/mol/min
EAR   = 2660 #K
V0    = 1200
CA0   = 2.0 #mol/L
CB0   = 2.0 #mol/L
CC0   = 0.0
NA0   = CA0*V0
NB0   = CB0*V0
NC0   = CC0*V0
T0    = 300 #K
#set up IVP solver, solve problem
var0    = [NA0, NB0, NC0, T0]
tspan = (0.0, 500.0)
ansc  = solve_ivp(heatex, tspan, var0, atol = 1e-10, rtol = 1e-10)
#Workup solution to get requested quantities
t     = ansc.t
NA    = ansc.y[0,:]
NB    = ansc.y[1,:]
T     = ansc.y[3,:]
CA    = NA/V0
CB    = NB/V0
XA    = (NA0 - NA)/NA0
k     = k0*np.exp(-EAR*(1/T - 1/T0))
r     = k*CA*CB
itpc  = interp1d(XA, t, kind = 'cubic')
tsolc = itpc(0.95)
print(f'For operation with heat exchange, it will take {tsolc:3.0f} minutes to achieve 95% conversion')
##Plot of conversion vs. time
plt.figure(1, figsize = (5, 5))
plt.plot(t, XA)
plt.xlabel('time (min)', fontsize = 12)
plt.ylabel('XA', fontsize = 12)
plt.xlim(0, max(tspan))
plt.ylim(0, 1)
plt.show()
##Create plot with 2 yaxes to plot Conversion and Temperature vs. time
fig1, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
conv = ax1.plot(t, XA, color = 'black', label = 'XA')
temp = ax2.plot(t, T, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, 1)
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('XA', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
##Create a plot with two y axes to show rate and temperature vs. time
fig2, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
rate = ax1.plot(t, r, color = 'black', label = 'rate')
temp = ax2.plot(t, T, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, max(r))
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('rate (mol/L/min)', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
##Create a plot with two y axes to show rate constant and temperature vs. time
fig3, ax1 = plt.subplots(figsize = (5, 5))
ax2  = ax1.twinx()
cons = ax1.plot(t, k, color = 'black', label = 'k')
temp = ax2.plot(t, T, color = 'red', label = 'Operating Temperature')
ax1.set_xlim(0, max(tspan))
ax1.set_ylim(0, max(k))
ax2.set_ylim(250, 600)
ax1.set_xlabel('time', fontsize = 12)
ax1.set_ylabel('rate constant (L/mol/min)', fontsize = 12)
ax2.set_ylabel('T(K)', fontsize = 12)
ax1.legend(loc = 'upper left')
ax2.legend(loc = 'lower right')
plt.show()
For operation with heat exchange, it will take 455 minutes to achieve 95% conversion
 
 
 
 
