A freezing bucket
A common laboratory experiment freezes an insultated bucket of water from the top down, using a metal lid to keep the top of the bucket at some constant, very cold temperature. In this example, we simulate such a scenario using SlabSeaIceModel
. Here, the bucket is perfectly insulated and infinitely deep, like many buckets are: if the Simulation
is run for longer, the ice will keep freezing, and freezing, and will never run out of water. Also, the water in the infinite bucket is (somehow) all at the same temperature, in equilibrium with the ice-water interface (and therefore fixed at the melting temperature). Yep, this kind of thing happens all the time.
We start by using Oceananigans
to bring in functions for building grids and Simulation
s and the like.
using Oceananigans
using Oceananigans.Units
Next we using ClimaSeaIce
to get some ice-specific names.
using ClimaSeaIce
An infinitely deep bucket with a single grid point
Perhaps surprisingly, we need just one grid point to model an possibly infinitely thick slab of ice with SlabSeaIceModel
. We would only need more than 1 grid point if our boundary conditions vary in the horizontal direction.
grid = RectilinearGrid(size=(), topology=(Flat, Flat, Flat))
1×1×1 RectilinearGrid{Float64, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat} on Oceananigans.Architectures.CPU with 0×0×0 halo
├── Flat x
├── Flat y
└── Flat z
Next, we build our model of an ice slab freezing into a bucket. We start by defining a constant internal ConductiveFlux
with ice_conductivity
conductivity = 2 # kg m s⁻³ K⁻¹
internal_heat_flux = ConductiveFlux(; conductivity)
ConductiveFlux{Float64}(2.0)
Note that other units besides Celsius can be used, but that requires setting model.phase_transitions` with appropriate parameters. We set the ice heat capacity and density as well,
ice_heat_capacity = 2100 # J kg⁻¹ K⁻¹
ice_density = 900 # kg m⁻³
phase_transitions = PhaseTransitions(; ice_heat_capacity, ice_density)
PhaseTransitions{Float64, ClimaSeaIce.SeaIceThermodynamics.LinearLiquidus{Float64}}(900.0, 2100.0, 999.8, 4186.0, 334000.0, 0.0, ClimaSeaIce.SeaIceThermodynamics.LinearLiquidus{Float64}(0.0, 0.054))
We set the top ice temperature,
top_temperature = -10 # ᵒC
top_heat_boundary_condition = PrescribedTemperature(-10)
PrescribedTemperature{Int64}(-10)
Construct the thermodynamics of sea ice, for this we use a simple slab sea ice representation of thermodynamics
ice_thermodynamics = SlabSeaIceThermodynamics(grid;
internal_heat_flux,
phase_transitions,
top_heat_boundary_condition)
SlabSeaIceThermodynamics
├── top_surface_temperature: ConstantField(-10)
└── minimium_ice_thickness: ConstantField(0.0)
Then we assemble it all into a model,
model = SeaIceModel(grid; ice_thermodynamics)
SeaIceModel{Oceananigans.Architectures.CPU, Oceananigans.Grids.RectilinearGrid}(time = 0 seconds, iteration = 0)
├── grid: 1×1×1 RectilinearGrid{Float64, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat} on Oceananigans.Architectures.CPU with 0×0×0 halo
├── ice_thermodynamics: SlabThermodynamics
├── advection: Nothing
└── external_heat_fluxes:
├── top: FluxFunction of slab_internal_heat_flux with parameters (flux=ConductiveFlux{Float64}, liquidus=ClimaSeaIce.SeaIceThermodynamics.LinearLiquidus{Float64}, bottom_heat_boundary_condition=ClimaSeaIce.SeaIceThermodynamics.HeatBoundaryConditions.IceWaterThermalEquilibrium{Int64})
└── bottom: 0
Note that the default bottom heat boundary condition for SlabSeaIceThermodynamics
is IceWaterThermalEquilibrium
with freshwater. That's what we want!
model.ice_thermodynamics.heat_boundary_conditions.bottom
ClimaSeaIce.SeaIceThermodynamics.HeatBoundaryConditions.IceWaterThermalEquilibrium{Int64}(0)
Ok, we're ready to freeze the bucket for 10 straight days with an initial ice thickness of 1 cm,
simulation = Simulation(model, Δt=10minute, stop_time=10days)
set!(model, h=0.01)
Collecting data and running the simulation
Before simulating the freezing bucket, we set up a Callback
to create a timeseries of the ice thickness saved at every time step.
# Container to hold the data
timeseries = []
# Callback function to collect the data from the `sim`ulation
function accumulate_timeseries(sim)
h = sim.model.ice_thickness
push!(timeseries, (time(sim), first(h)))
end
# Add the callback to `simulation`
simulation.callbacks[:save] = Callback(accumulate_timeseries)
Callback of accumulate_timeseries on IterationInterval(1)
Now we're ready to hit the Big Red Button (it should run pretty quick):
run!(simulation)
[ Info: Initializing simulation...
[ Info: ... simulation initialization complete (1.590 seconds)
[ Info: Executing initial time step...
[ Info: ... initial time step complete (1.570 seconds).
[ Info: Simulation is stopping after running for 3.617 seconds.
[ Info: Simulation time 10 days equals or exceeds stop time 10 days.
Visualizing the result
It'd be a shame to run such a "cool" simulation without looking at the results. We'll visualize it with Makie.
using CairoMakie
timeseries
is a Vector
of Tuple
. So we have to do a bit of processing to build Vector
s of time t
and thickness h
. It's not much work though:
t = [datum[1] for datum in timeseries]
h = [datum[2] for datum in timeseries]
1441-element Vector{Float64}:
0.01
0.013115723787646819
0.015618315847984355
0.01777126422864224
0.019689999447000853
0.02143760767315016
0.023053076484693194
0.024562507006759427
0.02598439207428873
0.0273324015606709
⋮
0.3209657543453097
0.32107769484037885
0.32118959632212435
0.32130145883130834
0.3214132824086221
0.3215250670946861
0.32163681293005025
0.321748519955194
0.3218601882105266
Just for fun, we also compute the velocity of the ice-water interface:
dhdt = @. (h[2:end] - h[1:end-1]) / simulation.Δt
1440-element Vector{Float64}:
5.192872979411364e-6
4.170986767229228e-6
3.588247301096473e-6
3.1978920305976893e-6
2.9126803769155148e-6
2.6924480192383877e-6
2.5157175367770553e-6
2.3698084458821715e-6
2.246682477303619e-6
2.1409626325971827e-6
⋮
1.8663258204348886e-7
1.8656749178188234e-7
1.8650246957583697e-7
1.8643751530665145e-7
1.8637262885627206e-7
1.863078101066451e-7
1.8624305894027193e-7
1.8617837523956146e-7
1.8611375888766267e-7
All that's left, really, is to put those lines!
in an Axis
:
set_theme!(Theme(fontsize=24, linewidth=4))
fig = Figure(size=(1200, 600))
axh = Axis(fig[1, 1], xlabel="Time (days)", ylabel="Ice thickness (cm)")
axd = Axis(fig[1, 2], xlabel="Ice thickness (cm)", ylabel="Freezing rate (μm s⁻¹)")
lines!(axh, t ./ day, 1e2 .* h)
lines!(axd, 1e2 .* h[1:end-1], 1e6 .* dhdt)
fig
If you want more ice, you can increase simulation.stop_time
and run!(simulation)
again (or just re-run the whole script).
This page was generated using Literate.jl.