# Steady-state solver for 2-Dimensional ordinary differential equations

### Description

Estimates the steady-state condition for a system of ordinary differential equations that result from 2-Dimensional partial differential equation models that have been converted to ODEs by numerical differencing.

It is assumed that exchange occurs only between adjacent layers.

### Usage

1 2 3 |

### Arguments

`y ` |
the initial guess of (state) values for the ODE system, a vector. |

`time ` |
time for which steady-state is wanted; the default is time=0. |

`func ` |
either an The return value of The derivatives
should be specified in the same order as the state variables |

`parms ` |
parameters passed to |

`nspec ` |
the number of *species* (components) in the model. |

`dimens ` |
a 2-valued vector with the dimensionality of the model, i.e. the number of *boxes* in x- and y-direction. |

`cyclicBnd ` |
if not |

`names ` |
the names of the components; used to label the output, and for plotting. |

`... ` |
additional arguments passed to function stodes. See note. |

### Details

This is the method of choice for 2-dimensional models, that are only subjected to transport between adjacent layers.

Based on the dimension of the problem, the method first calculates the
sparsity pattern of the Jacobian, under the assumption
that transport is only occurring between adjacent layers.
Then `stodes`

is called to find the steady-state.

In some cases, a cyclic boundary condition exists. This is when the first
boxes in x-or y-direction interact with the last boxes. In this case, there
will be extra non-zero fringes in the Jacobian which need to be taken
into account. The occurrence of cyclic boundaries can be
toggled on by specifying argument `cyclicBnd`

. For innstance,
`cyclicBnd = 1`

indicates that a cyclic boundary is required only for
the x-direction, whereas `cyclicBnd = c(1,2)`

imposes a cyclic boundary
for both x- and y-direction. The default is no cyclic boundaries.

As `stodes`

is used, it will probably be necessary to specify the
length of the real work array, `lrw`

.

Although a reasonable guess of `lrw`

is made, it may occur that this
will be too low.
In this case, `steady.2D`

will return with an error message telling
that there was insufficient storage. In the second try then, increase
`lrw`

. you may need to experiment to find suitable value. The smaller the better.

An error message that says to increase `lrw`

may look like this:

1 2 3 |

See `stodes`

for the additional options.

### Value

A list containing

`y ` |
a vector with the state variable values from the last iteration during estimation of steady-state condition of the system of equations. |

`... ` |
the "global" values returned. |

The output will have the `attribute`

`steady`

, which returns `TRUE`

,
if steady-state has been reached and the attribute
`precis`

with the precision attained during each iteration.
Another attribute, called `dims`

returns a.o. the length of the work
array actually required.
This can be specified with input argument `lrw`

. See note and first example

### Note

It is advisable though not mandatory to specify BOTH `nspec`

and
`dimens`

. In this case, the solver can check whether the input makes
sense (as nspec*dimens[1]*dimens[2] = length(y))

do NOT use this method for problems that are not 2D.

It is likely that the estimated length of the work array (argument `lrw`

),
required for the solver stodes will be too small.
If that is the case, the solver will return with an error
saying to increase `lrw`

. The current value of the work array can be
found via the `attributes`

of the output. See first example.

### Author(s)

Karline Soetaert <karline.soetaert@nioz.nl>

### See Also

`steady`

, for a general interface to most of the steady-state
solvers

`steady.band`

, to find the steady-state of ODE models with a
banded Jacobian

`steady.1D`

,
`steady.3D`

, steady-state solvers for 1-Dand 3-D
partial differential equations.

`stode`

, iterative steady-state solver for ODEs with full
or banded Jacobian.

`stodes`

, iterative steady-state solver for ODEs with arbitrary
sparse Jacobian.

`runsteady`

, steady-state solver by dynamically running to
steady-state

### Examples

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 | ```
## =======================================================================
## Diffusion in 2-D; imposed boundary conditions
## =======================================================================
diffusion2D <- function(t, Y, par) {
y <- matrix(nr=n,nc=n,data=Y) # vector to 2-D matrix
dY <- -r*y # consumption
BND <- rep(1,n) # boundary concentration
#diffusion in X-direction; boundaries=imposed concentration
Flux <- -Dx * rbind(y[1,]-BND,(y[2:n,]-y[1:(n-1),]),BND-y[n,])/dx
dY <- dY - (Flux[2:(n+1),]-Flux[1:n,])/dx
#diffusion in Y-direction
Flux <- -Dy * cbind(y[,1]-BND,(y[,2:n]-y[,1:(n-1)]),BND-y[,n])/dy
dY <- dY - (Flux[,2:(n+1)]-Flux[,1:n])/dy
return(list(as.vector(dY)))
}
# parameters
dy <- dx <- 1 # grid size
Dy <- Dx <- 1 # diffusion coeff, X- and Y-direction
r <- 0.025 # consumption rate
n <- 100
y <- matrix(nrow = n, ncol = n, 10.)
# stodes is used, so we should specify the size of the work array (lrw)
# We take a rather large value
ST2 <- steady.2D(y, func = diffusion2D, parms = NULL, pos = TRUE,
dimens = c(n, n), lrw = 1000000,
atol = 1e-10, rtol = 1e-10, ctol = 1e-10)
# the actual size of lrw is in the attributes()$dims vector.
# it is best to set lrw as small as possible
attributes(ST2)
image(ST2, legend = TRUE)
# The hard way of plotting:
#y <- matrix(nr = n, nc = n, data = ST2$y)
# filled.contour(y, color.palette = terrain.colors)
## =======================================================================
## Diffusion in 2-D; extra flux on 2 boundaries, cyclic in y
## =======================================================================
diffusion2Db <- function(t, Y, par) {
y <- matrix(nr=nx,nc=ny,data=Y) # vector to 2-D matrix
dY <- -r*y # consumption
BNDx <- rep(1,nx) # boundary concentration
BNDy <- rep(1,ny) # boundary concentration
#diffusion in X-direction; boundaries=imposed concentration
Flux <- -Dx * rbind(y[1,]-BNDy,(y[2:nx,]-y[1:(nx-1),]),BNDy-y[nx,])/dx
dY <- dY - (Flux[2:(nx+1),]-Flux[1:nx,])/dx
#diffusion in Y-direction
Flux <- -Dy * cbind(y[,1]-BNDx,(y[,2:ny]-y[,1:(ny-1)]),BNDx-y[,ny])/dy
dY <- dY - (Flux[,2:(ny+1)]-Flux[,1:ny])/dy
# extra flux on two sides
dY[,1] <- dY[,1]+ 10
dY[1,] <- dY[1,]+ 10
# and exchange between sides on y-direction
dY[,ny] <- dY[,ny]+ (y[,1]-y[,ny])*10
return(list(as.vector(dY)))
}
# parameters
dy <- dx <- 1 # grid size
Dy <- Dx <- 1 # diffusion coeff, X- and Y-direction
r <- 0.025 # consumption rate
nx <- 50
ny <- 100
y <- matrix(nrow = nx, ncol = ny, 10.)
print(system.time(
ST2 <- steady.2D(y, func = diffusion2Db, parms = NULL, pos = TRUE,
dimens = c(nx, ny), verbose = TRUE, lrw = 283800,
atol = 1e-10, rtol = 1e-10, ctol = 1e-10,
cyclicBnd = 2) # y-direction: cyclic boundary
))
image(ST2)
#y <- matrix(nrow = nx, ncol = ny, data = ST2$y)
# filled.contour(y,color.palette=terrain.colors)
``` |