source: LMDZ6/branches/contrails/libf/phylmd/StratAer/coagulate.f90 @ 5460

Last change on this file since 5460 was 5285, checked in by abarral, 2 months ago

As discussed internally, remove generic ONLY: ... for new _mod_h modules

  • Property svn:keywords set to Id
File size: 12.0 KB
Line 
1!
2! $Id: coagulate.f90 5285 2024-10-28 13:33:29Z fhourdin $
3!
4SUBROUTINE COAGULATE(pdtcoag,mdw,tr_seri,t_seri,pplay,dens_aer,is_strato)
5  !     -----------------------------------------------------------------------
6  !
7  !     Author : Christoph Kleinschmitt (with Olivier Boucher)
8  !     ------
9  !
10  !     purpose
11  !     -------
12  !
13  !     interface
14  !     ---------
15  !      input
16  !       pdtphys        time step duration                 [sec]
17  !       tr_seri        tracer mixing ratios               [kg/kg]
18  !       mdw             # or mass median diameter          [m]
19  !
20  !     method
21  !     ------
22  !
23  !     -----------------------------------------------------------------------
24
25  USE yomcst_mod_h
26  USE dimphy, ONLY : klon,klev
27  USE aerophys
28  USE infotrac_phy
29  USE phys_local_var_mod, ONLY: DENSO4, DENSO4B, f_r_wet, f_r_wetB
30  USE strataer_local_var_mod, ONLY: flag_new_strat_compo
31
32  IMPLICIT NONE
33
34  !--------------------------------------------------------
35
36  ! transfer variables when calling this routine
37  REAL,INTENT(IN)                               :: pdtcoag ! Time step in coagulation routine [s]
38  REAL,DIMENSION(nbtr_bin),INTENT(IN)           :: mdw     ! aerosol particle diameter in each bin [m]
39  REAL,DIMENSION(klon,klev,nbtr),INTENT(INOUT)  :: tr_seri ! Concentration Traceur [U/KgA]
40  REAL,DIMENSION(klon,klev),INTENT(IN)          :: t_seri  ! Temperature
41  REAL,DIMENSION(klon,klev),INTENT(IN)          :: pplay   ! pression pour le mileu de chaque couche (en Pa)
42  REAL,DIMENSION(klon,klev)                     :: dens_aer! density of aerosol [kg/m3 aerosol] with default H2SO4 mass
43  LOGICAL,DIMENSION(klon,klev),INTENT(IN)       :: is_strato
44
45  ! local variables in coagulation routine
46  INTEGER                                       :: i,j,k,nb,ilon,ilev
47  REAL, DIMENSION(nbtr_bin)                     :: radiusdry ! dry aerosol particle radius in each bin [m]
48  REAL, DIMENSION(nbtr_bin)                     :: radiuswet ! wet aerosol particle radius in each bin [m]
49  REAL, DIMENSION(klon,klev,nbtr_bin)           :: tr_t ! Concentration Traceur at time t [U/KgA]
50  REAL, DIMENSION(klon,klev,nbtr_bin)           :: tr_tp1 ! Concentration Traceur at time t+1 [U/KgA]
51  REAL, DIMENSION(nbtr_bin,nbtr_bin,nbtr_bin)   :: ff   ! Volume fraction of intermediate particles
52  REAL, DIMENSION(nbtr_bin)                     :: Vdry ! Volume dry of bins
53  REAL, DIMENSION(nbtr_bin)                     :: Vwet ! Volume wet of bins
54  REAL, DIMENSION(nbtr_bin,nbtr_bin)            :: Vij  ! Volume sum of i and j
55  REAL                                          :: eta  ! Dynamic viscosity of air
56  REAL, PARAMETER                               :: mair=4.8097E-26 ! Average mass of an air molecule [kg]
57  REAL                                          :: zrho ! Density of air
58  REAL                                          :: mnfrpth ! Mean free path of air
59  REAL, DIMENSION(nbtr_bin)                     :: Kn   ! Knudsen number of particle i
60  REAL, DIMENSION(nbtr_bin)                     :: Di   ! Particle diffusion coefficient
61  REAL, DIMENSION(nbtr_bin)                     :: m_par   ! Mass of particle i
62  REAL, DIMENSION(nbtr_bin)                     :: thvelpar! Thermal velocity of particle i
63  REAL, DIMENSION(nbtr_bin)                     :: mfppar  ! Mean free path of particle i
64  REAL, DIMENSION(nbtr_bin)                     :: delta! delta of particle i (from equation 21)
65  REAL, DIMENSION(nbtr_bin,nbtr_bin)            :: beta ! Coagulation kernel from Brownian diffusion
66  REAL                                          :: beta_const ! Constant coagulation kernel (for comparison)
67  REAL                                          :: num
68  REAL                                          :: numi
69  REAL                                          :: denom
70
71  ! Additional variables for coagulation enhancement factor due to van der Waals forces
72  ! Taken from Chan and Mozurkewich, Measurement of the coagulation rate constant for sulfuric acid
73  ! particles as a function of particle size using TDMA, Aerosol Science, 32, 321-339, 2001.
74  !--ok_vdw is 0 for no vdW forces, 1 for E(0), 2 for E(infinity)
75  INTEGER, PARAMETER                            :: ok_vdw = 0
76  REAL, PARAMETER                               :: avdW1 = 0.0757
77  REAL, PARAMETER                               :: avdW3 = 0.0015
78  REAL, PARAMETER                               :: bvdW0 = 0.0151
79  REAL, PARAMETER                               :: bvdW1 = -0.186
80  REAL, PARAMETER                               :: bvdW3 = -0.0163
81  REAL, PARAMETER                               :: AvdW = 6.4e-20 !Hamaker constant in J = 1e7 erg
82  REAL                                          :: AvdWi
83  REAL                                          :: xvdW
84  REAL                                          :: EvdW
85
86
87! ff(i,j,k): Volume fraction of Vi,j that is partitioned to each model bin k
88! just need to be calculated in model initialization because mdw(:) size is fixed
89! no need to recalculate radius, Vdry, Vij, and ff every timestep because it is for 
90! dry aerosols
91  DO i=1, nbtr_bin
92     radiusdry(i)=mdw(i)/2.
93     Vdry(i)=radiusdry(i)**3.  !neglecting factor 4*RPI/3
94     Vwet(i)=0.0
95  ENDDO
96
97  DO j=1, nbtr_bin
98     DO i=1, nbtr_bin
99        Vij(i,j)= Vdry(i)+Vdry(j)
100     ENDDO
101  ENDDO
102 
103!--pre-compute the f(i,j,k) from Jacobson equation 13
104  ff=0.0
105  DO k=1, nbtr_bin
106  DO j=1, nbtr_bin
107  DO i=1, nbtr_bin
108    IF (k.EQ.1) THEN
109      ff(i,j,k)= 0.0
110    ELSEIF (k.GT.1.AND.Vdry(k-1).LT.Vij(i,j).AND.Vij(i,j).LT.Vdry(k)) THEN
111      ff(i,j,k)= 1.-ff(i,j,k-1)
112    ELSEIF (k.EQ.nbtr_bin) THEN
113      IF (Vij(i,j).GE.Vdry(k)) THEN
114        ff(i,j,k)= 1.
115      ELSE
116        ff(i,j,k)= 0.0
117      ENDIF
118    ELSEIF (k.LE.(nbtr_bin-1).AND.Vdry(k).LE.Vij(i,j).AND.Vij(i,j).LT.Vdry(k+1)) THEN
119      ff(i,j,k)= Vdry(k)/Vij(i,j)*(Vdry(k+1)-Vij(i,j))/(Vdry(k+1)-Vdry(k))
120    ENDIF
121  ENDDO
122  ENDDO
123  ENDDO
124! End of just need to be calculated at initialization because mdw(:) size is fixed
125 
126  DO ilon=1, klon
127  DO ilev=1, klev
128  !only in the stratosphere
129  IF (is_strato(ilon,ilev)) THEN
130  !compute actual wet particle radius & volume for every grid box
131  IF(flag_new_strat_compo) THEN
132     DO i=1, nbtr_bin
133        radiuswet(i)=f_r_wetB(ilon,ilev,i)*mdw(i)/2.
134        Vwet(i)= radiuswet(i)**3.  !neglecting factor 4*RPI/3
135!!      Vwet(i)= Vdry(i)*(f_r_wetB(ilon,ilev,i)**3)
136     ENDDO
137  ELSE
138     DO i=1, nbtr_bin
139        radiuswet(i)=f_r_wet(ilon,ilev)*mdw(i)/2.
140        Vwet(i)= radiuswet(i)**3.  !neglecting factor 4*RPI/3
141!!      Vwet(i)= Vdry(i)*(f_r_wet(ilon,ilev)**3)
142     ENDDO
143  ENDIF
144 
145!--Calculations for the coagulation kernel---------------------------------------------------------
146
147  zrho=pplay(ilon,ilev)/t_seri(ilon,ilev)/RD
148
149!--initialize the tracer at time t and convert from [number/KgA] to [number/m3]
150  DO i=1, nbtr_bin
151  tr_t(ilon,ilev,i) = tr_seri(ilon,ilev,i+nbtr_sulgas) * zrho
152  ENDDO
153
154  ! mean free path of air (Pruppacher and Klett, 2010, p.417) [m]
155  mnfrpth=6.6E-8*(1.01325E+5/pplay(ilon,ilev))*(t_seri(ilon,ilev)/293.15)
156  ! mnfrpth=2.*eta/(zrho*thvelair)
157  ! mean free path of air (Prupp. Klett) in [10^-6 m]
158  ! ZLAIR = 0.066 *(1.01325E+5/PPLAY)*(T_SERI/293.15)*1.E-06
159
160  ! dynamic viscosity of air (Pruppacher and Klett, 2010, p.417) [kg/(m*s)]
161  IF (t_seri(ilon,ilev).GE.273.15) THEN
162    eta=(1.718+0.0049*(t_seri(ilon,ilev)-273.15))*1.E-5
163  ELSE
164    eta=(1.718+0.0049*(t_seri(ilon,ilev)-273.15)-1.2E-5*(t_seri(ilon,ilev)-273.15)**2)*1.E-5
165  ENDIF
166
167!--pre-compute the particle diffusion coefficient Di(i) from equation 18
168  Di=0.0
169  DO i=1, nbtr_bin
170      Kn(i)=mnfrpth/radiuswet(i)
171      Di(i)=RKBOL*t_seri(ilon,ilev)/(6.*RPI*radiuswet(i)*eta)*(1.+Kn(i)*(1.249+0.42*exp(-0.87/Kn(i))))
172  ENDDO
173
174!--pre-compute the thermal velocity of a particle thvelpar(i) from equation 20
175  thvelpar=0.0
176  IF(flag_new_strat_compo) THEN
177     DO i=1, nbtr_bin
178        m_par(i)=4./3.*RPI*radiuswet(i)**3.*DENSO4B(ilon,ilev,i)*1000.
179        thvelpar(i)=sqrt(8.*RKBOL*t_seri(ilon,ilev)/(RPI*m_par(i)))
180     ENDDO
181  ELSE
182     DO i=1, nbtr_bin
183        m_par(i)=4./3.*RPI*radiuswet(i)**3.*DENSO4(ilon,ilev)*1000.
184        thvelpar(i)=sqrt(8.*RKBOL*t_seri(ilon,ilev)/(RPI*m_par(i)))
185     ENDDO
186  ENDIF
187
188!--pre-compute the particle mean free path mfppar(i) from equation 22
189  mfppar=0.0
190  DO i=1, nbtr_bin
191   mfppar(i)=8.*Di(i)/(RPI*thvelpar(i))
192  ENDDO
193
194!--pre-compute the mean distance delta(i) from the center of a sphere reached by particles
195!--leaving the surface of the sphere and traveling a distance of particle mfppar(i) from equation 21
196  delta=0.0
197  DO i=1, nbtr_bin
198      delta(i)=((2.*radiuswet(i)+mfppar(i))**3.-(4.*radiuswet(i)**2.+mfppar(i)**2.)**1.5)/ &
199           & (6.*radiuswet(i)*mfppar(i))-2.*radiuswet(i)
200  ENDDO
201
202!   beta(i,j): coagulation kernel (rate coefficient) of 2 colliding particles i,j
203!--pre-compute the beta(i,j) from equation 17 in Jacobson
204  num=0.0
205  DO j=1, nbtr_bin
206  DO i=1, nbtr_bin
207!
208     num=4.*RPI*(radiuswet(i)+radiuswet(j))*(Di(i)+Di(j))
209     denom=(radiuswet(i)+radiuswet(j))/(radiuswet(i)+radiuswet(j)+sqrt(delta(i)**2.+delta(j)**2.))+ &
210          & 4.*(Di(i)+Di(j))/(sqrt(thvelpar(i)**2.+thvelpar(j)**2.)*(radiuswet(i)+radiuswet(j)))
211     beta(i,j)=num/denom
212!
213!--compute enhancement factor due to van der Waals forces
214   IF (ok_vdw .EQ. 0) THEN      !--no enhancement factor
215      Evdw=1.0
216   ELSEIF (ok_vdw .EQ. 1) THEN  !--E(0) case
217      AvdWi = AvdW/(RKBOL*t_seri(ilon,ilev))*(4.*radiuswet(i)*radiuswet(j))/(radiuswet(i)+radiuswet(j))**2.
218      xvdW = LOG(1.+AvdWi)
219      EvdW = 1. + avdW1*xvdW + avdW3*xvdW**3
220   ELSEIF (ok_vdw .EQ. 2) THEN  !--E(infinity) case
221      AvdWi = AvdW/(RKBOL*t_seri(ilon,ilev))*(4.*radiuswet(i)*radiuswet(j))/(radiuswet(i)+radiuswet(j))**2.
222      xvdW = LOG(1.+AvdWi)
223      EvdW = 1. + SQRT(AvdWi/3.)/(1.+bvdW0*SQRT(AvdWi)) + bvdW1*xvdW + bvdW3*xvdW**3.
224   ENDIF
225!
226   beta(i,j)=beta(i,j)*EvdW
227
228  ENDDO
229  ENDDO
230
231!--external loop for equation 14
232  DO k=1, nbtr_bin
233
234!--calculating denominator sum
235  denom=0.0
236  DO j=1, nbtr_bin
237     !    fraction of coagulation of k and j that is not giving k
238     denom=denom+(1.-ff(k,j,k))*beta(k,j)*tr_t(ilon,ilev,j)
239  ENDDO
240
241  IF (k.EQ.1) THEN
242!--calculate new concentration of smallest bin
243    tr_tp1(ilon,ilev,k)=tr_t(ilon,ilev,k)/(1.+pdtcoag*denom)
244    ELSE
245!--calculating double sum terms in numerator of eq 14
246    num=0.0
247    DO j=1, k
248       numi=0.0
249       DO i=1, k-1
250!           
251!           see Jacobson: " In order to conserve volume and volume concentration (which
252!           coagulation physically does) while giving up some accuracy in number concentration"
253!
254!           Coagulation of i and j giving k
255!           with V(i) and then V(j) because it considers i,j and j,i with the double loop
256!
257!           BUT WHY WET VOLUME V(i) in old STRATAER? tracers are already dry aerosols and coagulation
258!           kernel beta(i,j) accounts for wet aerosols -> reply below
259!
260!             numi=numi+ff(i,j,k)*beta(i,j)*V(i)*tr_tp1(ilon,ilev,i)*tr_t(ilon,ilev,j)
261            numi=numi+ff(i,j,k)*beta(i,j)*Vdry(i)*tr_tp1(ilon,ilev,i)*tr_t(ilon,ilev,j)
262       ENDDO
263       num=num+numi
264    ENDDO
265
266!--calculate new concentration of other bins
267!      tr_tp1(ilon,ilev,k)=(V(k)*tr_t(ilon,ilev,k)+pdtcoag*num)/( (1.+pdtcoag*denom)*V(k) )
268    tr_tp1(ilon,ilev,k)=(Vdry(k)*tr_t(ilon,ilev,k)+pdtcoag*num)/( (1.+pdtcoag*denom)*Vdry(k) )
269!
270!       In constant composition (no dependency on aerosol size because no kelvin effect)
271!       V(l)= (f_r_wet(ilon,ilev)**3)*((mdw(l)/2.)**3) = (f_r_wet(ilon,ilev)**3)*Vdry(i)
272!       so numi and num are proportional (f_r_wet(ilon,ilev)**3)
273!       and so
274!        tr_tp1(ilon,ilev,k)=(V(k)*tr_t(ilon,ilev,k)+pdtcoag*num)/( (1.+pdtcoag*denom)*V(k) )
275!                     =(Vdry(k)*tr_t(ilon,ilev,k)+pdtcoag*num_dry)/( (1.+pdtcoag*denom)*Vdry(k) )
276!          with num_dry=...beta(i,j)*Vdry(i)*....
277!       so in old STRATAER (.not.flag_new_strat_compo), it was correct
278  ENDIF
279
280  ENDDO !--end of loop k
281
282!--convert tracer concentration back from [number/m3] to [number/KgA] and write into tr_seri
283  DO i=1, nbtr_bin
284     tr_seri(ilon,ilev,i+nbtr_sulgas) = tr_tp1(ilon,ilev,i) / zrho
285  ENDDO
286
287  ENDIF ! IF IN STRATOSPHERE
288  ENDDO !--end of loop klev
289  ENDDO !--end of loop klon
290! *********************************************
291
292END SUBROUTINE COAGULATE
Note: See TracBrowser for help on using the repository browser.