Material Balances XIV#

This lecture continues the discussion of reactor design for systems involving multiple reactions.

import numpy as np
import matplotlib.pyplot as plt
import scipy.optimize as opt
from scipy.integrate import solve_ivp
from scipy.interpolate import interp1d

Example Problem 01: Revisiting Benzene Pyrolysis in a PFR#

You are carrying out Benzene Pyrolysis in a Plug Flow Reactor operating at 1033K and 1.0 atm. The two reactions occurring in this system are benzene coupling to form diphenyl and hydrogen followed by a secondary reaction between benzene and diphenyl to form triphenyl and hydrogen:

2BD+H2B+DT+H2

Both reactions are reversible and follow elementary rate laws. Rate constants and equilibrium concentration ratios (KC) are given below.

k1=7.0 ×105 L mol1 h1k2=4.0 ×105 L mol1 h1KC1=0.31KC2=0.48

If pure benzene is fed into the reactor at 60,000 moles per hour, find the PFR volume required to achieve 50% conversion of Benzene. Also, plot the mole fraction of each species as a function of PFR volume.

Answer: 403.3 L

Solution to Example Problem 01#

This is exactly the same as our solution from L22, we are just going to expand consideration of the PFR.

Even though there are multiple reactions, we approach this problem the same way as usual: we write a material balance on Benzene.

dFBdV=RB

We generally know that RB is going to be a complex function of concentrations of all of the species present in this system. That means we can’t solve the above balance on Benzene without also solving balances on Diphenyl, Hydrogen, and Triphenyl at the same time, i.e., we have to solve the coupled system of differential equations below:

(98)#dFBdV=RBdFDdV=RDdFHdV=RHdFTdV=RT

So, we’ve got 4 differential equations that tell us how 4 dependent variables (FB, FD, FH, FT) change as a function of the one independent variable, V. We can solve this system if we can define everything on the right hand sides of the above equations (RB, RD, RH, and RT) in terms of FB, FD, FH, FT, and/or V.

We know how to do this!!

Generally:

Rj=iνi,jri

So:

(99)#RB=2r1r2RD=r1r2RH=r2+r2RT=r2

We next define reaction rates:

(100)#r1=k1,fCB2k1,rCHCDr2=k2,fCBCDk2,rCHCT

We define concentrations in terms of molar flowrates:

Fj=FjQ

Which throws a volumetric flowrate, Q, into the mix. Fortunately, this is a gas phase reaction at low pressure, so we know we can define Q in terms of the total molar flowrate:

Q=FtotRTP

Where

Ftot=jFj

With that, we’ve defined out system of ODES fully as a function of molar flowrates and reactor volume. We can solve this numerically using solve_ivp. See below

def P01(vol, var):
    
    #Dependent variables are all in var
    FB, FD, FH, FT = var
    
    #Constants from problem statement
    T   = 1033    #K
    P   = 1.0     #atm
    R   = 0.08206 #L*atm/mol/K
    k1f = 7.0e5   #L/mol/h
    k2f = 4.0e5   #L/mol/h
    KC1 = 0.31
    KC2 = 0.48
    k1r = k1f/KC1 #L/mol/h
    k2r = k2f/KC2 #L/mol/h=
    
    #total molar flowrate, function of individual molar flowrates
    FTOT = FB + FD + FH + FT
    
    #volumetric flowrate
    Q    = FTOT*R*T/P
    
    #Define concentrations
    CB   = FB/Q
    CD   = FD/Q
    CH   = FH/Q
    CT   = FT/Q
    
    #now that we have concentrations, we define reaction rates
    r1   = k1f*CB**2 - k1r*CD*CH
    r2   = k2f*CB*CD - k2r*CT*CH
    
    #With reaction rates, you can define production rates
    RB   = -2*r1 - r2
    RD   =    r1 - r2
    RH   =    r1 + r2
    RT   =         r2
    
    #For a PFR, dFj/dV = Rj, so these are our derivatives of FB, FD, FH, FT
    D1   = RB
    D2   = RD
    D3   = RH
    D4   = RT
    
    #return derivatives of each dependent variable w.r.t. volume
    return [D1, D2, D3, D4]
#This cell solves the ODE system (an initial value problem) using solve_ivp
FBf     = 60000 #mol/h
FDf     = 0
FHf     = 0
FTf     = 0

