Self-Balancing Social Distancing
Do you have questions or comments about this model? Ask them here! (You'll first need to log in.)
WHAT IS IT?
A model of an infectious disease within a community.
People move about randomly, and if they run into a person who is sick, they have a chance of getting sick themselves. The illness runs on average two weeks, and has a 1% death rate. These processes are all stochastic, so somebody could die on day 2 of the illness, or could have it for a month
Here is what makes this model unique:
I have seen a couple of models online which illustrate the effects of social distancing on the spread of a disease, but they typically did this by social distancing the entire population, which to me seems like a bit of an unnecessary oversimplification. For this model, I'm interested in breaking down social distancing into two of its components: 1. How many people are distancing? Is it half the population? 75%? 10%? 2. How aggressively are they distancing? Are they literally staying in one place? or are they just being more cautious, going out less?
This model allows us to explore how varying 1 and 2 affect the outcome of the entire community. Is social distancing useless unless most people do it? Is social distancing useless unless it's extreme? Can we effect a meaningful change in outcome by having only a small proportion of citizens social distance?
Additionally, I've created a self-balancing mechanism which can be turned on or off. This self-balancing mechanism is to help try to model how a society would eaase out of social distancing while the virus was still present. A couple of new variables that the self-balancing uses:
[threshold] - this is the "acceptable" infected proportion of the population (ie, "the healthcare system can't handle more than 5% of the population sick with this thing at any one time, so we have to keep it under 5%")
[distance-step-ups] - when the proportion of people currently infected ([prop-infected]) nears the threshold, the proportion of the population social distancing increases. If the [prop-infected] rises above the threshold, or daily net new cases are too high (indicating rapid growth), the proportion of the population taking social distancing measures increases. So, for example, if 11% of the population is infected, the number of people taking social distancing measures would increase by ([distance-step-ups] * [num-people]) amount every day until prop-infected falls below 10%.
[distance-step-downs] - the same as the above, but for easing social distancing restrictions. Once prop-infected falls below ([threshold]*.75) and net new cases are negative, the proportion of people social distancing is decreased by ([distance-step-downs] * [num-people]).
What should the step-ups and step-downs be? That’s a good question. The granularity with which a society can adjust how many of its citizens are social distancing unclear and subject to political pressures. On one end of the spectrum, we could imagine a government announcing every morning which birthdays were allowed to leave the house. On the other end of the spectrum, we could imagine a government announcing at the start of the month that everyone must stay inside, and at the end of the month, letting everybody out. The most realistic course of action is probably somewhere in the middle. I think most citizens would get awfully fed up with a government micromanaging and making frequent adjustments, however, we probably want something more attentive than the second scenario. This is an added challenge — we want a self-balancing system that takes large enough steps so as not to require super frequent adjustments, but also keeps the virus at acceptable levels. To start, I’ve set the distancing-step-ups at 10%. I’ve done the same for distancing-step-downs. We can imagine city governments being able to adjust social distancing behavior with this sort of granularity by taking steps such as expanding and contracting the list of “essential businesses”. (Note: I am well aware of the fact that opening and shuttering businesses adds a whole new level of economic stress and greatly impacts these business’s ability to plan. That being said, I think what we’re interested in here is the effect of slowly opening or closing the economy on the spread of a virus, not the economic difficulties and feasibility of opening and closing the economy.)
HOW IT WORKS
&&
HOW TO USE IT
(what rules the agents use to create the overall behavior of the model)
I'll go through the variables and what is a good range for them to be set at:
num-people = This is the number of total people in the community, for this size grid, I think 1500 is a decent starting point. Less people -> disease has harder time spreading, more people -> disease spreads more easily
init-infected = This is how many initial cases the community has. You can set it as low as you want
num-people-social-distancing = of the total number of people, how many of them are socially distancing? Play around with this one, you start to be able to really see the impact once it's over 60% of the population. Also try it with zero! REMEMBER: this number must be lower than num-people
sociability-of-non-distancers = how much do non social distancers move around? The higher this number, the larger their movements. Try different numbers for this.
sociability-of-distancers = how much do social distancers move around? This number should be pretty low, I think generally, below 1. Play around with it, but just remember that it should be lower than sociability-of-non-distancers.
NOTE: the sociability of an individual does not change if he becomes infected. Distancers will continue to distance even if they become infected, but they will change color to red.
Once there are no more active infections in the community, people stop social distancing.
THINGS TO NOTICE
(suggested things for the user to notice while running the model)
THINGS TO TRY
(suggested things for the user to try to do (move sliders, switches, etc.) with the model)
EXTENDING THE MODEL
(suggested things to add or change in the Code tab to make the model more complicated, detailed, accurate, etc.)
NETLOGO FEATURES
(interesting or unusual features of NetLogo that the model uses, particularly in the Code tab; or where workarounds were needed for missing features)
RELATED MODELS
(models in the NetLogo Models Library and elsewhere which are of related interest)
CREDITS AND REFERENCES
To begin building this, I modified code for an SIR model by Paul Smaldino
Comments and Questions
globals [ max-infected cumulative-output daily-delta total-infected avg-daily-delta total-delta moving-average day-1 day-2 day-3 day-4 day-5 transmissibility ] turtles-own[ infected? immune? distanced? dead? ] to setup clear-all set-patch-size (275 / max-pxcor) setup-turtles setup-infected setup-distancers set max-infected (count turtles with [infected?]) set cumulative-output (0) set transmissibility (1) reset-ticks end to setup-turtles create-turtles num-people [ set color white set shape "person" set size 2 set infected? false set immune? false set distanced? false set dead? false setxy random-pxcor random-pycor ] end to setup-distancers ask n-of init-num-distancing turtles [ set color blue set distanced? true ] end to setup-infected ask n-of init-infected turtles [ set color red set infected? true ] end to go ;;stop if everyone or noone is infected ;;if (count turtles with [infected? and not dead?] = 0) ;;or (count turtles with [infected?] = num-people) infect-susceptibles recover-infected death recolor move-normal move-distancers calculate-max-infected calculalate-cumulative-output calculate-daily-delta if ticks > 0 [calculate-avg-daily-delta] calculate-total-infected if self-balancing = true[ adjust-distancers ] recolor calculate-moving-average tick end to adjust-distancers let daily-delta-prop (daily-delta / num-people) if ((total-infected / num-people) > (threshold) or ((daily-delta-prop) > .005)) [ if (daily-delta > 0) ;; let's also say that we need at least 5% of the population NOT distancing. [ if (((count turtles with [distanced? and not immune? and not dead?] + (num-people * distance-step-ups)) < (num-people * .95)) and ((count turtles with [not distanced? and not immune? and not dead?]) > (num-people * distance-step-ups))) [ let new-distanced ((num-people * distance-step-ups)) ask n-of new-distanced turtles with [not distanced? and not immune? and not dead?] [ set distanced? true if (color = black) [ set color blue] ] ] ] ] if ((total-infected / num-people) < (threshold * .75)) [ if (daily-delta < 0) [ let new-not-distanced (num-people * distance-step-downs) if (new-not-distanced < count turtles with [distanced? and not immune? and not dead?])[ ask n-of new-not-distanced turtles with [distanced? and not immune? and not dead?] [ set distanced? false if (color = blue) [ set color black] ] ] ] ] end to infect-susceptibles ;; S -> I ask turtles with [not infected? and not immune?][ let infected-neighbors (count turtles-here with [infected?]) if (random-float 1 < 1 - (((1 - transmissibility) ^ infected-neighbors))) [set infected? true] ;let infected-neighbors (count other turtles with [infected?] in-radius 1) ; if (random-float 1 < 1 - (((1 - transmissibility) ^ infected-neighbors)) and not immune?) ;[set infected? true] ] end to recolor ask turtles with [infected? and not dead?] [ set color red] ask turtles with [distanced? and not infected?] [ set color blue] ask turtles with [not distanced? and not infected?] [ set color white] ask turtles with [immune?] [set color gray] ask turtles with [dead?] [ set color pink] end to move-normal ask turtles with [not dead? and not distanced?] [ right random 360 ;;get a new random heading forward random sociability-of-non-distancers ] end to move-distancers ask turtles with [distanced? and not dead?][ right random 360 forward random Sociability-of-Distancers ] ask turtles with [dead?][ forward 0 ] end to recover-infected ;;I -> R ;;avg case length is 2 weeks. ;;should have 50% chance of becoming immune at 2 weeks ;;if we are saying each tick equals 1 day, ;;daily odds of recovering should be (1-x)^14=.5, x= 0.0483 ask turtles with [infected? and not dead?] [ if random-float 1 < 0.0483 [ set infected? false ifelse are-survivors-immune? [ set immune? true set color gray set distanced? false ] [set color white] ] ] end to death ;;avg case length is 2 weeks. ;;1% of infected die, ;;if we are saying each tick equals 1 day, ;;and 1% of sick patients should be dead at 2 weeks ;;daily mortality should be (1-x)^14=.99, x= 0.000717624 ask turtles with [infected?] [ if random-float 1 < 0.000717624 [set dead? true set color pink set infected? false ] ] end to calculate-max-infected let x (count turtles with [infected? and not dead?]) if x > max-infected [set max-infected x] if x = 0 [ask turtles with [distanced?][ set distanced? false set color white ] ] end to calculalate-cumulative-output let y ((count turtles with [infected? and not dead?] * .5) + (count turtles with [not infected? and not distanced? and not dead?] * 2) + (count turtles with [not infected? and distanced? and not dead?])) set cumulative-output (cumulative-output + y) end to calculate-total-infected set total-infected ((count turtles with [infected? and not dead?])) end to calculate-daily-delta set daily-delta (count turtles with [infected? and not dead?] - total-infected) end to calculate-avg-daily-delta let y (daily-delta) set total-delta (total-delta + y) set avg-daily-delta (total-delta / ticks) end to calculate-moving-average set day-1 (day-2) set day-2 (day-3) set day-3 (day-4) set day-4 (day-5) set day-5 (daily-delta) if (ticks > 4) [ set moving-average ((day-1 + day-2 + day-3 + day-4 + day-5) / 5) ] end to-report total-adjusted-output report cumulative-output / (num-people * 2 * (ticks)) end to-report calculate-daily-output report (((count turtles with [infected? and not dead?] * -1) + (count turtles with [dead?] * -5) + (count turtles with [not infected? and not distanced?] * 2) + (count turtles with [not infected? and distanced?] * 1.5)) / (num-people * 2)) end to-report max-infected-prop report max-infected / num-people end to-report prop-dead let y (count turtles with [dead?]) report y / num-people end to-report prop-distancing report ((count turtles with [distanced? and not dead? and not immune?]) / num-people) end to-report prop-uninfected report (count turtles with [not infected? and not immune?]) / num-people end
There are 2 versions of this model.
Attached files
No files
This model does not have any ancestors.
This model does not have any descendants.