In this blog post, we introduce an extension that allows you to calculate dynamic growth curves and ages at size in mizer. This addition will improve how mizer is calibrated to data and also allows it to be interpreted and compared to age-based models, potentially increasing the utility of mizer in the assessment and advice process.
The majority of assessment models for category 1 stocks in the International Council for Exploration of the Seas (ICES) are age based. This is a natural way of thinking about population dynamics as in a closed system, the number of individuals in a cohort, \(N\), can only go down as they die. The dynamics of a cohort are \[\begin{equation} \frac{dN}{dt}=-ZN \tag{1} \end{equation}\] where \(Z=F+M\), with \(F\) being the fishing mortality and \(M\) being the natural mortality. Fishing mortality and natural mortality can vary in time. For an age-based model this has the solution that the numbers at time \(t\) in age-class \(a\) is \[\begin{equation} N_{a,t}=N_{a-1,t-1}\exp(-Z_{a-1,t}) \tag{2} \end{equation}\] for \(a=2,\ldots{},A-1\) and \(t=1,\ldots\). For the first age-class, the number in the new cohort is the recruits \[\begin{equation} N_{1,t}=R_{t}\exp(-Z_{0,t}) \tag{3} \end{equation}\] for all \(t\). Often \(Z_{0,t}\) is zero if there are no catches of age-class one in the data. The final age class, \(A\), is a plus group, which means there is no maximum age of a fish, \[\begin{equation} N_{A,t}=N_{A-1,t-1}\exp(-Z_{A-1,t}) + N_{A,t-1}\exp(-Z_{A,t}), \tag{4} \end{equation}\] with \(Z_{a,t}=M_{a,t}+F_{a,t}\) for \(a=0,\ldots{},A\) and all times. Note that these are age classes that have the interval one year and not the ages of the fish. Some stocks have the first age class being age 0 (e.g. North Sea whiting), where as other have the first age class being older (e.g. age 3 for North Sea saithe).
Most category 1 assessments are based on equations (2)–(4), with additional features (e.g., process error in Nielsen and Berg 2014). The same equations are commonly the basis to give catch advice, calculate reference points (e.g. MSY) and perform management strategy evaluations (MSE).
A lot of data studies are specifically designed to collect data for age-based models. For example, fish that are aged on surveys using a length-stratified design. This design is good for finding the age at length, so that fish can be aged just by knowing their length and numbers at age can be calculated. However, the design is not as good at finding length from age, which means that growth curves can be difficult to calculate.
There are also many multispecies models that are age-based (Trijoulet et al. 2019; Trijoulet, Fay, and Miller 2020; Lewy and Vinther 2004). In some cases, emergent natural mortality from age-based multispecies models are used as inputs to single-species assessments (e.g., ICES 2024).
Despite strong arguments for adopting size-based models over age-based models (Andersen 2019) , we are unaware of any data-rich assessments that use size-based models. This work was motivated by the aim to use the multispecies size spectrum model, mizer (Hartvig, Andersen, and Beyer 2011; Blanchard et al. 2014; Scott, Blanchard, and Andersen 2014) as an age-based model to fit with current assessment and MSE frameworks. This will take advantage of the mechanistic dynamics of the mizer model, while keeping to the age-structured frameworks used in ICES.
In this blog we will explain and demonstrate how we calculate emergent age in mizer.
Growth is an emergent property of feeding (Andersen 2019) and so is size-at-age. To calculate the emergent size at age in mizer at time \(t_0\), we introduce an immortal individual (II) of species \(i\) and track the size of that individual. At time \(t\), the size of the II will be \(I_i(t_0,t)\), with \[\begin{equation} \frac{\partial I_i}{\partial t} \;=\; g_i\bigl(I_i(t_0,t),t\bigr). \tag{5} \end{equation}\] Solving this from \(t_0\) to time \(T\) with boundary condition \(I_i(t_0, t_0) = w_{0,i}\), we know that an individual of species \(i\) and age \(T - t_0\) will be size \(I_i(t_0, T)\). In mizer, fish cannot overtake each other in size, therefore we know that fish smaller than \(I_i(T)\) will be younger, and larger fish will be older. If we introduce such IIs at annual increments, then we can follow different age classes throughout the model. Figure 1 demonstrates the process, showing how, for a given species, age is calculated.
Figure 1: Demonstration of how we calculate weights for immortal individuals across time. Each line represents the weight of one II, each introduced annually.
We solve equation (5) in mizer for time dt
numerically given weight classes w
, with differences dw
. If at the current time the II is size cur_waa
then it will be in size class
and will grow at rate
g <- grow[cur_idx]
time_to_end <- (togro - cur_waa) / g # how far is next size class, can i grow it in dt?
It will grow at this rate until it reaches the next size class
togro <- params@w[cur_idx + 1]
which will take time
time_to_end <- (togro - cur_waa) / g
Initially the II will grow for time dt
but will move through weight classes. We monitor how long is left to grow with variable time_left
intially setting it to dt
. If time_to_end<=time_left
then the II’s weight will be
cur_waa <- cur_waa + g * time_left
otherwise it will grow to the next size class for time_to_end
and we update the time remaining
cur_waa <- cur_waa + g * time_to_end
time_left <- time_left - time_to_end
The process is repeated until time_to_end<=time_left
.
Figure 2 shows the emergent weight at age for whiting in mizer.
Figure 2: Emergent growth curve for whiting.
We added the above to the project
and project_simple
functions so that it updates the size of the IIs each dt
. In addition we added a new slot to the MizerSim
objects so that we can store the weight at age for all ages, time steps and species, waa
.
#the mizerSim object - now with a waa section
setClass(
"MizerSim",
slots = c(
params = "MizerParams",
n = "array",
effort = "array",
n_pp = "array",
n_other = "array",
waa = "array"
)
)
Each year we add a new II, and also remove the oldest II at the same time. We give an argument to project
to input the initial weights at age (waa_initial
), a matrix with dimensions number of species and maximum age (defaulted to 20), nages
. If absent the initial weights at age are calculated from the getGrowthCurves
function.
The project
function will now return a mizerSim
object with a waa
slot which contains an array
with the dimensions t_max*t_save
, number of species and nages
.
We demonstrate the weight at age using the North Sea model with fishing mortality from stock assessments (historicfishingM
see Spence et al. (2024)) from 1984 to 2019. We ran the model to the steady state by running it for 100 years using 1984 fishing and then projected it to 2019.
#project to steady state - so that we can use the waa generated here as initial_waa
#fishing array is the historicfishingM remade for mizer
tosteady <- project(params, effort = fishing_array[1,],t_max =100, initial_n = params@initial_n)
#projecting, using the values from the end of the steady state run for n, npp and waa
projection <- project(params, effort = fishing_array, initial_n = (tosteady@n)[(dim(tosteady@n)[1]),,],
initial_n_pp = (tosteady@n_pp)[(dim(tosteady@n)[1]),],
waa_initial = tosteady@waa[1001,,])
Figure 3: The black line shows the weight at age inputs to SMS’ key-run (ICES 2024) (and the stock assessent) for whiting and the red line shows the mizer estimate of weight at age for whiting using NS_params
In Figure 3 we show the size at age of whiting and compare it to weight at age inputs to SMS’ key-run (ICES 2024). Using the default mizer model NS_params
we show that the weight at age of whiting changes over the simulation, with a similar pattern to the inputs to SMS, however, the scale and range are different.
Given the IIs we can calculate the numbers at age. Suppose that at time \(t\) the density of numbers at weight is shown below, with IIs aged one, \(I_i(t-1,t)=0.2\), and two, \(I_i(t-2,t)=5\), being shown.
The numbers younger than age one are all of the individuals smaller than \(I_i(t-1,t)=0.2\), which in the figure above is 14671.88. The number of individuals younger than two years old are smaller than \(I_i(t-2,t)=5\), which in the figure is 17671.88. Therefore, 14671.88 are in the youngest age class (less than one years old) and 3000 individuals are in the age class between one and two years old.
In this example the IIs were on the boundary of the weight-classes. However, in practice this will almost surely not happen. The IIs will fall between weight-classes, and so, to assign an age to individuals in that weight-class we describe how individuals are distributed within the weight class. The plot below is more likely:
It is clear that the individuals in the red bars are less than one, and the individuals in the blue bar are less than two but older than one, but it is not clear how old the individuals in the white bars are.
Defining how the individuals in the white bars are distributed is down to the user. In this work we defined it as following a power law. Specifically the density of individuals between sizes \(w_0\) and \(w_1\) was \[\begin{equation} \alpha{}w^{\beta}. \end{equation}\] We set \[\begin{equation} \beta=\frac{\log(n(w_1))-\log(n(w_0))}{\log(w_1)-\log(w_0)}, \end{equation}\] and \[\begin{equation} \alpha=\frac{n(w_0)(w_1-w_0)(\beta+1)}{w_1^{\beta+1}-w_0^{\beta+1}}, \end{equation}\] which ensures that the total number in the weight-class is constant in both cases.
The example above becomes:
Figure 4: The top plot shows the evolution of individual whiting cohorts through time. The lower plots shows the fishing mortality over time for whiting from Spence et al. (2024)
By calculating the numbers at age, as done in Section ??, we can calculate how many individuals die in a cohort in a year. Consider that at time \(t-1\) and \(t\) we have calculated numbers age \(a-1\) and \(a\) respectively, then we can rearrange equation (2) to find \[\begin{equation} Z_{a-1,t-1}=-\ln(N_{a,t})+\ln(N_{a-1,t-1}). \tag{6} \end{equation}\] This is the total mortality for age \(a-1\) at time \(t-1\). By calculating the numbers caught at age \(a\) in time \(t-1\), \(C_{a-1,t-1}\), in a similar way to how we calculate the numbers at age, we can calculate the fishing mortality \[\begin{equation} F_{a-1,t-1}=\frac{Z_{a-1,t-1}C_{a-1,t-1}}{N_{a-1,t-1}(1-\exp(-Z_{a-1,t-1}))} \tag{7} \end{equation}\] and the natural mortality \[\begin{equation} M_{a-1,t-1}=Z_{a-1,t-1}-F_{a-1,t-1}. \tag{8} \end{equation}\]
After a simulation we get the numbers at age and the numbers caught at age
numbers <- get_num_by_year_by_age(sim)
catch <- get_catch_by_year_by_age(sim)
The get_catch_by_year_by_age
function works the same as get_num_by_year_by_age
, but prior to the numbers in size classes being summed over ages, they are multiplied by the size, species specific fishing mortality, acquired from getFMort
. The catch here is the same as in mizer’s getYield
.
We calculate equations (6) and (7) for the i
th year, j
th species and k
th age class to get \(Z_{i,k}\) and \(F_{i,k}\)
<- -log(numbers[i+1,j,k+1]) + log(numbers[i,j,k])
z <- catch[i-1,j,k]*z)/(numbers[i,j,k]-next_alive) ### F_m rather than F as F is protected F_m
and equation (8) to get \(M_{i,k}\)
m <- z - F_m
Figure 5 is the emergent \(Z\), \(F\) and \(M\) over time using the historical fishing mortality.
Figure 5: Plots depict the \(Z\), \(F\) and \(M\) rates for whiting of different ages over time
no_w
and dt
Due to the numerical integration of the partial differential equation (pde) in mizer, the II do not follow the distribution of fish density exactly and this sometimes results in negative \(M\). To investigate this we followed one individual and their growth for different values of dt
and no_w
. We initialised the params
object for no_w
equal to 100, 1000 and 10000
params <- newMultispeciesParams(NS_params@species_params, no_w = no_w)
allgrow <- getEGrowth(params)
gs <- allgrow[6,]
and assumed a constant growth rate gs
for a whiting individual. We initialised the population at zero except in the first size class where we set it so that there was only one individual.
Assuming no mortality and no recruitment we updated the numbers at size for 10 years using the numerical pde solver in mizer (see supplementary for the code) and compared the size distribution with the size of an II at the same age.
Figure 6: Comparison of II size prediction to mizer’s numerical solution. The grid cell value is the proportion of individuals in the numerical solution that are smaller than the II. Each simulation has a different dt
and no_w
.
Figure 7: The numerical solution of the growth with different dt
and no_w
. Each plot contains 3 lines, indicating a different value of dt
- black is 0.1, blue is 0.01 and red is 0.001. Each row of plots indicates that the example’s no_w
, increasing from top to bottom. The left columns shows age 1 and right age 4.
We compared the proportion of individuals that are smaller than the II and found that as dt
got smaller and no_w
got bigger the proportion approached 50%, Figure 6. Further, as dt
got smaller and no_w
got bigger the range of sizes at age decreased Figure 7.
Therefore for more accurate age-based information we are required to choose a finer resolution for the numerical solver in mizer. We demonstrate this on the North Sea model for values of dt
equal to 0.1, 0.01 and 0.001, and no_w
equal to 100, 1000 and 2000.
100 | 1000 | 2000 | |
---|---|---|---|
0.1 | 0.1015873 | 0.0099206 | 0.002381 |
0.01 | 0.0746032 | 0.0000000 | 0.000000 |
0.001 | 0.0726190 | 0.0000000 | NA |
Decreasing dt
and increasing no_w
makes the numerial integration more accurate and stablises the results. This comes at a cost, as the MizerSim
object gets very large.
Most models used in advice are age-based. In this blog post, we have introduced a way of calculating emergent age in mizer, potentially increasing the utility of mizer when it comes to giving management advice.
For example, mizer could be used as an operating model for MSEs. A common MSE framework is to use an operating model to test management actions. The operating model stands in for reality and management actions, such as harvest control rules, are run (Punt et al. 2016). This process involves the operating model generating data, which assessment models are fitted to. As data for assessment models is age resolved, the operating model has to be able to sample age resolved data. By introducing emergent age in mizer, mizer can now generate age-based data, meaning that it can be used as an operating model in the MSE frameworks.
Some category 1 assessments, those that have full analytical assessments, use natural mortality at age from fitted multispecies models as inputs to the assessment. For example, many North Sea stocks use natural mortality estimates from the stochastic multispecies model (Lewy and Vinther 2004). Typically, assessment models require natural mortality at age rather than size. In Section ?? we showed that we can calculate emergent natural mortality at age. Therefore, coupled with dynamical fitting, (Spence et al. 2021), mizer could be used to provide natural mortality estimates as inputs to single-species assessments.
Many data collection schemes are designed to be fit to age-based models, for example, length stratified sample is used on surveys to calculate age-length keys. For these reasons, it can be difficult to incorporate age-based data when calibrating mizer. Furthermore, the data collected are not in a steady state (e.g., see Figure 3). Using the ages process described here, we can compare these data with mizer. For example, growth could be tuned by looking at emergent growth rates rather than steady states.
In many cases, fishing mortality from stock assessments has been used to drive the dynamics of the mizer model (Blanchard et al. 2014; Spence, Blackwell, and Blanchard 2016; Spence et al. 2024). Typically fishing mortality from stock assessments is age-based. In previous studies, the fishing mortality inputs were some kind of average, for example Fbar, or some steady-state assumption about age (Spence et al. 2024). Using this approach, age-based fishing mortality can be implemented in age classes as they change size throughout the simulation as done in the assessment. However, how selectivity occurs within an age can be difficult to calculate (Spence et al. 2024), particularly at lower ages.
The work presented here is exact in the limit when dt
and dw
are 0, however practically this is not possible. We showed that as dt
and dw
get smaller the age estimation becomes more accurate, however, inaccuracies in the ageing process can lead to some strange behaviour. For example, we found that for the default values of dt
and dw
some cohorts natural mortality was less than 0, i.e. \(M<0\), which could cause difficulties when comparing mizer to age-based models. This could be caused by the numerical integration and/or how we say the lengths of fish within an age class are distributed. More research is required, although we suspect these problems occur due to how the emergent catch is calculated.
In this blog post, we introduced an extension that allows you to calculate dynamic growth curves and ages at size in mizer. This addition will improve how mizer is calibrated to data and also allows it to be interpreted and compared to age-based models, potentially increasing the utility of mizer in the assessment and advice process.
If you see mistakes or want to suggest changes, please create an issue on the source repository.
Text and figures are licensed under Creative Commons Attribution CC BY 4.0. Source code is available at https://github.com/sizespectrum/MizerBlog/, unless otherwise noted. The figures that have been reused from other sources don't fall under this license and can be recognized by a note in their caption: "Figure from ...".
For attribution, please cite this work as
Broadbent & Spence (2025, April 2). mizer blog: Age in mizer. Retrieved from https://blog.mizer.sizespectrum.org/posts/2025-04-02-age-in-mizer/
BibTeX citation
@misc{broadbent2025age, author = {Broadbent, Luke and Spence, Michael A.}, title = {mizer blog: Age in mizer}, url = {https://blog.mizer.sizespectrum.org/posts/2025-04-02-age-in-mizer/}, year = {2025} }