#Solve the problem
vspan   = (0, 5000)
var0    = [FBf, FDf, FHf, FTf]
solP01 = solve_ivp(P01, vspan, var0, atol = 1e-12, rtol = 1e-12)

#Extract data from the solution structure.

#Volumes from the ODE solver
Vout    = solP01.t

#Molar flowrates as a function of volume.
FBout   = solP01.y[0]
FDout   = solP01.y[1]
FHout   = solP01.y[2]
FTout   = solP01.y[3]

#Sum things up to get the total molar flowrate as a function of volume.
FTot_o  = FBout + FDout + FHout + FTout

#Calculate mole fractions as a function of volume.
yBout   = FBout/FTot_o
yDout   = FDout/FTot_o
yHout   = FHout/FTot_o
yTout   = FTout/FTot_o

#Calculate conversion
XBout   = (FBf - FBout)/FBf

#Plot conversion vs. reactor volume.
plt.figure(1, figsize = (5, 5))
plt.title('Benzene Conversion vs. PFR Volume', fontsize = 14)
plt.plot(Vout, XBout)
plt.xlim(0, max(vspan))
plt.ylim(0, 0.6)
plt.hlines(0.5, 0, max(vspan), linestyle = 'dashed', color = 'black', linewidth = 1)
plt.xlabel('Volume (L)', fontsize = 14)
plt.ylabel('Benzene Conversion', fontsize = 14)
plt.show(1)
../_images/587-L23_5_0.png

Consider Maximixing Diphenyl Production as opposed to Benzene Conversion#

We already know (from L21) that we need a 403L PFR in order to achieve 50% conversion of Benzene. A different question might be what volume should the reactor be. As we discussed in class, that depends on our goal. If it is to maximize Benzene conversion, then something like a 3000L reactor…but if it is to maximize the production of a certain compound, like diphenyl, we’d make a different choice. In fact, if we really wanted to maximize Diphenyl production, we’d probably want to stick with a reactor volume of about 400L because that is where the mole fraction of diphenyl is at a maximum. This leads us into a discussion about optimization, which is the third type of numerical method we will use in this course (the first one being algebraic equation solvers like those in opt.root() or opt.newton() and the second one being ODE/Initial Value Problem solvers like those in solve_ivp()). We will take a look at numerical methods for optimization (minimization) in Recitation 08 on Thursday.

#Plot mole fractions vs. volume
plt.figure(2, figsize = (5, 5))
plt.title('Species Mole Fractions vs. PFR Volume', fontsize = 14)
plt.plot(Vout, yBout, label = 'yB')
plt.plot(Vout, yDout, label = 'yD')
plt.plot(Vout, yHout, label = 'yH')
plt.plot(Vout, yTout, label = 'yT')
plt.xlim(0, max(vspan))
plt.ylim(0, 1)
plt.xlabel('Volume (L)', fontsize = 14)
plt.ylabel('mole fraction', fontsize = 14)
plt.legend()
plt.show(2)

#Plot mole fractions vs. volume
plt.figure(3, figsize = (5, 5))
plt.title('Finding the maximum value for yD', fontsize = 14)
plt.plot(Vout, yBout, label = 'yB')
plt.plot(Vout, yDout, label = 'yD')
plt.plot(Vout, yHout, label = 'yH')
plt.plot(Vout, yTout, label = 'yT')
plt.xlim(0, 800)
plt.ylim(0, 0.25)
plt.xlabel('Volume (L)', fontsize = 14)
plt.ylabel('mole fraction', fontsize = 14)
plt.legend()
plt.show(3)
../_images/587-L23_7_0.png ../_images/587-L23_7_1.png

Using max() and np.argmax()#

For this example, since we’ve already solved for mole fractions as a function of volume, it is relatively easy to find the PFR volume that would maximize production of diphenyl.

Note: This is a relatively crude optimization because we are finding the maximum mole fraction of diphenyl that the solver stepped to when solving the ODE system. The true maximum is probably slightly different from this.
yDmax1 = max(yDout)
print(f'The maximum diphenyl mole fraction is {yDmax1:0.3f}. \n')

