notesofdabbler avatar notesofdabbler committed 2932ce4

directory structure change done

Comments (0)

Files changed (15)

01_learnR_pt1/20120804_CompApp.Rmd

+# Learning R using a Chemical Reaction Engineering Book
+
+[Chemical Reactor Analysis and Design Fundamentals](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/) by J.B. Rawlings and J. G. Ekerdt is a textbook for studying Chemical Reaction Engineering. The popular open source package [Octave](http://www.gnu.org/software/octave/) has its origins to the reaction engineering course offered by Prof. Rawlings. This book is accompanied by Octave and Matlab code for solving typical problems encountered in Reaction Engineering. 
+
+I figured that maybe one way to learn R is so see whether I can code some of th examples from this book in R. I am by no means suggesting that R can replace MATLAB/Octave for engineering problems but merely it is a way for me to learn the language.
+
+I started with the [computational appendix](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/web-appendix.pdf) listed in the book's website and am trying to work through some of the examples there. It will be good to refer to the computational appendix to follow the R code below.
+
+### Setting up a stoichiometric matrix, reaction rate vector and determining the rank
+
+```{r}
+# stoichiometric matrix
+stoi=matrix(c(0,1,0,-1,-1,1,
+       -1,1,1,-1,0,0,
+       1,0,-1,0,-1,1),
+       ncol=6,byrow=T)
+stoi
+
+# rank of the stoichiometrix matrix
+rank=qr(stoi)$rank
+rank
+
+# reaction rate vector
+r=c(1,2,3)
+```
+Given the reaction rate r=(r1,r2)', the rate of change of species concentration R is given by
+$$latex R=\nu^Tr $$
+where $\nu$ is the stoichiometrix matrix
+```{r}
+# rate of change of components
+R=t(stoi)%*%r
+R
+```
+### Example A1: Estimating reaction rates
+
+The stoichometrix matrix $\nu$ is input below
+```{r}
+# stoichiometry
+stoi=matrix(c(0,1,0,-1,-1,1,
+              -1,1,1,-1,0,0),nrow=2,byrow=T)
+stoi
+# number of species and number of reactions
+nspec=ncol(stoi)
+nr=nrow(stoi)
+
+nspec
+nr
+
+# true rxn rates
+r=c(1,2)
+r
+
+# true component rates
+R=t(stoi)%*%r
+R
+```
+
+### simulate 2000 measured component rates
+
+Add random noise (normally distributed with mean 0 and standard deviation 0.05) to true species rate vector R
+$$ R^m = R + \epsilon, \epsilon \sim N(0,0.0025) $$
+
+```{r}
+## simulate 2000 noise estimates
+e=matrix(0.05*rnorm(2000*nspec,0,1),nrow=2000,byrow=T)
+Rmeas=matrix(rep(R,2000),ncol=nspec,byrow=T)+e
+```
+The least squares estimate of reaction rate vector $\cap{r}$ is
+$$ \hat{r}=(\nu\nu^T)^{-1}{\nu}R $$
+
+```{r}
+## estimate reaction rates
+rest=solve(stoi%*%t(stoi),stoi%*%t(Rmeas))
+```
+I was trying different plot features in R and applying to this data of estimated rates
+
+I found the following function that plots scatterplot with marginal histograms
+```{r}
+# plotting scatterplot with histogram
+# downloaded from web
+#  http://www.r-bloggers.com/example-8-41-scatterplot-with-marginal-histograms/
+#
+scatterhist = function(x, y, xlab="", ylab=""){
+  par.default <- par(no.readonly=TRUE)
+  zones=matrix(c(2,0,1,3), ncol=2, byrow=TRUE)
+  layout(zones, widths=c(4/5,1/5), heights=c(1/5,4/5))
+  xhist = hist(x, plot=FALSE)
+  yhist = hist(y, plot=FALSE)
+  top = max(c(xhist$counts, yhist$counts))
+  par(mar=c(3,3,1,1))
+  plot(x,y)
+  par(mar=c(0,3,1,1))
+  barplot(xhist$counts, axes=FALSE, ylim=c(0, top), space=0)
+  par(mar=c(3,0,1,1))
+  barplot(yhist$counts, axes=FALSE, xlim=c(0, top), space=0, horiz=TRUE)
+  par(oma=c(3,3,0,0))
+  mtext(xlab, side=1, line=1, outer=TRUE, adj=0, 
+        at=0.9 * (mean(x) - min(x))/(max(x)-min(x)))
+  mtext(ylab, side=2, line=1, outer=TRUE, adj=0, 
+        at=(0.9 * (mean(y) - min(y))/(max(y) - min(y))))
+  par=par(par.default)
+}
+
+# scatter plot of reaction rates with marginal histograms
+scatterhist(t(rest)[,1],t(rest)[,2],xlab="r_1",ylab="r_2")
+```
+There is library cars that has a command 'scatterplot' for plotting scatterplot with box plots and has several other options
+In the plot below 50% and 90% ellipses are overlaid on the data
+
+```{r}
+# scatter plot of reaction rates with marginal box plots
+library(car)
+scatterplot(t(rest)[,1],t(rest)[,2],reg.line=FALSE,smooth=FALSE,ellipse=TRUE)
+```
+I tried ggplot2 also for the same plot
+
+```{r}
+# 2d contours/density with ggplot2 for reaction rates
+
+# create dataframe of reaction rates
+rest_df=data.frame(t(rest))
+names(rest_df)=c("r1","r2")
+
+library(ggplot2)
+ggplot(data=rest_df,aes(x=r1,y=r2))+geom_point()
+ggplot(data=rest_df,aes(x=r1,y=r2))+stat_density2d()
+ggplot(data=rest_df,aes(x=r1,y=r2))+stat_density2d(aes(fill=..level..),geom="polygon")
+```
+
+

02_learnR_pt2_3/20120811_CompApp_pt_2_3.Rmd

+# Learning R using a Chemical Reaction Engineering Book (Part 2)
+
+In case you missed part 1, you can view it [here](http://rdabbler.blogspot.com/2012/08/learning-r-using-chemical-reaction.html). In this part, I tried to recreate the examples in sections A.2.2 and A.2.3 of the [computational appendix](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/web-appendix.pdf) in the [reaction engineering book](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/) (by Rawlings and Ekerdt). 
+
+## Solving a nonlinear system of equations
+
+This example involves determining reaction equilibrium conditions by solving the following system of nonlinear equations.
+
+$$latex
+ \begin{aligned}
+   PK_1y_Iy_B-y_{P1}=0, \\
+   PK_2y_Iy_B-y_{P2}=0
+ \end{aligned}   
+$$
+
+The relation between the variables $latex y_I,y_B,y_{P1},y_{P2}$ and extent of reactions $latex x_1,x_2$ are:
+$$latex
+  \begin{aligned}
+    y_I=\frac{y_{I0}-x_1-x_2}{1-x_1-x_2}, \\
+    y_B=\frac{y_{B0}-x_1-x_2}{1-x_1-x_2}, \\ 
+    y_{P1}=\frac{y_{p10}+x_1}{1-x_1-x_2}, \\
+    y_{P2}=\frac{y_{p20}+x_2}{1-x_1-x_2}
+  \end{aligned}
+$$  
+
+Here I have used R package [rootSolve](http://cran.r-project.org/web/packages/rootSolve/index.html) for solving the above set of equations. The library is loaded and the functions to be solved are defined in the R function fns.
+
+```{r}
+# load library rootSolve
+library(rootSolve)
+
+# function defining F(x)=0
+fns=function(x){
+  K1=108; K2=284; P=2.5
+  yI0=0.5; yB0=0.5; yP10=0; yP20=0;
+  d=1-x[1]-x[2]
+  yI=(yI0-x[1]-x[2])/d
+  yB=(yB0-x[1]-x[2])/d
+  yP1=(yP10+x[1])/d
+  yP2=(yP20+x[2])/d
+  F1=P*K1*yI*yB-yP1
+  F2=P*K2*yI*yB-yP2
+  c(F1=F1,F2=F2)
+}
+```
+Next, an initial guess of (0.2,0.2) is set for the variables and the equations are solved using the function multiroot (from package rootSolve)
+
+```{r}
+# initial guess for x
+xinit=c(0.2,0.2)
+
+# solve the equations
+xans=multiroot(f=fns,start=xinit)
+
+# object returned by multiroot
+xans
+
+# solution to the equations
+xans$root
+```
+The solution to the equations is accessed from the variable xans$root which in this case is (0.1334,0.3507)
+
+## Function Minimization
+
+Here the function to be minimized is
+
+
+$$latex
+  G=-(x_1lnK_1+x_2lnK_2)+(1-x_1-x_2)lnP+(y_{I0}-x_1-x_2)ln(y_I)+(y_{B0}-x_1-x_2)ln(y_B)+(y_{P10}+x_1)ln(y_{p1})+(y_{P20}+x_2)ln(y_{P2})
+$$
+
+The following constraints need to be satisfied for $latex x_1$ and $latex x_2$
+
+$$latex
+  0 \leq x_1 \leq 0.5, 0 \leq x_2 \leq 0.5, x_1+x_2 \leq 0.5
+$$latex
+
+First I used [constrOptim](http://stat.ethz.ch/R-manual/R-patched/library/stats/html/constrOptim.html) function for minimization. We need to specify the function to be minimized
+
+```{r}
+# function to be minimized
+eval_f0=function(x){
+  dg1=-3.72e3; dg2=-4.49e3; T=400; R=1.987; P=2.5
+  K1=exp(-dg1/(R*T)); K2=exp(-dg2/(R*T))
+  
+  yI0=0.5; yB0=0.5; yP10=0; yP20=0;
+  d=1-x[1]-x[2]
+  yI=(yI0-x[1]-x[2])/d
+  yB=(yB0-x[1]-x[2])/d
+  yP1=(yP10+x[1])/d
+  yP2=(yP20+x[2])/d
+  
+  f=-(x[1]*log(K1)+x[2]*log(K2))+(1-x[1]-x[2])*log(P)+yI*d*log(yI)+
+       yB*d*log(yB)+yP1*d*log(yP1)+yP2*d*log(yP2)
+  
+  return(f)
+}
+```
+The constraints need to be specified in the form $latex Ax-b \geq 0$
+
+```{r}
+#  constraint
+A=matrix(c(-1,-1,1,0,-1,0,0,1,0,-1),ncol=2,byrow=TRUE)
+A
+b=c(-0.5,0,-0.5,0,-0.5)
+b
+```
+Next, the function is minimized using constrOptim (starting from an initial guess of (0.2,0.2)). Here Nelder-Mead method is used since BFGS method requires specifying the gradient of the function by the user. R taskview [optimization](http://cran.r-project.org/web/views/Optimization.html) lists other options.
+
+```{r}
+# initial guess
+xinit=c(0.2,0.2)
+
+# minimize function subject to bounds and constraints
+xans2=constrOptim(theta=xinit, f=eval_f0, grad=NULL, ui=A, ci=b,
+            method = "Nelder-Mead")
+xans2
+```
+The solution can be accessed from xans2$par and is (0.1331,0.3509)
+
+Next, I also tried function minimization with ConstrOptim.nl from the package [alabama](http://cran.r-project.org/web/packages/alabama/index.html). Here the constraints are specified in terms of $latex h(x) \geq 0$. Even if gradient is not supplied, this function will estimate it using finite-differences.
+
+Definition of constraints in the format for constrOptim.nl
+
+```{r}
+# load library alabama
+library(alabama)
+library(numDeriv)
+
+h_ineq=function(x){
+   h=rep(NA,1)
+   h[1]=-x[1]-x[2]+0.5
+   h[2]=x[1]
+   h[3]=-x[1]+0.5
+   h[4]=x[2]
+   h[5]=-x[2]+0.5
+   return(h)
+}
+
+xans3=constrOptim.nl(par=xinit,fn=eval_f0,hin=h_ineq)
+
+xans3
+```
+The solution can be accessed from xans3$par and is (0.1332,0.3508)
+
+Since this is a 2 dimensional problem, the solution can also be visualized using a contour plot of the function.
+
+```{r message=FALSE,warning=FALSE}
+# Region of interest: 0.01<=x1<=0.49, 0.01<=x2<=0.49
+x1=seq(0.01,0.49,by=0.01)
+x2=seq(0.01,0.49,by=0.01)
+
+# vectorizing function eval_f0 so that it can be evaluted in the outer function
+fcont=function(x,y) eval_f0(c(x,y))
+fcontv=Vectorize(fcont,SIMPLIFY=FALSE)
+z=outer(x1,x2,fcontv)
+
+# filled.coutour and contour are overlaid with the minimum point (0.133,0.351)
+filled.contour(x1,x2,z,xlab="x1",ylab="x2",
+               plot.axes={axis(1); axis(2); contour(x1,x2,z,levels=c(-3,-2.5,-2,-1.5,-1,0),vfont=c("sans serif","bold"),labcex=1,lwd=2,add=TRUE); 
+                          points(0.133,0.351,pch=15,col="blue")})
+```
+
+MATLAB/Octave functions for solving nonlinear equations (fsolve and fmincon) have been used in Chemical Engineering computations for a long time and are robust. R has traditionally not been used in this domain. So it is hard to say how the functions I have used in this blog will perform across the range of problems encountered in Reaction Engineering.
+
+This post was created with R 2.15.1 and knitr 0.7.1 (converted markdown to html)

03_learnR_pt4/20121209_CompApp_pt4.Rmd

+# Learning R using a Chemical Reaction Engineering Book (Part 3)
+
+In case you missed the previous parts, the links to them are listed below:
+* [Part1](http://rdabbler.blogspot.com/2012/08/learning-r-using-chemical-reaction.html). 
+* [Part2](http://rdabbler.blogspot.com/2012/08/learning-r-using-chemical-reaction_11.html)
+
+In this part, I tried to recreate the examples in sections A.3.1 of the [computational appendix](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/web-appendix.pdf) in the [reaction engineering book](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/) (by Rawlings and Ekerdt). 
+
+## Solving a system of ordinary differential equations
+
+This example involves reaction (Benzene pyrolysis) in a plug flow reactor. The actual reactions happening are:
+
+$$ (Rxn1) \; 2B = D + H $$
+$$ (Rxn2) \; B + D = T+H $$
+
+
+The rate of each reaction is given by:
+$$ r_1=k_1(c_B^2-\frac{c_Dc_H}{K_1}) $$
+$$ r_2=k_2(c_Bc_D-\frac{c_Tc_H}{K_2}) $$
+
+The feed to the reactor consists of 60kmol/hr of Benzene (B). The temperature of the reactor is $T=1033K$ and the pressure is $P=1atm$.The rate constants and equilibrium constants for this example are:
+$$ k_1=7\times 10^5\; L/mol.hr,\; k_2=4\times 10^5 \; L/mol.hr,\; K_1=0.31,\; K_2=0.48 $$
+
+```{r, tidy=FALSE}
+# set working directory
+setwd('~/R/RxnEngg_Rawlings')
+
+# load libraries
+library(deSolve)
+
+
+# Appendix A.3.1: Solution of Differential Equations
+
+# Benzene pyrolysis example
+
+# Parameters
+# NBf - feed benzene rate - mol/h
+# R - Universal gas constant
+# T - Reactor temperature K
+# P - Reactor pressure atm
+# k1 - rxn1 forward rate constant L/mol.h
+# k2 - rxn2 forward rate constant L/mol.h
+# Keq1 - rxn1 equilibrium constant
+# Keq2 - rxn2 equilibrium constant
+pars=c(
+NBf=60e3,  
+R=0.08205,  
+T=1033, 
+P=1, 
+k1=7e5, 
+k2=4e5, 
+Keq1=0.31, 
+Keq2=0.48  
+)
+```
+
+The governing equations for conversion versus volume in a plug flow reactor is based on extent of each of the reactions:
+$$ \frac{d\epsilon_1}{dV}=r_1,\; \frac{d\epsilon_2}{dV}=r_2 $$
+The initial conditions (corresponding to feed conditions $N_B(0)=60kmol/h,\;N_D(0)=N_H(0)=N_T(0)=0$) are that the extent of reaction is zero. 
+$$ \epsilon_1(0)=0, \; \epsilon_2(0)=0 $$
+
+The flow rates of each component along the reactor volume can be calculated from reaction extent
+$$ N_B=N_B(0)-2\epsilon_1-\epsilon_2, \; N_D=\epsilon_1-\epsilon_2, \; N_H=\epsilon_1+\epsilon_2, \; N_T=\epsilon_2 $$ 
+
+These are setup in a function that can be passed to an ODE solver. In this case the ODE solver we use lsode from the R package [deSolve](http://desolve.r-forge.r-project.org/). The inputs to the function are:
+* Variable over which the integration is done (Volume in this case)
+* The state variables of the system (Extent of the two reactions)
+* Parameters that are needed for description of the system (Rate constants, Temperature, Pressure, etc.)
+The output from this function is the rate of change as described by the equations previously.
+
+```{r, tidy=FALSE}
+# function that will be passed to odesolver
+# vol is the variable over which the system is integrated (equivalent of time in batch reactions)
+# ext is the extent of reactions 1 and 2
+# params are the parameters passed to the system
+rxnrate=function(vol,ext,params) {
+     with(as.list(c(ext,params)),{
+        NB=NBf-2*ext1-ext2
+        ND=ext1-ext2
+        NH=ext1+ext2
+        NT=ext2
+        Q=NBf*R*T/P
+        cB=NB/Q
+        cD=ND/Q
+        cT=NT/Q
+        cH=NH/Q
+        dext1=k1*(cB*cB-cD*cH/Keq1)
+        dext2=k2*(cB*cD-cT*cH/Keq2)
+        return(list(c(dext1=dext1,dext2=dext2)))
+     })
+}
+```
+
+Since the reaction start only after the feed enters the reactor, the extent of reaction is zero for both reactions at the beginning of the reactor (V=0L). The set of volumes where the concentration and reaction extent is computed is chosen in this case to be from 0L to 1600L at every 50L. The ODE solver lsode from [deSolve](http://desolve.r-forge.r-project.org/) package is used to solve the system of equations.
+
+```{r, tidy=FALSE}
+# initial extent of reaction (zero in this case for both reactions)
+extinit=c(ext1=0,ext2=0)
+# Volumes where the concentration is reported (in this case 0 to 1600L at every 50L)
+vols=seq(0,1600,length=50)
+# Solution of the set of differential equations using lsode solver in deSolve package
+extout=lsode(times=vols,y=extinit,func=rxnrate,parms=pars)
+```
+**extout** contains the extent of reaction vs volume data. That is used to compute mole fraction and conversion at different volumes along the reactor.
+
+```{r, tidy=FALSE}
+# Calcuation of mole fraction and conversion from extent of reaction at different volumes
+extoutdf=data.frame(extout)
+NBf=pars["NBf"]
+extoutdf$conv=(extoutdf$ext1*2+extoutdf$ext2)/NBf
+extoutdf$yB=(NBf-2*extoutdf$ext1-extoutdf$ext2)/NBf
+extoutdf$yD=(extoutdf$ext1-extoutdf$ext2)/NBf
+extoutdf$yT=(extoutdf$ext2)/NBf
+extoutdf$yH=(extoutdf$ext1+extoutdf$ext2)/NBf
+```
+Next conversion and mole fraction is plotted as a function of reaction volume
+
+```{r, tidy=FALSE, fig.width=10, fig.height=5}
+# load library ggplot2 for plotting
+library(ggplot2)
+# load library reshape2 for data reshaping
+library(reshape2)
+# plot of conversion vs volume
+ggplot(extoutdf,aes(x=time,y=conv))+geom_line()+
+  scale_x_continuous(breaks=seq(0,1600,by=200))+xlab('Volume (L)')+ylab('Conversion')+theme_bw(20)
+
+# plot of mole fraction vs volume
+tmp=melt(extoutdf[,c("time","yB","yD","yT","yH")],id.vars=c("time"),variable.name="moleFraction")
+ggplot(tmp,aes(x=time,y=value,color=moleFraction))+geom_line()+
+    scale_x_continuous(breaks=seq(0,1600,by=200))+xlab('Volume (L)')+ylab('moleFraction')+theme_bw(20)
+```
+Info on the session
+```{r, tidy=FALSE}
+print(sessionInfo(),locale=FALSE)
+```   

04_learnR_parfitODE/ABC_data.dat

+   0.000   0.957  -0.031  -0.015
+   0.263   0.557   0.330   0.044
+   0.526   0.342   0.512   0.156
+   0.789   0.224   0.499   0.310
+   1.053   0.123   0.428   0.454
+   1.316   0.079   0.396   0.556
+   1.579   0.035   0.303   0.651
+   1.842   0.029   0.287   0.658
+   2.105   0.025   0.221   0.750
+   2.368   0.017   0.148   0.854
+   2.632  -0.002   0.182   0.845
+   2.895   0.009   0.116   0.893
+   3.158  -0.023   0.079   0.942
+   3.421   0.006   0.078   0.899
+   3.684   0.016   0.059   0.942
+   3.947   0.014   0.036   0.991
+   4.211  -0.009   0.014   0.988
+   4.474  -0.030   0.036   0.941
+   4.737   0.004   0.036   0.971
+   5.000  -0.024   0.028   0.985
Add a comment to this file

04_learnR_parfitODE/ConfRegion_95pct.jpeg

Added
New image
Add a comment to this file

04_learnR_parfitODE/ExpvsPred.jpeg

Added
New image

04_learnR_parfitODE/compAppendix_parest.R

+#
+# Computational Appendix of Book
+# Chemical Reactor Analysis and Design Fundamentals - Rawlings and Ekerdt
+#
+# Example A.5: Estimating rate constants for A->B->C from concentration vs time data
+#
+#
+
+# set working directory
+setwd("~/R/wkspace")
+
+# load libraries
+library(ggplot2) #library for plotting
+library(reshape2) # library for reshaping data (tall-narrow <-> short-wide)
+library(deSolve) # library for solving differential equations
+library(minpack.lm) # library for least squares fit using levenberg-marquart algorithm
+
+#load concentration data
+df=read.table("ABC_data.dat")
+names(df)=c("time","ca","cb","cc")
+
+# plot data
+tmp=melt(df,id.vars=c("time"),variable.name="species",value.name="conc")
+ggplot(data=tmp,aes(x=time,y=conc,color=species))+geom_point(size=3)
+
+# prediction of concentration
+# rate function
+rxnrate=function(t,c,parms){
+  
+  # rate constant passed through a list called parms
+  k1=parms$k1
+  k2=parms$k2
+  
+  # c is the concentration of species
+  
+  # derivatives dc/dt are computed below
+  r=rep(0,length(c))
+  r[1]=-k1*c["A"]  #dcA/dt
+  r[2]=k1*c["A"]-k2*c["B"] #dcB/dt
+  r[3]=k2*c["B"] #dcC/dt
+  
+  # the computed derivatives are returned as a list
+  # order of derivatives needs to be the same as the order of species in c
+  return(list(r))
+  
+}
+
+# predicted concentration for a given parameter set
+cinit=c(A=1,B=0,C=0)
+t=df$time
+parms=list(k1=2,k2=1)
+out=ode(y=cinit,times=t,func=rxnrate,parms=parms)
+head(out)
+
+#plot of predicted concentration
+outdf=data.frame(out)
+tmp=melt(outdf,id.var="time",variable.name="species",value.name="conc")
+ggplot(data=tmp,aes(x=time,y=conc,color=species))+geom_line()
+
+# function that calculates residual sum of squares
+ssq=function(parms){
+ 
+  # inital concentration
+  cinit=c(A=1,B=0,C=0)
+  # time points for which conc is reported
+  # include the points where data is available
+  t=c(seq(0,5,0.1),df$time)
+  t=sort(unique(t))
+  # parameters from the parameter estimation routine
+  k1=parms[1]
+  k2=parms[2]
+  # solve ODE for a given set of parameters
+  out=ode(y=cinit,times=t,func=rxnrate,parms=list(k1=k1,k2=k2))
+  
+  # Filter data that contains time points where data is available
+  outdf=data.frame(out)
+  outdf=outdf[outdf$time %in% df$time,]
+  # Evaluate predicted vs experimental residual
+  preddf=melt(outdf,id.var="time",variable.name="species",value.name="conc")
+  expdf=melt(df,id.var="time",variable.name="species",value.name="conc")
+  ssqres=preddf$conc-expdf$conc
+  
+  # return predicted vs experimental residual
+  return(ssqres)
+  
+}
+
+# parameter fitting using levenberg marquart algorithm
+# initial guess for parameters
+parms=c(k1=0.5,k2=0.5)
+# fitting
+fitval=nls.lm(par=parms,fn=ssq)
+
+# Summary of fit
+summary(fitval)
+# Estimated parameter
+parest=as.list(coef(fitval))
+parest
+# degrees of freedom: # data points - # parameters
+dof=3*nrow(df)-2
+dof
+# mean error
+ms=sqrt(deviance(fitval)/dof)
+ms
+# variance Covariance Matrix
+S=vcov(fitval)
+S
+
+# plot of predicted vs experimental data
+
+# simulated predicted profile at estimated parameter values
+cinit=c(A=1,B=0,C=0)
+t=seq(0,5,0.2)
+parms=as.list(parest)
+out=ode(y=cinit,times=t,func=rxnrate,parms=parms)
+outdf=data.frame(out)
+names(outdf)=c("time","ca_pred","cb_pred","cc_pred")
+
+# Overlay predicted profile with experimental data
+tmppred=melt(outdf,id.var=c("time"),variable.name="species",value.name="conc")
+tmpexp=melt(df,id.var=c("time"),variable.name="species",value.name="conc")
+p=ggplot(data=tmppred,aes(x=time,y=conc,color=species,linetype=species))+geom_line()
+p=p+geom_line(data=tmpexp,aes(x=time,y=conc,color=species,linetype=species))
+p=p+geom_point(data=tmpexp,aes(x=time,y=conc,color=species))
+p=p+scale_linetype_manual(values=c(0,1,0,1,0,1))
+p=p+scale_color_manual(values=rep(c("red","blue","green"),each=2))+theme_bw()
+print(p)
+
+# Get the 95% confidence region
+
+# Inverse of covariance matrix
+Sinv=solve(S)
+
+# draw the confidence region
+# get points for a circle with radius r
+r=sqrt(qf(0.95,2,58)*2)
+theta=seq(0,2*pi,length.out=100)
+z=cbind(r*cos(theta),r*sin(theta))
+# transform points of circle into points of ellipse using 
+# svd of inverse covariance matrix
+Sinv_svd=svd(Sinv)      # inverse of covariance matrix
+xt=t(Sinv_svd$v)%*%diag(1/sqrt(Sinv_svd$d))%*%t(z) # transform from circle to ellispse
+x=t(xt)
+# translate the ellipse so that center is the estimated parameter value
+x=x+matrix(rep(as.numeric(parest),100),nrow=100,byrow=T)
+
+plot(x[,1],x[,2],type="l",xlab="k1",ylab="k2",lwd=2)
+points(parest$k1,parest$k2,pch=20,col="blue",cex=2)
+
+
+# Simulation based estimation of uncertainty
+
+# store original experimental data in a separate dataframe
+dforig=df
+
+# conc profile based on estimated k1 and k2
+cinit=c(A=1,B=0,C=0)
+t=dforig$time
+parms=parest
+out=ode(y=cinit,times=t,func=rxnrate,parms=parms)
+
+outsim=matrix(0,nrow=nrow(dforig),ncol=4)
+outsim[,1]=out[,1]
+
+# number of simulations
+nsim=1000
+
+parsim=matrix(0,nrow=nsim,ncol=2)
+colnames(parsim)=c("k1","k2")
+
+for (i in 1:nsim){
+  
+  # Simulate data set by adding normal random variable with mean 0 and stdev from fit
+  
+  outsim[,2:4]=out[,2:4]+matrix(rnorm(3*nrow(dforig)),nrow=nrow(dforig),ncol=3)*ms
+  df=data.frame(outsim)
+  names(df)=c("time","ca","cb","cc")
+  
+  # get parameter estimate for the simulated dataset
+  parms=as.numeric(parest)
+  fitsim=nls.lm(par=parms,fn=ssq)
+  # store estimated parameters in the ith row
+  parsim[i,]=coef(fitsim)
+  
+  
+}
+
+# plot the parameter estimates from the 1000 simulations
+plot(parsim[,1],parsim[,2],xlab="k1",ylab="k2")
+# overlay the 95% ellipse computed previously
+lines(x[,1],x[,2],col="blue",lwd=2)
+
+# percentage of parameters from simulation within the 95% ellipse
+tmp=rep(0,length.out=nsim)
+for(i in 1:nsim){
+  tmp[i]=(parsim[i,]-as.numeric(parest))%*%Sinv%*%(parsim[i,]-as.numeric(parest))
+}
+sum(tmp <= qf(0.95,2,58)*2)/nsim
+
+# session Info
+sessionInfo()

04_learnR_parfitODE/compAppendix_parest.html

+<!DOCTYPE html>
+<!-- saved from url=(0014)about:internet -->
+<html>
+<head>
+<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
+
+<title>compAppendix_parest.R</title>
+
+<style type="text/css">
+body, td {
+   font-family: sans-serif;
+   background-color: white;
+   font-size: 12px;
+   margin: 8px;
+}
+
+tt, code, pre {
+   font-family: 'DejaVu Sans Mono', 'Droid Sans Mono', 'Lucida Console', Consolas, Monaco, monospace;
+}
+
+h1 { 
+   font-size:2.2em; 
+}
+
+h2 { 
+   font-size:1.8em; 
+}
+
+h3 { 
+   font-size:1.4em; 
+}
+
+h4 { 
+   font-size:1.0em; 
+}
+
+h5 { 
+   font-size:0.9em; 
+}
+
+h6 { 
+   font-size:0.8em; 
+}
+
+a:visited {
+   color: rgb(50%, 0%, 50%);
+}
+
+pre {	
+   margin-top: 0;
+   max-width: 95%;
+   border: 1px solid #ccc;
+   white-space: pre-wrap;
+}
+
+pre code {
+   display: block; padding: 0.5em;
+}
+
+code.r, code.cpp {
+   background-color: #F8F8F8;
+}
+
+table, td, th {
+  border: none;
+}
+
+blockquote {
+   color:#666666;
+   margin:0;
+   padding-left: 1em;
+   border-left: 0.5em #EEE solid;
+}
+
+hr {
+   height: 0px;
+   border-bottom: none;
+   border-top-width: thin;
+   border-top-style: dotted;
+   border-top-color: #999999;
+}
+
+@media print {
+   * { 
+      background: transparent !important; 
+      color: black !important; 
+      filter:none !important; 
+      -ms-filter: none !important; 
+   }
+
+   body { 
+      font-size:12pt; 
+      max-width:100%; 
+   }
+       
+   a, a:visited { 
+      text-decoration: underline; 
+   }
+
+   hr { 
+      visibility: hidden;
+      page-break-before: always;
+   }
+
+   pre, blockquote { 
+      padding-right: 1em; 
+      page-break-inside: avoid; 
+   }
+
+   tr, img { 
+      page-break-inside: avoid; 
+   }
+
+   img { 
+      max-width: 100% !important; 
+   }
+
+   @page :left { 
+      margin: 15mm 20mm 15mm 10mm; 
+   }
+     
+   @page :right { 
+      margin: 15mm 10mm 15mm 20mm; 
+   }
+
+   p, h2, h3 { 
+      orphans: 3; widows: 3; 
+   }
+
+   h2, h3 { 
+      page-break-after: avoid; 
+   }
+}
+
+</style>
+
+<!-- Styles for R syntax highlighter -->
+<style type="text/css">
+   pre .operator,
+   pre .paren {
+     color: rgb(104, 118, 135)
+   }
+
+   pre .literal {
+     color: rgb(88, 72, 246)
+   }
+
+   pre .number {
+     color: rgb(0, 0, 205);
+   }
+
+   pre .comment {
+     color: rgb(76, 136, 107);
+   }
+
+   pre .keyword {
+     color: rgb(0, 0, 255);
+   }
+
+   pre .identifier {
+     color: rgb(0, 0, 0);
+   }
+
+   pre .string {
+     color: rgb(3, 106, 7);
+   }
+</style>
+
+<!-- R syntax highlighter -->
+<script type="text/javascript">
+var hljs=new function(){function m(p){return p.replace(/&/gm,"&amp;").replace(/</gm,"&lt;")}function f(r,q,p){return RegExp(q,"m"+(r.cI?"i":"")+(p?"g":""))}function b(r){for(var p=0;p<r.childNodes.length;p++){var q=r.childNodes[p];if(q.nodeName=="CODE"){return q}if(!(q.nodeType==3&&q.nodeValue.match(/\s+/))){break}}}function h(t,s){var p="";for(var r=0;r<t.childNodes.length;r++){if(t.childNodes[r].nodeType==3){var q=t.childNodes[r].nodeValue;if(s){q=q.replace(/\n/g,"")}p+=q}else{if(t.childNodes[r].nodeName=="BR"){p+="\n"}else{p+=h(t.childNodes[r])}}}if(/MSIE [678]/.test(navigator.userAgent)){p=p.replace(/\r/g,"\n")}return p}function a(s){var r=s.className.split(/\s+/);r=r.concat(s.parentNode.className.split(/\s+/));for(var q=0;q<r.length;q++){var p=r[q].replace(/^language-/,"");if(e[p]){return p}}}function c(q){var p=[];(function(s,t){for(var r=0;r<s.childNodes.length;r++){if(s.childNodes[r].nodeType==3){t+=s.childNodes[r].nodeValue.length}else{if(s.childNodes[r].nodeName=="BR"){t+=1}else{if(s.childNodes[r].nodeType==1){p.push({event:"start",offset:t,node:s.childNodes[r]});t=arguments.callee(s.childNodes[r],t);p.push({event:"stop",offset:t,node:s.childNodes[r]})}}}}return t})(q,0);return p}function k(y,w,x){var q=0;var z="";var s=[];function u(){if(y.length&&w.length){if(y[0].offset!=w[0].offset){return(y[0].offset<w[0].offset)?y:w}else{return w[0].event=="start"?y:w}}else{return y.length?y:w}}function t(D){var A="<"+D.nodeName.toLowerCase();for(var B=0;B<D.attributes.length;B++){var C=D.attributes[B];A+=" "+C.nodeName.toLowerCase();if(C.value!==undefined&&C.value!==false&&C.value!==null){A+='="'+m(C.value)+'"'}}return A+">"}while(y.length||w.length){var v=u().splice(0,1)[0];z+=m(x.substr(q,v.offset-q));q=v.offset;if(v.event=="start"){z+=t(v.node);s.push(v.node)}else{if(v.event=="stop"){var p,r=s.length;do{r--;p=s[r];z+=("</"+p.nodeName.toLowerCase()+">")}while(p!=v.node);s.splice(r,1);while(r<s.length){z+=t(s[r]);r++}}}}return z+m(x.substr(q))}function j(){function q(x,y,v){if(x.compiled){return}var u;var s=[];if(x.k){x.lR=f(y,x.l||hljs.IR,true);for(var w in x.k){if(!x.k.hasOwnProperty(w)){continue}if(x.k[w] instanceof Object){u=x.k[w]}else{u=x.k;w="keyword"}for(var r in u){if(!u.hasOwnProperty(r)){continue}x.k[r]=[w,u[r]];s.push(r)}}}if(!v){if(x.bWK){x.b="\\b("+s.join("|")+")\\s"}x.bR=f(y,x.b?x.b:"\\B|\\b");if(!x.e&&!x.eW){x.e="\\B|\\b"}if(x.e){x.eR=f(y,x.e)}}if(x.i){x.iR=f(y,x.i)}if(x.r===undefined){x.r=1}if(!x.c){x.c=[]}x.compiled=true;for(var t=0;t<x.c.length;t++){if(x.c[t]=="self"){x.c[t]=x}q(x.c[t],y,false)}if(x.starts){q(x.starts,y,false)}}for(var p in e){if(!e.hasOwnProperty(p)){continue}q(e[p].dM,e[p],true)}}function d(B,C){if(!j.called){j();j.called=true}function q(r,M){for(var L=0;L<M.c.length;L++){if((M.c[L].bR.exec(r)||[null])[0]==r){return M.c[L]}}}function v(L,r){if(D[L].e&&D[L].eR.test(r)){return 1}if(D[L].eW){var M=v(L-1,r);return M?M+1:0}return 0}function w(r,L){return L.i&&L.iR.test(r)}function K(N,O){var M=[];for(var L=0;L<N.c.length;L++){M.push(N.c[L].b)}var r=D.length-1;do{if(D[r].e){M.push(D[r].e)}r--}while(D[r+1].eW);if(N.i){M.push(N.i)}return f(O,M.join("|"),true)}function p(M,L){var N=D[D.length-1];if(!N.t){N.t=K(N,E)}N.t.lastIndex=L;var r=N.t.exec(M);return r?[M.substr(L,r.index-L),r[0],false]:[M.substr(L),"",true]}function z(N,r){var L=E.cI?r[0].toLowerCase():r[0];var M=N.k[L];if(M&&M instanceof Array){return M}return false}function F(L,P){L=m(L);if(!P.k){return L}var r="";var O=0;P.lR.lastIndex=0;var M=P.lR.exec(L);while(M){r+=L.substr(O,M.index-O);var N=z(P,M);if(N){x+=N[1];r+='<span class="'+N[0]+'">'+M[0]+"</span>"}else{r+=M[0]}O=P.lR.lastIndex;M=P.lR.exec(L)}return r+L.substr(O,L.length-O)}function J(L,M){if(M.sL&&e[M.sL]){var r=d(M.sL,L);x+=r.keyword_count;return r.value}else{return F(L,M)}}function I(M,r){var L=M.cN?'<span class="'+M.cN+'">':"";if(M.rB){y+=L;M.buffer=""}else{if(M.eB){y+=m(r)+L;M.buffer=""}else{y+=L;M.buffer=r}}D.push(M);A+=M.r}function G(N,M,Q){var R=D[D.length-1];if(Q){y+=J(R.buffer+N,R);return false}var P=q(M,R);if(P){y+=J(R.buffer+N,R);I(P,M);return P.rB}var L=v(D.length-1,M);if(L){var O=R.cN?"</span>":"";if(R.rE){y+=J(R.buffer+N,R)+O}else{if(R.eE){y+=J(R.buffer+N,R)+O+m(M)}else{y+=J(R.buffer+N+M,R)+O}}while(L>1){O=D[D.length-2].cN?"</span>":"";y+=O;L--;D.length--}var r=D[D.length-1];D.length--;D[D.length-1].buffer="";if(r.starts){I(r.starts,"")}return R.rE}if(w(M,R)){throw"Illegal"}}var E=e[B];var D=[E.dM];var A=0;var x=0;var y="";try{var s,u=0;E.dM.buffer="";do{s=p(C,u);var t=G(s[0],s[1],s[2]);u+=s[0].length;if(!t){u+=s[1].length}}while(!s[2]);if(D.length>1){throw"Illegal"}return{r:A,keyword_count:x,value:y}}catch(H){if(H=="Illegal"){return{r:0,keyword_count:0,value:m(C)}}else{throw H}}}function g(t){var p={keyword_count:0,r:0,value:m(t)};var r=p;for(var q in e){if(!e.hasOwnProperty(q)){continue}var s=d(q,t);s.language=q;if(s.keyword_count+s.r>r.keyword_count+r.r){r=s}if(s.keyword_count+s.r>p.keyword_count+p.r){r=p;p=s}}if(r.language){p.second_best=r}return p}function i(r,q,p){if(q){r=r.replace(/^((<[^>]+>|\t)+)/gm,function(t,w,v,u){return w.replace(/\t/g,q)})}if(p){r=r.replace(/\n/g,"<br>")}return r}function n(t,w,r){var x=h(t,r);var v=a(t);var y,s;if(v){y=d(v,x)}else{return}var q=c(t);if(q.length){s=document.createElement("pre");s.innerHTML=y.value;y.value=k(q,c(s),x)}y.value=i(y.value,w,r);var u=t.className;if(!u.match("(\\s|^)(language-)?"+v+"(\\s|$)")){u=u?(u+" "+v):v}if(/MSIE [678]/.test(navigator.userAgent)&&t.tagName=="CODE"&&t.parentNode.tagName=="PRE"){s=t.parentNode;var p=document.createElement("div");p.innerHTML="<pre><code>"+y.value+"</code></pre>";t=p.firstChild.firstChild;p.firstChild.cN=s.cN;s.parentNode.replaceChild(p.firstChild,s)}else{t.innerHTML=y.value}t.className=u;t.result={language:v,kw:y.keyword_count,re:y.r};if(y.second_best){t.second_best={language:y.second_best.language,kw:y.second_best.keyword_count,re:y.second_best.r}}}function o(){if(o.called){return}o.called=true;var r=document.getElementsByTagName("pre");for(var p=0;p<r.length;p++){var q=b(r[p]);if(q){n(q,hljs.tabReplace)}}}function l(){if(window.addEventListener){window.addEventListener("DOMContentLoaded",o,false);window.addEventListener("load",o,false)}else{if(window.attachEvent){window.attachEvent("onload",o)}else{window.onload=o}}}var e={};this.LANGUAGES=e;this.highlight=d;this.highlightAuto=g;this.fixMarkup=i;this.highlightBlock=n;this.initHighlighting=o;this.initHighlightingOnLoad=l;this.IR="[a-zA-Z][a-zA-Z0-9_]*";this.UIR="[a-zA-Z_][a-zA-Z0-9_]*";this.NR="\\b\\d+(\\.\\d+)?";this.CNR="\\b(0[xX][a-fA-F0-9]+|(\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)";this.BNR="\\b(0b[01]+)";this.RSR="!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|\\.|-|-=|/|/=|:|;|<|<<|<<=|<=|=|==|===|>|>=|>>|>>=|>>>|>>>=|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~";this.ER="(?![\\s\\S])";this.BE={b:"\\\\.",r:0};this.ASM={cN:"string",b:"'",e:"'",i:"\\n",c:[this.BE],r:0};this.QSM={cN:"string",b:'"',e:'"',i:"\\n",c:[this.BE],r:0};this.CLCM={cN:"comment",b:"//",e:"$"};this.CBLCLM={cN:"comment",b:"/\\*",e:"\\*/"};this.HCM={cN:"comment",b:"#",e:"$"};this.NM={cN:"number",b:this.NR,r:0};this.CNM={cN:"number",b:this.CNR,r:0};this.BNM={cN:"number",b:this.BNR,r:0};this.inherit=function(r,s){var p={};for(var q in r){p[q]=r[q]}if(s){for(var q in s){p[q]=s[q]}}return p}}();hljs.LANGUAGES.cpp=function(){var a={keyword:{"false":1,"int":1,"float":1,"while":1,"private":1,"char":1,"catch":1,"export":1,virtual:1,operator:2,sizeof:2,dynamic_cast:2,typedef:2,const_cast:2,"const":1,struct:1,"for":1,static_cast:2,union:1,namespace:1,unsigned:1,"long":1,"throw":1,"volatile":2,"static":1,"protected":1,bool:1,template:1,mutable:1,"if":1,"public":1,friend:2,"do":1,"return":1,"goto":1,auto:1,"void":2,"enum":1,"else":1,"break":1,"new":1,extern:1,using:1,"true":1,"class":1,asm:1,"case":1,typeid:1,"short":1,reinterpret_cast:2,"default":1,"double":1,register:1,explicit:1,signed:1,typename:1,"try":1,"this":1,"switch":1,"continue":1,wchar_t:1,inline:1,"delete":1,alignof:1,char16_t:1,char32_t:1,constexpr:1,decltype:1,noexcept:1,nullptr:1,static_assert:1,thread_local:1,restrict:1,_Bool:1,complex:1},built_in:{std:1,string:1,cin:1,cout:1,cerr:1,clog:1,stringstream:1,istringstream:1,ostringstream:1,auto_ptr:1,deque:1,list:1,queue:1,stack:1,vector:1,map:1,set:1,bitset:1,multiset:1,multimap:1,unordered_set:1,unordered_map:1,unordered_multiset:1,unordered_multimap:1,array:1,shared_ptr:1}};return{dM:{k:a,i:"</",c:[hljs.CLCM,hljs.CBLCLM,hljs.QSM,{cN:"string",b:"'\\\\?.",e:"'",i:"."},{cN:"number",b:"\\b(\\d+(\\.\\d*)?|\\.\\d+)(u|U|l|L|ul|UL|f|F)"},hljs.CNM,{cN:"preprocessor",b:"#",e:"$"},{cN:"stl_container",b:"\\b(deque|list|queue|stack|vector|map|set|bitset|multiset|multimap|unordered_map|unordered_set|unordered_multiset|unordered_multimap|array)\\s*<",e:">",k:a,r:10,c:["self"]}]}}}();hljs.LANGUAGES.r={dM:{c:[hljs.HCM,{cN:"number",b:"\\b0[xX][0-9a-fA-F]+[Li]?\\b",e:hljs.IMMEDIATE_RE,r:0},{cN:"number",b:"\\b\\d+(?:[eE][+\\-]?\\d*)?L\\b",e:hljs.IMMEDIATE_RE,r:0},{cN:"number",b:"\\b\\d+\\.(?!\\d)(?:i\\b)?",e:hljs.IMMEDIATE_RE,r:1},{cN:"number",b:"\\b\\d+(?:\\.\\d*)?(?:[eE][+\\-]?\\d*)?i?\\b",e:hljs.IMMEDIATE_RE,r:0},{cN:"number",b:"\\.\\d+(?:[eE][+\\-]?\\d*)?i?\\b",e:hljs.IMMEDIATE_RE,r:1},{cN:"keyword",b:"(?:tryCatch|library|setGeneric|setGroupGeneric)\\b",e:hljs.IMMEDIATE_RE,r:10},{cN:"keyword",b:"\\.\\.\\.",e:hljs.IMMEDIATE_RE,r:10},{cN:"keyword",b:"\\.\\.\\d+(?![\\w.])",e:hljs.IMMEDIATE_RE,r:10},{cN:"keyword",b:"\\b(?:function)",e:hljs.IMMEDIATE_RE,r:2},{cN:"keyword",b:"(?:if|in|break|next|repeat|else|for|return|switch|while|try|stop|warning|require|attach|detach|source|setMethod|setClass)\\b",e:hljs.IMMEDIATE_RE,r:1},{cN:"literal",b:"(?:NA|NA_integer_|NA_real_|NA_character_|NA_complex_)\\b",e:hljs.IMMEDIATE_RE,r:10},{cN:"literal",b:"(?:NULL|TRUE|FALSE|T|F|Inf|NaN)\\b",e:hljs.IMMEDIATE_RE,r:1},{cN:"identifier",b:"[a-zA-Z.][a-zA-Z0-9._]*\\b",e:hljs.IMMEDIATE_RE,r:0},{cN:"operator",b:"<\\-(?!\\s*\\d)",e:hljs.IMMEDIATE_RE,r:2},{cN:"operator",b:"\\->|<\\-",e:hljs.IMMEDIATE_RE,r:1},{cN:"operator",b:"%%|~",e:hljs.IMMEDIATE_RE},{cN:"operator",b:">=|<=|==|!=|\\|\\||&&|=|\\+|\\-|\\*|/|\\^|>|<|!|&|\\||\\$|:",e:hljs.IMMEDIATE_RE,r:0},{cN:"operator",b:"%",e:"%",i:"\\n",r:1},{cN:"identifier",b:"`",e:"`",r:0},{cN:"string",b:'"',e:'"',c:[hljs.BE],r:0},{cN:"string",b:"'",e:"'",c:[hljs.BE],r:0},{cN:"paren",b:"[[({\\])}]",e:hljs.IMMEDIATE_RE,r:0}]}};
+hljs.initHighlightingOnLoad();
+</script>
+
+
+
+
+</head>
+
+<body>
+<!-- Automatically generated by RStudio [12861c30b10411e1afa60800200c9a66] -->
+
+<h3>compAppendix_parest.R</h3>
+
+<p>admin &mdash; <em>Jun 30, 2013, 10:43 AM</em></p>
+
+<pre><code class="r">#
+# Computational Appendix of Book
+# Chemical Reactor Analysis and Design Fundamentals - Rawlings and Ekerdt
+#
+# Example A.5: Estimating rate constants for A-&gt;B-&gt;C from concentration vs time data
+#
+#
+
+# set working directory
+setwd(&quot;~/R/wkspace&quot;)
+
+# load libraries
+library(ggplot2) #library for plotting
+library(reshape2) # library for reshaping data (tall-narrow &lt;-&gt; short-wide)
+library(deSolve) # library for solving differential equations
+library(minpack.lm) # library for least squares fit using levenberg-marquart algorithm
+
+#load concentration data
+df=read.table(&quot;ABC_data.dat&quot;)
+names(df)=c(&quot;time&quot;,&quot;ca&quot;,&quot;cb&quot;,&quot;cc&quot;)
+
+# plot data
+tmp=melt(df,id.vars=c(&quot;time&quot;),variable.name=&quot;species&quot;,value.name=&quot;conc&quot;)
+ggplot(data=tmp,aes(x=time,y=conc,color=species))+geom_point(size=3)
+</code></pre>
+
+<p><img src="" alt="plot of chunk unnamed-chunk-1"/> </p>
+
+<pre><code class="r">
+# prediction of concentration
+# rate function
+rxnrate=function(t,c,parms){
+
+  # rate constant passed through a list called parms
+  k1=parms$k1
+  k2=parms$k2
+
+  # c is the concentration of species
+
+  # derivatives dc/dt are computed below
+  r=rep(0,length(c))
+  r[1]=-k1*c[&quot;A&quot;]  #dcA/dt
+  r[2]=k1*c[&quot;A&quot;]-k2*c[&quot;B&quot;] #dcB/dt
+  r[3]=k2*c[&quot;B&quot;] #dcC/dt
+
+  # the computed derivatives are returned as a list
+  # order of derivatives needs to be the same as the order of species in c
+  return(list(r))
+
+}
+
+# predicted concentration for a given parameter set
+cinit=c(A=1,B=0,C=0)
+t=df$time
+parms=list(k1=2,k2=1)
+out=ode(y=cinit,times=t,func=rxnrate,parms=parms)
+head(out)
+</code></pre>
+
+<pre><code>      time       A      B       C
+[1,] 0.000 1.00000 0.0000 0.00000
+[2,] 0.263 0.59096 0.3556 0.05348
+[3,] 0.526 0.34924 0.4834 0.16731
+[4,] 0.789 0.20639 0.4958 0.29779
+[5,] 1.053 0.12172 0.4543 0.42395
+[6,] 1.316 0.07193 0.3925 0.53552
+</code></pre>
+
+<pre><code class="r">
+#plot of predicted concentration
+outdf=data.frame(out)
+tmp=melt(outdf,id.var=&quot;time&quot;,variable.name=&quot;species&quot;,value.name=&quot;conc&quot;)
+ggplot(data=tmp,aes(x=time,y=conc,color=species))+geom_line()
+</code></pre>
+
+<p><img src="" alt="plot of chunk unnamed-chunk-1"/> </p>
+
+<pre><code class="r">
+# function that calculates residual sum of squares
+ssq=function(parms){
+
+  # inital concentration
+  cinit=c(A=1,B=0,C=0)
+  # time points for which conc is reported
+  # include the points where data is available
+  t=c(seq(0,5,0.1),df$time)
+  t=sort(unique(t))
+  # parameters from the parameter estimation routine
+  k1=parms[1]
+  k2=parms[2]
+  # solve ODE for a given set of parameters
+  out=ode(y=cinit,times=t,func=rxnrate,parms=list(k1=k1,k2=k2))
+
+  # Filter data that contains time points where data is available
+  outdf=data.frame(out)
+  outdf=outdf[outdf$time %in% df$time,]
+  # Evaluate predicted vs experimental residual
+  preddf=melt(outdf,id.var=&quot;time&quot;,variable.name=&quot;species&quot;,value.name=&quot;conc&quot;)
+  expdf=melt(df,id.var=&quot;time&quot;,variable.name=&quot;species&quot;,value.name=&quot;conc&quot;)
+  ssqres=preddf$conc-expdf$conc
+
+  # return predicted vs experimental residual
+  return(ssqres)
+
+}
+
+# parameter fitting using levenberg marquart algorithm
+# initial guess for parameters
+parms=c(k1=0.5,k2=0.5)
+# fitting
+fitval=nls.lm(par=parms,fn=ssq)
+
+# Summary of fit
+summary(fitval)
+</code></pre>
+
+<pre><code>
+Parameters:
+   Estimate Std. Error t value Pr(&gt;|t|)    
+k1   2.0191     0.0487    41.5   &lt;2e-16 ***
+k2   0.9930     0.0178    55.8   &lt;2e-16 ***
+---
+Signif. codes:  0 &#39;***&#39; 0.001 &#39;**&#39; 0.01 &#39;*&#39; 0.05 &#39;.&#39; 0.1 &#39; &#39; 1
+
+Residual standard error: 0.0212 on 58 degrees of freedom
+Number of iterations to termination: 7 
+Reason for termination: Relative error in the sum of squares is at most `ftol&#39;. 
+</code></pre>
+
+<pre><code class="r"># Estimated parameter
+parest=as.list(coef(fitval))
+parest
+</code></pre>
+
+<pre><code>$k1
+[1] 2.019
+
+$k2
+[1] 0.993
+</code></pre>
+
+<pre><code class="r"># degrees of freedom: # data points - # parameters
+dof=3*nrow(df)-2
+dof
+</code></pre>
+
+<pre><code>[1] 58
+</code></pre>
+
+<pre><code class="r"># mean error
+ms=sqrt(deviance(fitval)/dof)
+ms
+</code></pre>
+
+<pre><code>[1] 0.0212
+</code></pre>
+
+<pre><code class="r"># variance Covariance Matrix
+S=vcov(fitval)
+S
+</code></pre>
+
+<pre><code>           k1         k2
+k1  0.0023685 -0.0003606
+k2 -0.0003606  0.0003165
+</code></pre>
+
+<pre><code class="r">
+# plot of predicted vs experimental data
+
+# simulated predicted profile at estimated parameter values
+cinit=c(A=1,B=0,C=0)
+t=seq(0,5,0.2)
+parms=as.list(parest)
+out=ode(y=cinit,times=t,func=rxnrate,parms=parms)
+outdf=data.frame(out)
+names(outdf)=c(&quot;time&quot;,&quot;ca_pred&quot;,&quot;cb_pred&quot;,&quot;cc_pred&quot;)
+
+# Overlay predicted profile with experimental data
+tmppred=melt(outdf,id.var=c(&quot;time&quot;),variable.name=&quot;species&quot;,value.name=&quot;conc&quot;)
+tmpexp=melt(df,id.var=c(&quot;time&quot;),variable.name=&quot;species&quot;,value.name=&quot;conc&quot;)
+p=ggplot(data=tmppred,aes(x=time,y=conc,color=species,linetype=species))+geom_line()
+p=p+geom_line(data=tmpexp,aes(x=time,y=conc,color=species,linetype=species))
+p=p+geom_point(data=tmpexp,aes(x=time,y=conc,color=species))
+p=p+scale_linetype_manual(values=c(0,1,0,1,0,1))
+p=p+scale_color_manual(values=rep(c(&quot;red&quot;,&quot;blue&quot;,&quot;green&quot;),each=2))+theme_bw()
+print(p)
+</code></pre>
+
+<p><img src="" alt="plot of chunk unnamed-chunk-1"/> </p>
+
+<pre><code class="r">
+# Get the 95% confidence region
+
+# Inverse of covariance matrix
+Sinv=solve(S)
+
+# draw the confidence region
+# get points for a circle with radius r
+r=sqrt(qf(0.95,2,58)*2)
+theta=seq(0,2*pi,length.out=100)
+z=cbind(r*cos(theta),r*sin(theta))
+# transform points of circle into points of ellipse using 
+# svd of inverse covariance matrix
+Sinv_svd=svd(Sinv)      # inverse of covariance matrix
+xt=t(Sinv_svd$v)%*%diag(1/sqrt(Sinv_svd$d))%*%t(z) # transform from circle to ellispse
+x=t(xt)
+# translate the ellipse so that center is the estimated parameter value
+x=x+matrix(rep(as.numeric(parest),100),nrow=100,byrow=T)
+
+plot(x[,1],x[,2],type=&quot;l&quot;,xlab=&quot;k1&quot;,ylab=&quot;k2&quot;,lwd=2)
+points(parest$k1,parest$k2,pch=20,col=&quot;blue&quot;,cex=2)
+</code></pre>
+
+<p><img src="" alt="plot of chunk unnamed-chunk-1"/> </p>
+
+<pre><code class="r">
+
+# Simulation based estimation of uncertainty
+
+# store original experimental data in a separate dataframe
+dforig=df
+
+# conc profile based on estimated k1 and k2
+cinit=c(A=1,B=0,C=0)
+t=dforig$time
+parms=parest
+out=ode(y=cinit,times=t,func=rxnrate,parms=parms)
+
+outsim=matrix(0,nrow=nrow(dforig),ncol=4)
+outsim[,1]=out[,1]
+
+# number of simulations
+nsim=1000
+
+parsim=matrix(0,nrow=nsim,ncol=2)
+colnames(parsim)=c(&quot;k1&quot;,&quot;k2&quot;)
+
+for (i in 1:nsim){
+
+  # Simulate data set by adding normal random variable with mean 0 and stdev from fit
+
+  outsim[,2:4]=out[,2:4]+matrix(rnorm(3*nrow(dforig)),nrow=nrow(dforig),ncol=3)*ms
+  df=data.frame(outsim)
+  names(df)=c(&quot;time&quot;,&quot;ca&quot;,&quot;cb&quot;,&quot;cc&quot;)
+
+  # get parameter estimate for the simulated dataset
+  parms=as.numeric(parest)
+  fitsim=nls.lm(par=parms,fn=ssq)
+  # store estimated parameters in the ith row
+  parsim[i,]=coef(fitsim)
+
+
+}
+
+# plot the parameter estimates from the 1000 simulations
+plot(parsim[,1],parsim[,2],xlab=&quot;k1&quot;,ylab=&quot;k2&quot;)
+# overlay the 95% ellipse computed previously
+lines(x[,1],x[,2],col=&quot;blue&quot;,lwd=2)
+</code></pre>
+
+<p><img src="" alt="plot of chunk unnamed-chunk-1"/> </p>
+
+<pre><code class="r">
+# percentage of parameters from simulation within the 95% ellipse
+tmp=rep(0,length.out=nsim)
+for(i in 1:nsim){
+  tmp[i]=(parsim[i,]-as.numeric(parest))%*%Sinv%*%(parsim[i,]-as.numeric(parest))
+}
+sum(tmp &lt;= qf(0.95,2,58)*2)/nsim
+</code></pre>
+
+<pre><code>[1] 0.957
+</code></pre>
+
+<pre><code class="r">
+# session Info
+sessionInfo()
+</code></pre>
+
+<pre><code>R version 3.0.1 (2013-05-16)
+Platform: i386-w64-mingw32/i386 (32-bit)
+
+locale:
+[1] LC_COLLATE=English_United States.1252 
+[2] LC_CTYPE=English_United States.1252   
+[3] LC_MONETARY=English_United States.1252
+[4] LC_NUMERIC=C                          
+[5] LC_TIME=English_United States.1252    
+
+attached base packages:
+[1] stats     graphics  grDevices utils     datasets  methods   base     
+
+other attached packages:
+[1] minpack.lm_1.1-7 deSolve_1.10-6   reshape2_1.2.2   ggplot2_0.9.3.1 
+[5] knitr_1.2       
+
+loaded via a namespace (and not attached):
+ [1] colorspace_1.2-2   dichromat_2.0-0    digest_0.6.3      
+ [4] evaluate_0.4.3     formatR_0.8        grid_3.0.1        
+ [7] gtable_0.1.2       labeling_0.2       MASS_7.3-26       
+[10] munsell_0.4        plyr_1.8           proto_0.3-10      
+[13] RColorBrewer_1.0-5 scales_0.2.3       stringr_0.6.2     
+[16] tools_3.0.1       
+</code></pre>
+
+</body>
+
+</html>
+
Add a comment to this file

04_learnR_parfitODE/concProfileData.jpeg

Added
New image
Add a comment to this file

04_learnR_parfitODE/confRegion_simul95pct.jpeg

Added
New image

README

-This is a place to put codes and material I am using to learn different aspects of math, computation and visualization. I put some of these in the blog http://notesofdabbler.wordpress.com
+This is a place to put codes and material I am using to learn different aspects of math, computation and visualization. I put some of these in the blog http://notesofdabbler.wordpress.com

R_codes/20120804_CompApp.Rmd

-# Learning R using a Chemical Reaction Engineering Book
-
-[Chemical Reactor Analysis and Design Fundamentals](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/) by J.B. Rawlings and J. G. Ekerdt is a textbook for studying Chemical Reaction Engineering. The popular open source package [Octave](http://www.gnu.org/software/octave/) has its origins to the reaction engineering course offered by Prof. Rawlings. This book is accompanied by Octave and Matlab code for solving typical problems encountered in Reaction Engineering. 
-
-I figured that maybe one way to learn R is so see whether I can code some of th examples from this book in R. I am by no means suggesting that R can replace MATLAB/Octave for engineering problems but merely it is a way for me to learn the language.
-
-I started with the [computational appendix](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/web-appendix.pdf) listed in the book's website and am trying to work through some of the examples there. It will be good to refer to the computational appendix to follow the R code below.
-
-### Setting up a stoichiometric matrix, reaction rate vector and determining the rank
-
-```{r}
-# stoichiometric matrix
-stoi=matrix(c(0,1,0,-1,-1,1,
-       -1,1,1,-1,0,0,
-       1,0,-1,0,-1,1),
-       ncol=6,byrow=T)
-stoi
-
-# rank of the stoichiometrix matrix
-rank=qr(stoi)$rank
-rank
-
-# reaction rate vector
-r=c(1,2,3)
-```
-Given the reaction rate r=(r1,r2)', the rate of change of species concentration R is given by
-$$latex R=\nu^Tr $$
-where $\nu$ is the stoichiometrix matrix
-```{r}
-# rate of change of components
-R=t(stoi)%*%r
-R
-```
-### Example A1: Estimating reaction rates
-
-The stoichometrix matrix $\nu$ is input below
-```{r}
-# stoichiometry
-stoi=matrix(c(0,1,0,-1,-1,1,
-              -1,1,1,-1,0,0),nrow=2,byrow=T)
-stoi
-# number of species and number of reactions
-nspec=ncol(stoi)
-nr=nrow(stoi)
-
-nspec
-nr
-
-# true rxn rates
-r=c(1,2)
-r
-
-# true component rates
-R=t(stoi)%*%r
-R
-```
-
-### simulate 2000 measured component rates
-
-Add random noise (normally distributed with mean 0 and standard deviation 0.05) to true species rate vector R
-$$ R^m = R + \epsilon, \epsilon \sim N(0,0.0025) $$
-
-```{r}
-## simulate 2000 noise estimates
-e=matrix(0.05*rnorm(2000*nspec,0,1),nrow=2000,byrow=T)
-Rmeas=matrix(rep(R,2000),ncol=nspec,byrow=T)+e
-```
-The least squares estimate of reaction rate vector $\cap{r}$ is
-$$ \hat{r}=(\nu\nu^T)^{-1}{\nu}R $$
-
-```{r}
-## estimate reaction rates
-rest=solve(stoi%*%t(stoi),stoi%*%t(Rmeas))
-```
-I was trying different plot features in R and applying to this data of estimated rates
-
-I found the following function that plots scatterplot with marginal histograms
-```{r}
-# plotting scatterplot with histogram
-# downloaded from web
-#  http://www.r-bloggers.com/example-8-41-scatterplot-with-marginal-histograms/
-#
-scatterhist = function(x, y, xlab="", ylab=""){
-  par.default <- par(no.readonly=TRUE)
-  zones=matrix(c(2,0,1,3), ncol=2, byrow=TRUE)
-  layout(zones, widths=c(4/5,1/5), heights=c(1/5,4/5))
-  xhist = hist(x, plot=FALSE)
-  yhist = hist(y, plot=FALSE)
-  top = max(c(xhist$counts, yhist$counts))
-  par(mar=c(3,3,1,1))
-  plot(x,y)
-  par(mar=c(0,3,1,1))
-  barplot(xhist$counts, axes=FALSE, ylim=c(0, top), space=0)
-  par(mar=c(3,0,1,1))
-  barplot(yhist$counts, axes=FALSE, xlim=c(0, top), space=0, horiz=TRUE)
-  par(oma=c(3,3,0,0))
-  mtext(xlab, side=1, line=1, outer=TRUE, adj=0, 
-        at=0.9 * (mean(x) - min(x))/(max(x)-min(x)))
-  mtext(ylab, side=2, line=1, outer=TRUE, adj=0, 
-        at=(0.9 * (mean(y) - min(y))/(max(y) - min(y))))
-  par=par(par.default)
-}
-
-# scatter plot of reaction rates with marginal histograms
-scatterhist(t(rest)[,1],t(rest)[,2],xlab="r_1",ylab="r_2")
-```
-There is library cars that has a command 'scatterplot' for plotting scatterplot with box plots and has several other options
-In the plot below 50% and 90% ellipses are overlaid on the data
-
-```{r}
-# scatter plot of reaction rates with marginal box plots
-library(car)
-scatterplot(t(rest)[,1],t(rest)[,2],reg.line=FALSE,smooth=FALSE,ellipse=TRUE)
-```
-I tried ggplot2 also for the same plot
-
-```{r}
-# 2d contours/density with ggplot2 for reaction rates
-
-# create dataframe of reaction rates
-rest_df=data.frame(t(rest))
-names(rest_df)=c("r1","r2")
-
-library(ggplot2)
-ggplot(data=rest_df,aes(x=r1,y=r2))+geom_point()
-ggplot(data=rest_df,aes(x=r1,y=r2))+stat_density2d()
-ggplot(data=rest_df,aes(x=r1,y=r2))+stat_density2d(aes(fill=..level..),geom="polygon")
-```
-
-

R_codes/20120811_CompApp_pt_2_3.Rmd

-# Learning R using a Chemical Reaction Engineering Book (Part 2)
-
-In case you missed part 1, you can view it [here](http://rdabbler.blogspot.com/2012/08/learning-r-using-chemical-reaction.html). In this part, I tried to recreate the examples in sections A.2.2 and A.2.3 of the [computational appendix](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/web-appendix.pdf) in the [reaction engineering book](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/) (by Rawlings and Ekerdt). 
-
-## Solving a nonlinear system of equations
-
-This example involves determining reaction equilibrium conditions by solving the following system of nonlinear equations.
-
-$$latex
- \begin{aligned}
-   PK_1y_Iy_B-y_{P1}=0, \\
-   PK_2y_Iy_B-y_{P2}=0
- \end{aligned}   
-$$
-
-The relation between the variables $latex y_I,y_B,y_{P1},y_{P2}$ and extent of reactions $latex x_1,x_2$ are:
-$$latex
-  \begin{aligned}
-    y_I=\frac{y_{I0}-x_1-x_2}{1-x_1-x_2}, \\
-    y_B=\frac{y_{B0}-x_1-x_2}{1-x_1-x_2}, \\ 
-    y_{P1}=\frac{y_{p10}+x_1}{1-x_1-x_2}, \\
-    y_{P2}=\frac{y_{p20}+x_2}{1-x_1-x_2}
-  \end{aligned}
-$$  
-
-Here I have used R package [rootSolve](http://cran.r-project.org/web/packages/rootSolve/index.html) for solving the above set of equations. The library is loaded and the functions to be solved are defined in the R function fns.
-
-```{r}
-# load library rootSolve
-library(rootSolve)
-
-# function defining F(x)=0
-fns=function(x){
-  K1=108; K2=284; P=2.5
-  yI0=0.5; yB0=0.5; yP10=0; yP20=0;
-  d=1-x[1]-x[2]
-  yI=(yI0-x[1]-x[2])/d
-  yB=(yB0-x[1]-x[2])/d
-  yP1=(yP10+x[1])/d
-  yP2=(yP20+x[2])/d
-  F1=P*K1*yI*yB-yP1
-  F2=P*K2*yI*yB-yP2
-  c(F1=F1,F2=F2)
-}
-```
-Next, an initial guess of (0.2,0.2) is set for the variables and the equations are solved using the function multiroot (from package rootSolve)
-
-```{r}
-# initial guess for x
-xinit=c(0.2,0.2)
-
-# solve the equations
-xans=multiroot(f=fns,start=xinit)
-
-# object returned by multiroot
-xans
-
-# solution to the equations
-xans$root
-```
-The solution to the equations is accessed from the variable xans$root which in this case is (0.1334,0.3507)
-
-## Function Minimization
-
-Here the function to be minimized is
-
-
-$$latex
-  G=-(x_1lnK_1+x_2lnK_2)+(1-x_1-x_2)lnP+(y_{I0}-x_1-x_2)ln(y_I)+(y_{B0}-x_1-x_2)ln(y_B)+(y_{P10}+x_1)ln(y_{p1})+(y_{P20}+x_2)ln(y_{P2})
-$$
-
-The following constraints need to be satisfied for $latex x_1$ and $latex x_2$
-
-$$latex
-  0 \leq x_1 \leq 0.5, 0 \leq x_2 \leq 0.5, x_1+x_2 \leq 0.5
-$$latex
-
-First I used [constrOptim](http://stat.ethz.ch/R-manual/R-patched/library/stats/html/constrOptim.html) function for minimization. We need to specify the function to be minimized
-
-```{r}
-# function to be minimized
-eval_f0=function(x){
-  dg1=-3.72e3; dg2=-4.49e3; T=400; R=1.987; P=2.5
-  K1=exp(-dg1/(R*T)); K2=exp(-dg2/(R*T))
-  
-  yI0=0.5; yB0=0.5; yP10=0; yP20=0;
-  d=1-x[1]-x[2]
-  yI=(yI0-x[1]-x[2])/d
-  yB=(yB0-x[1]-x[2])/d
-  yP1=(yP10+x[1])/d
-  yP2=(yP20+x[2])/d
-  
-  f=-(x[1]*log(K1)+x[2]*log(K2))+(1-x[1]-x[2])*log(P)+yI*d*log(yI)+
-       yB*d*log(yB)+yP1*d*log(yP1)+yP2*d*log(yP2)
-  
-  return(f)
-}
-```
-The constraints need to be specified in the form $latex Ax-b \geq 0$
-
-```{r}
-#  constraint
-A=matrix(c(-1,-1,1,0,-1,0,0,1,0,-1),ncol=2,byrow=TRUE)
-A
-b=c(-0.5,0,-0.5,0,-0.5)
-b
-```
-Next, the function is minimized using constrOptim (starting from an initial guess of (0.2,0.2)). Here Nelder-Mead method is used since BFGS method requires specifying the gradient of the function by the user. R taskview [optimization](http://cran.r-project.org/web/views/Optimization.html) lists other options.
-
-```{r}
-# initial guess
-xinit=c(0.2,0.2)
-
-# minimize function subject to bounds and constraints
-xans2=constrOptim(theta=xinit, f=eval_f0, grad=NULL, ui=A, ci=b,
-            method = "Nelder-Mead")
-xans2
-```
-The solution can be accessed from xans2$par and is (0.1331,0.3509)
-
-Next, I also tried function minimization with ConstrOptim.nl from the package [alabama](http://cran.r-project.org/web/packages/alabama/index.html). Here the constraints are specified in terms of $latex h(x) \geq 0$. Even if gradient is not supplied, this function will estimate it using finite-differences.
-
-Definition of constraints in the format for constrOptim.nl
-
-```{r}
-# load library alabama
-library(alabama)
-library(numDeriv)
-
-h_ineq=function(x){
-   h=rep(NA,1)
-   h[1]=-x[1]-x[2]+0.5
-   h[2]=x[1]
-   h[3]=-x[1]+0.5
-   h[4]=x[2]
-   h[5]=-x[2]+0.5
-   return(h)
-}
-
-xans3=constrOptim.nl(par=xinit,fn=eval_f0,hin=h_ineq)
-
-xans3
-```
-The solution can be accessed from xans3$par and is (0.1332,0.3508)
-
-Since this is a 2 dimensional problem, the solution can also be visualized using a contour plot of the function.
-
-```{r message=FALSE,warning=FALSE}
-# Region of interest: 0.01<=x1<=0.49, 0.01<=x2<=0.49
-x1=seq(0.01,0.49,by=0.01)
-x2=seq(0.01,0.49,by=0.01)
-
-# vectorizing function eval_f0 so that it can be evaluted in the outer function
-fcont=function(x,y) eval_f0(c(x,y))
-fcontv=Vectorize(fcont,SIMPLIFY=FALSE)
-z=outer(x1,x2,fcontv)
-
-# filled.coutour and contour are overlaid with the minimum point (0.133,0.351)
-filled.contour(x1,x2,z,xlab="x1",ylab="x2",
-               plot.axes={axis(1); axis(2); contour(x1,x2,z,levels=c(-3,-2.5,-2,-1.5,-1,0),vfont=c("sans serif","bold"),labcex=1,lwd=2,add=TRUE); 
-                          points(0.133,0.351,pch=15,col="blue")})
-```
-
-MATLAB/Octave functions for solving nonlinear equations (fsolve and fmincon) have been used in Chemical Engineering computations for a long time and are robust. R has traditionally not been used in this domain. So it is hard to say how the functions I have used in this blog will perform across the range of problems encountered in Reaction Engineering.
-
-This post was created with R 2.15.1 and knitr 0.7.1 (converted markdown to html)

R_codes/20121209_CompApp_pt4.Rmd

-# Learning R using a Chemical Reaction Engineering Book (Part 3)
-
-In case you missed the previous parts, the links to them are listed below:
-* [Part1](http://rdabbler.blogspot.com/2012/08/learning-r-using-chemical-reaction.html). 
-* [Part2](http://rdabbler.blogspot.com/2012/08/learning-r-using-chemical-reaction_11.html)
-
-In this part, I tried to recreate the examples in sections A.3.1 of the [computational appendix](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/web-appendix.pdf) in the [reaction engineering book](http://jbrwww.che.wisc.edu/home/jbraw/chemreacfun/) (by Rawlings and Ekerdt). 
-
-## Solving a system of ordinary differential equations
-
-This example involves reaction (Benzene pyrolysis) in a plug flow reactor. The actual reactions happening are:
-
-$$ (Rxn1) \; 2B = D + H $$
-$$ (Rxn2) \; B + D = T+H $$
-
-
-The rate of each reaction is given by:
-$$ r_1=k_1(c_B^2-\frac{c_Dc_H}{K_1}) $$
-$$ r_2=k_2(c_Bc_D-\frac{c_Tc_H}{K_2}) $$
-
-The feed to the reactor consists of 60kmol/hr of Benzene (B). The temperature of the reactor is $T=1033K$ and the pressure is $P=1atm$.The rate constants and equilibrium constants for this example are:
-$$ k_1=7\times 10^5\; L/mol.hr,\; k_2=4\times 10^5 \; L/mol.hr,\; K_1=0.31,\; K_2=0.48 $$
-
-```{r, tidy=FALSE}
-# set working directory
-setwd('~/R/RxnEngg_Rawlings')
-
-# load libraries
-library(deSolve)
-
-
-# Appendix A.3.1: Solution of Differential Equations
-
-# Benzene pyrolysis example
-
-# Parameters
-# NBf - feed benzene rate - mol/h
-# R - Universal gas constant
-# T - Reactor temperature K
-# P - Reactor pressure atm
-# k1 - rxn1 forward rate constant L/mol.h
-# k2 - rxn2 forward rate constant L/mol.h
-# Keq1 - rxn1 equilibrium constant
-# Keq2 - rxn2 equilibrium constant
-pars=c(
-NBf=60e3,  
-R=0.08205,  
-T=1033, 
-P=1, 
-k1=7e5, 
-k2=4e5, 
-Keq1=0.31, 
-Keq2=0.48  
-)
-```
-
-The governing equations for conversion versus volume in a plug flow reactor is based on extent of each of the reactions:
-$$ \frac{d\epsilon_1}{dV}=r_1,\; \frac{d\epsilon_2}{dV}=r_2 $$
-The initial conditions (corresponding to feed conditions $N_B(0)=60kmol/h,\;N_D(0)=N_H(0)=N_T(0)=0$) are that the extent of reaction is zero. 
-$$ \epsilon_1(0)=0, \; \epsilon_2(0)=0 $$
-
-The flow rates of each component along the reactor volume can be calculated from reaction extent
-$$ N_B=N_B(0)-2\epsilon_1-\epsilon_2, \; N_D=\epsilon_1-\epsilon_2, \; N_H=\epsilon_1+\epsilon_2, \; N_T=\epsilon_2 $$ 
-
-These are setup in a function that can be passed to an ODE solver. In this case the ODE solver we use lsode from the R package [deSolve](http://desolve.r-forge.r-project.org/). The inputs to the function are:
-* Variable over which the integration is done (Volume in this case)
-* The state variables of the system (Extent of the two reactions)
-* Parameters that are needed for description of the system (Rate constants, Temperature, Pressure, etc.)
-The output from this function is the rate of change as described by the equations previously.
-
-```{r, tidy=FALSE}
-# function that will be passed to odesolver
-# vol is the variable over which the system is integrated (equivalent of time in batch reactions)
-# ext is the extent of reactions 1 and 2
-# params are the parameters passed to the system
-rxnrate=function(vol,ext,params) {
-     with(as.list(c(ext,params)),{
-        NB=NBf-2*ext1-ext2
-        ND=ext1-ext2
-        NH=ext1+ext2
-        NT=ext2
-        Q=NBf*R*T/P
-        cB=NB/Q
-        cD=ND/Q
-        cT=NT/Q
-        cH=NH/Q
-        dext1=k1*(cB*cB-cD*cH/Keq1)
-        dext2=k2*(cB*cD-cT*cH/Keq2)
-        return(list(c(dext1=dext1,dext2=dext2)))
-     })
-}
-```
-
-Since the reaction start only after the feed enters the reactor, the extent of reaction is zero for both reactions at the beginning of the reactor (V=0L). The set of volumes where the concentration and reaction extent is computed is chosen in this case to be from 0L to 1600L at every 50L. The ODE solver lsode from [deSolve](http://desolve.r-forge.r-project.org/) package is used to solve the system of equations.
-
-```{r, tidy=FALSE}
-# initial extent of reaction (zero in this case for both reactions)
-extinit=c(ext1=0,ext2=0)
-# Volumes where the concentration is reported (in this case 0 to 1600L at every 50L)
-vols=seq(0,1600,length=50)
-# Solution of the set of differential equations using lsode solver in deSolve package
-extout=lsode(times=vols,y=extinit,func=rxnrate,parms=pars)
-```
-**extout** contains the extent of reaction vs volume data. That is used to compute mole fraction and conversion at different volumes along the reactor.
-
-```{r, tidy=FALSE}
-# Calcuation of mole fraction and conversion from extent of reaction at different volumes
-extoutdf=data.frame(extout)
-NBf=pars["NBf"]
-extoutdf$conv=(extoutdf$ext1*2+extoutdf$ext2)/NBf
-extoutdf$yB=(NBf-2*extoutdf$ext1-extoutdf$ext2)/NBf
-extoutdf$yD=(extoutdf$ext1-extoutdf$ext2)/NBf
-extoutdf$yT=(extoutdf$ext2)/NBf
-extoutdf$yH=(extoutdf$ext1+extoutdf$ext2)/NBf
-```
-Next conversion and mole fraction is plotted as a function of reaction volume
-
-```{r, tidy=FALSE, fig.width=10, fig.height=5}
-# load library ggplot2 for plotting
-library(ggplot2)
-# load library reshape2 for data reshaping
-library(reshape2)
-# plot of conversion vs volume
-ggplot(extoutdf,aes(x=time,y=conv))+geom_line()+
-  scale_x_continuous(breaks=seq(0,1600,by=200))+xlab('Volume (L)')+ylab('Conversion')+theme_bw(20)
-
-# plot of mole fraction vs volume
-tmp=melt(extoutdf[,c("time","yB","yD","yT","yH")],id.vars=c("time"),variable.name="moleFraction")
-ggplot(tmp,aes(x=time,y=value,color=moleFraction))+geom_line()+
-    scale_x_continuous(breaks=seq(0,1600,by=200))+xlab('Volume (L)')+ylab('moleFraction')+theme_bw(20)
-```
-Info on the session
-```{r, tidy=FALSE}
-print(sessionInfo(),locale=FALSE)
-```   
Tip: Filter by directory path e.g. /media app.js to search for public/media/app.js.
Tip: Use camelCasing e.g. ProjME to search for ProjectModifiedEvent.java.
Tip: Filter by extension type e.g. /repo .js to search for all .js files in the /repo directory.
Tip: Separate your search with spaces e.g. /ssh pom.xml to search for src/ssh/pom.xml.
Tip: Use ↑ and ↓ arrow keys to navigate and return to view the file.
Tip: You can also navigate files with Ctrl+j (next) and Ctrl+k (previous) and view the file with Ctrl+o.
Tip: You can also navigate files with Alt+j (next) and Alt+k (previous) and view the file with Alt+o.