indexmax = np.argmax(yDout)
yDmax2   = yDout[indexmax]
Vmax2    = Vout[indexmax]
print(f'The maximum diphenyl mole fraction is {yDmax2:0.3f}, which we obtain at a PFR volume of {Vmax2:0.0f}L.')
The maximum diphenyl mole fraction is 0.205. 

The maximum diphenyl mole fraction is 0.205, which we obtain at a PFR volume of 488L.

Introducing Yield and Selectivity#

The benzene pyrolysis examples highlight the fact that conversion alone is not an adequate process specification when we’re dealing with multiple reactions. We also need a way to quantify the amounts of each product that we make. This is done using two metrics: Yield and Selectivity.

We define the yield as the percentage of our reactant that was converted into a specific product.

We define the overall selectivity as the percentage of consumed reactant that was converted into a specific product.

Consider A generic pair of reactions

Let’s discuss two reactions. The first one converts our reactant, A, into a desired product (valuable, non-toxic, beneficial to society somehow, etc). The second one occurs in parallel and converts our reactant A into something undesirable (toxic, costly to dispose, greenhouse gas, etc). We would like to establish a general way for discussing the amounts of D and U that are produced in reaction.

(101)#(1)νA1AνD1D(2)νA2AνD1U

Yields

Consider the yield: This would be the percentage of our reactant, A, that is converted into each product. So, for a batch reactor (where we usually work with moles of species), the “yield of D with respect to A” would be defined conceptually as: the number of moles of A converted into species D divided by the initial number of moles of A put into the system. In equation form, that looks like:

In a batch reactor

YD/A=|νA1νD1|NDNA0

Similarly, we would define the yield of U with respect to A as the number of moles of A converted into species U divided by the initial number of moles of A put into the system, i.e.:

YU/A=|νA2νU2|NUNA0

In a Flow Reactor

If we were working with a flow reactor, it usually makes more sense to work with molar flowrates instead of number of moles, so we would define the two yields as:

YD/A=|νA1νD1|FDFAf

and

YU/A=|νA2νU2|FUFAf

Selectivities

For now, we’ll only work with a quantity that we call the “Overall Selectivity,” which is defined as the percentage of the consumed reactant that was converted into a specific product. So, at a conceptual level, if I wanted to define the “overall selectivity of D with respect to A,” that would be defined as the number of moles of A that were converted to make D divided by the total number of moles of A consumed. In Equation form:

In a Batch Reactor

SD/A=|νA1νD1|NDNA0NA

And the “overall selectivity of U with respect to A” is given by the number of moles of A converted to make U divided by the total number of moles of A consumed.

SU/A=|νA2νU2|NUNA0NA

In a Flow Reactor

And, similar to yields, if I’m working with a flow process, I will define these quantities in terms of molar flowrates.

SD/A=|νA1νD1|FDFAfFA

and

SU/A=|νA2νU2|FUFAfFA

Fractional Conversion

These definitions complement the one for fractional conversion:

Batch Reactor

XA=NA0NANA0

Flow Reactor

XA=FAfFAFAf

As a final note:

With the definitions above, fractional conversion, selectivity, and yield will always be between 0 and 1.

Example Problem 02: Benzene Pyrolysis with Yields and Selectivities#

You are carrying out Benzene Pyrolysis in a Plug Flow Reactor operating at 1033K and 1.0 atm. The two reactions occurring in this system are benzene coupling to form diphenyl and hydrogen followed by a secondary reaction between benzene and diphenyl to form triphenyl and hydrogen:

2BD+H2B+DT+H2

Both reactions are reversible and follow elementary rate laws. Rate constants and equilibrium concentration ratios (KC) are given below.

k1=7.0 ×105 L mol1 h1k2=4.0 ×105 L mol1 h1KC1=0.31KC2=0.48

Pure benzene is fed into the reactor at 60,000 moles per hour. Plot the overall yield and overall selectivity toward diphenyl (with respect to benzene) as a function of reactor volume.

Solution to Example Problem 02#

We solve this essentially the same way as we did Example Problem 01 – we write the material balances for all species in the reactor, and we solve the resultant coupled system of ODEs using solve_ivp(). This part is identical to the solution from L22.

We recognize that solution of the problem using solve_ivp() will give us the molar flowrates of Benzene, Diphenyl, Triphenyl, and Hydrogen as a function of reactor volume. This is enough information for us to calculate yields and selectivity as a function of reactor volume since both of those quantities are defined in terms of molar flowrates for flow reactors. Specifically:

Yield of Diphenyl with respect to Benzene

According to the first reaction, each diphenyl produced will consume two benzenes, so the correct definition for yield according to the equations in the preceding section is:

YD/B=2FDFBf

The coefficients are less clear for triphenyl since each triphenyl consumes one diphenyl…which consumes two benzenes. The easiest way to handle the yield definition for a sequential reaction like this is to add the overall reactions so that we can relate triphenyl directly to benzene:

(102)#2BD+HB+DT+H3BT+2H

From that overall equation, it becomes clear that it takes 3 benzenes to make one triphenyl, so the yield definition is:

YT/B=3FTFBf

We extend these ideas to the selectivity definitions, which quantify the percentage of consumed reactant that went to produce diphenyl or triphenyl.

Selectivity to diphenyl with respect to benzene:

SD/B=2FDFBfFB

Selectivity to triphenyl with respect to benzene:

ST/B=3FTFBfFB

It is also worth pointing out the the product of fractional conversion and product selectivity (when defined as above) will give the yield to a specific product:

YD/B=XBSD/B=FBfFBFBf2FDFBfFB=2FDFBf
YT/B=XBST/B=FBfFBFBf3FTFBfFB=3FTFBf
## Note that we can still use the function that contains the ODE system in P01(vol, var) above. 
## Nothing about the problem has changed other than how we process the ODE solution.
## Commented sections of cell here were solved above; I'm adding just to remind us.
## No need to re-solve.

# #This cell solves the ODE system (an initial value problem) using solve_ivp
# FBf     = 60000 #mol/h
# FDf     = 0
# FHf     = 0
# FTf     = 0

# #Solve the problem
# vspan   = (0, 5000)
# var0    = [FBf, FDf, FHf, FTf]
# solP01 = solve_ivp(P01, vspan, var0, atol = 1e-12, rtol = 1e-12)

# #Extract data from the solution structure.

# #Volumes from the ODE solver
# Vout    = solP01.t

# #Molar flowrates as a function of volume.
# FBout   = solP01.y[0]
# FDout   = solP01.y[1]
# FHout   = solP01.y[2]
# FTout   = solP01.y[3]

# #Sum things up to get the total molar flowrate as a function of volume.
# FTot_o  = FBout + FDout + FHout + FTout

# #Calculate mole fractions as a function of volume.
# yBout   = FBout/FTot_o
# yDout   = FDout/FTot_o
# yHout   = FHout/FTot_o
# yTout   = FTout/FTot_o

# #Calculate conversion
# XBout   = (FBf - FBout)/FBf

#Calculate diphenyl/triphenyl yield
YDB     = FDout*2/FBf
YTB     = FTout*3/FBf

#Calculate diphenyl selectivity
SDB     = FDout*2/(FBf - FBout)
STB     = FTout*3/(FBf - FBout)

#Plot yield, selectivity, conversion vs. reactor volume.
plt.figure(1, figsize = (5, 5))
plt.title('Conversion and Diphenyl Yield in a PFR', fontsize = 14)
plt.plot(Vout, XBout, label = 'Conversion')
plt.plot(Vout, YDB, label = 'Yield to D')
plt.xlim(0, max(vspan))
plt.ylim(0, 1)
plt.xlabel('Volume (L)', fontsize = 14)
plt.ylabel('Benzene Conversion/Diphenyl Yield', fontsize = 14)
plt.legend()
plt.show(1)

plt.figure(2, figsize = (5, 5))
plt.title('Diphenyl and Triphenyl Selectivity', fontsize = 14)
plt.plot(XBout, SDB, label = 'Selectivity to D')
plt.plot(XBout, STB, label = 'Selectivity to T')
plt.xlim(0, 1)
plt.ylim(0, 1)
plt.xlabel('Benzene Conversion', fontsize = 14)
plt.ylabel('Product Selectivity', fontsize = 14)
plt.legend()
plt.show(2)

plt.figure(3, figsize = (5, 5))
plt.title('Diphenyl and Triphenyl Yield', fontsize = 14)
plt.plot(XBout, YDB, label = 'Yield to D')
plt.plot(XBout, YTB, label = 'Yield to T')
plt.xlim(0, 1)
plt.ylim(0, 1)
plt.xlabel('Benzene Conversion', fontsize = 14)
plt.ylabel('Product Yield', fontsize = 14)
plt.legend()
plt.show(3)
C:\Users\jqbond\AppData\Local\Temp\ipykernel_9132\1217347526.py:43: RuntimeWarning: invalid value encountered in true_divide
  SDB     = FDout*2/(FBf - FBout)
C:\Users\jqbond\AppData\Local\Temp\ipykernel_9132\1217347526.py:44: RuntimeWarning: invalid value encountered in true_divide
  STB     = FTout*3/(FBf - FBout)
../_images/587-L23_14_1.png ../_images/587-L23_14_2.png ../_images/587-L23_14_3.png

Finding the value and location of the maximum diphenyl yield#

We’ll use max() and np.argmax() as above to get a crude approximation of the maximum, and we’ll refine that a bit by creating an interpolating polynomial that returns Volume as a function of benzene yield. We’ll use that with opt.minimize_scalar() to find the location of the optimum. We’ll cover this in more detail in Recitation 08.

YDBmax1 = max(YDB)
print(f'The maximum diphenyl yield is approximately {YDBmax1:0.3f}. \n')

indexmax = np.argmax(YDB)
YDBmax2   = YDB[indexmax]
Vmax2    = Vout[indexmax]
print(f'The maximum diphenyl yield is {YDBmax2:0.3f}, which we obtain at a PFR volume of {Vmax2:0.0f}L. \n')

YvVol = interp1d(Vout, YDB)
objtemp = lambda V: -1*YvVol(V)
opt_answer = opt.minimize_scalar(objtemp)
print(f'The maximum yield is {-1*opt_answer.fun:3.3f} at a PFR volume of {opt_answer.x:3.0f}L')
The maximum diphenyl yield is approximately 0.409. 

The maximum diphenyl yield is 0.409, which we obtain at a PFR volume of 488L. 

The maximum yield is 0.409 at a PFR volume of 488L

Example Problem 03#

This is a slightly modified version of Example 8.4 from Fogler

The following pair of liquid-phase reactions occur in series. Each reaction is first order in its reactant.

(103)#ABBC

By “series” or “sequential” reactions we mean that a product of the first reaction is a reactant in a second reaction, and so on. These are very common systems in Chemical Reaction Engineering – especially in the petrochemical industry, where we often make reactive things from inert alkanes. Quite often, the reactive things we make then react away to form things we don’t want. Selectivity control in series reactions requires careful attention to operating conditions and reaction times (space times for flow reactors).

You carry out these reactions in a CSTR operating at steady state. You may assume the density of the fluid does not change with reaction and so the inlet volumetric flowrate is equal to the outlet volumetric flowrate. You have the following data available:

(104)#k1=0.5 h1k2=0.2 h1CAf=20 mol L1

Find the space time required to maximize the yield of B (with respect to A) in this CSTR.

Solution to Example 03#

Balance on A#

To get yields, selectivities, and conversions, we need to write material balances on all species. This allows us to determine the exit molar flowrates of all species. Starting with a balance on A:

0=FA,fFA+RAV

We aren’t given a ton of information except for the feed concentration of A. Presumably, the feed concentrations of B and C are both zero. We are also told we want to work with a space time, which is defined for a CSTR as:

τ=VQf

I’m going to convert my balance so that it is written in terms of concentration and space time instead of molar flowrate and volume. I can express molar flowrates in terms of concentrations and volumetric flowrates:

0=CA,fQfCAQ+RAV

We know the intensive production rate of A:

RA=r1

Since the reactions are first order in the reactant:

r1=k1CA

Making these substitutions, we find:

0=CA,fQfCAQk1CAV

We are told that we can assume constant density in this system, which means (at steady state)

Q=Qf

Thus:

0=CA,fQfCAQfk1CAV

Dividing by Qf, we express the balance in terms of a space time and concentrations.

0=CA,fCAk1CAτ

And we can solve this for CA, the exit concentration of A from the CSTR:

CA=CA,f1+k1τ

This is an important result. It gives us the exit concentration of A as a function of space time. Now we move on to a balance on B:

Balance on B#

0=FB,fFB+RBV

We can replace molar flowrates with the product of concentration and volumetric flowrate, noting again that Q=Qf for this problem:

0=CB,fQfCBQf+RBV

We can define the intensive production rate of B, RB:

RB=r1r2

We know the reactions are both first order in their reactant, so:

RB=k1CAk2CB

And we can substitute this into the material balance:

0=CB,fQfCBQf+(k1CAk2CB)V

Dividing through by Qf:

0=CB,fCB+(k1CAk2CB)τ

Noting that CB,f=0:

0=CB+(k1CAk2CB)τ

And we can solve this for CB:

CB=k1CAτ1+k2τ

The above is expressed in terms of CA. We can substitute that expression in to get:

CB=k1CA,fτ(1+k1τ)(1+k2τ)

This is another important equation as it gives us CB as a function of τ.

Balance on C#

Technically, we don’t need this to solve for yield/selectivity to B, but I’m curious to see how it behaves, so we’ll go ahead and write a balance on C:

0=FC,fFC+RCV

Converting to a function of concentration:

0=CC,fQfCCQf+RCV

Defining the production rate for C:

RC=r2=k2CB

We get:

0=CC,fQfCCQf+k2CBV

Dividing by Qf:

0=CC,fCC+k2CBτ

Noting that Cf=0:

0=CC+k2CBτ

Which we can solve for CC:

CC=k2CBτ

And we can substitute the expression for CB here to get:

CC=k1k2CA,fτ2(1+k1τ)(1+k2τ)

There we have our concentrations expressed as 3 functions of tau. These are developed as lambda functions below, which are plotted as a function of tau.

k1 = 0.5 #1/h
k2 = 0.2 #1/h
CAf = 20 #mol/L

CA = lambda tau: CAf/(1 + k1*tau)
CB = lambda tau: k1*CAf*tau/(1+k1*tau)/(1+k2*tau)
CC = lambda tau: k1*k2*CAf*tau**2/(1+k1*tau)/(1+k2*tau)
tauset = np.linspace(0, 30, 200)
plt.figure(1, figsize = (5, 5))
plt.plot(tauset, CA(tauset), label = 'CA')
plt.plot(tauset, CB(tauset), label = 'CB')
plt.plot(tauset, CC(tauset), label = 'CC')
plt.title('Concentration vs. space time (τ)', fontsize = 12)
plt.xlabel('τ (h)', fontsize = 14)
plt.ylabel('Concentration (mol/L)', fontsize = 14)
plt.xlim(0, max(tauset))
plt.ylim(0, 20)
plt.legend()
plt.show(1)
../_images/587-L23_20_0.png

Definining Conversion, Yield, and Selectivity#

With the concentrations of each species defined, we can now consider Conversion, Yields and Selectivities.

Conversion#

We define the fractional conversion of A as:

XA=FA,fFAFA,f

Although we have no way to solve for molar flowrates (since Q is unknown), we can express them in terms of concentrations, noting symbolically that Q=Qf.

XA=CA,fQfCAQfCA,fQf

We can factor and cancel the volumetric flowrate, giving us the result below, which is always true for a constant density fluid:

XA=CA,fCACA,f

Yields#

For a CSTR, we would generally define the yield of B with respect to A as:

YB/A=FB|νA1νB1|FA,f

Which simplfies for this problem to:

YB/A=FBFA,f

We actually don’t know the molar flowrates since we don’t know the size (Q, V) of this system, but we can express these in terms of concentrations, noting that since density is constant Q=Qf:

YB/A=CBQfCA,fQf

This reduces to:

YB/A=CBCA,f

When we go to calculate a yield of C with respect to A, we have to note that by summing the two reactions, we can get the overall stoichiometry of converting A into C:

AC

With that, we can use a similar approach to that above to define the yield to C in terms of concentrations:

YC/A=CBCA,f

Selectivities#

We can define the selectivity of species B with respect to A:

SB/A=FB|νA1νB1|FA,fFA

Substituting stoichiometric coefficients, expressing in terms of concentrations, and noting that Q=Qf:

SB/A=CBQfCA,fQfCAQf

Factoring and cancelling the volumetric flowrate (which we can do in cases where it is constant):

SB/A=CBCA,fCA

You can follow similar steps to define a selectivity to species C with respect to the reactant A:

SC/A=CCCA,fCA

We’ll write lambda functions for each of these; note that I’m re-using my concentration functions defined above.

YB = lambda tau: CB(tau)/CAf
YC = lambda tau: CC(tau)/CAf
SB = lambda tau: CB(tau)/(CAf - CA(tau))
SC = lambda tau: CC(tau)/(CAf - CA(tau))
XA = lambda tau: (CAf - CA(tau))/CAf
#This just prints out the first few selectivities to show the NAN ("not a number") error
print(np.array([SB(tauset)[0:5], SC(tauset)[0:5]]).T) 

plt.figure(1, figsize = (5, 5))
plt.title('Conversion and Selectivity in a CSTR', fontsize = 14)
plt.plot(tauset, XA(tauset), label = 'Conversion of A')
plt.plot(tauset, SB(tauset), label = 'Selectivity to B')
plt.plot(tauset, SC(tauset), label = 'Selectivity to C')
plt.xlabel('τ (h)', fontsize = 14)
plt.ylabel('Conversion/Selectivity', fontsize = 14)
plt.xlim(0,30)
plt.ylim(0,1)
plt.legend()
plt.show(1)

plt.figure(2, figsize = (5, 5))
plt.title('Conversion and Yield in a CSTR', fontsize = 14)
plt.plot(tauset, XA(tauset), label = 'Conversion of A')
plt.plot(tauset, YB(tauset), label = 'Yield to B')
plt.plot(tauset, YC(tauset), label = 'Yield to C')
plt.xlabel('τ (h)', fontsize = 14)
plt.ylabel('Conversion/Yield', fontsize = 14)
plt.xlim(0,30)
plt.ylim(0,1)
plt.legend()
plt.show(2)
[[       nan        nan]
 [0.97073171 0.02926829]
 [0.94312796 0.05687204]
 [0.91705069 0.08294931]
 [0.89237668 0.10762332]]
C:\Users\jqbond\AppData\Local\Temp\ipykernel_9132\3616170257.py:3: RuntimeWarning: invalid value encountered in true_divide
  SB = lambda tau: CB(tau)/(CAf - CA(tau))
C:\Users\jqbond\AppData\Local\Temp\ipykernel_9132\3616170257.py:4: RuntimeWarning: invalid value encountered in true_divide
  SC = lambda tau: CC(tau)/(CAf - CA(tau))
../_images/587-L23_23_2.png ../_images/587-L23_23_3.png

Maximizing the Yield of B#

An important question is clearly what space time do we need to use to maximize the yield to species B (the intermediate in the series reaction). From the graph of Yield vs. tau, we can see this occurs at a space time of about 3 hours. To get more precise, we can either solve it analytically by taking a derivative of YB/A with respect to tau; after some calculus and algebra, we get:

dYB/Adτ=k1(1+k1τ)(1+k2τ)k1τ[k1(1+k2τ)+k2(1+k1τ)][(1+k1τ)(1+k1τ)]2

To find the optimum value of tau, we solve the following expression for τ:

0=k1(1+k1τ)(1+k2τ)k1τ[k1(1+k2τ)+k2(1+k1τ)][(1+k1τ)(1+k1τ)]2

I did this by hand to find:

τopt=1k1k2

Which we can plug into a calculator to get:

τopt=3.16 h

Or, you know, we could just create an objective function (return negative yield of B at a specific volume) and minimize that with an optimization routine. See below. You get the same answer.

obj    = lambda tau: -1*YB(tau) #this objective function returns -1*YB for a given value of tau
optsol = opt.minimize_scalar(obj) #this varies tau until we hit minimum value of -1*YB
print(optsol, '\n')

print(f'The maximum yield to species B is {-1*optsol.fun:3.2f} which occurs at a space time of {optsol.x:3.2f} hours')
     fun: -0.3752470442573563
 message: '\nOptimization terminated successfully;\nThe returned value satisfies the termination criteria\n(using xtol = 1.48e-08 )'
    nfev: 17
     nit: 11
 success: True
       x: 3.162277615990361 

The maximum yield to species B is 0.38 which occurs at a space time of 3.16 hours