|
. V' L& D. _! v, @8 c
2021.10.24更新:两个小Tips:1️⃣这个包我现在不用了,毕竟主力平台还是Win,现在推荐使用ProPlot来画图(ProPlot — ProPlot documentation),真的很好用(导师看了都直呼内行,开个玩笑)!2️⃣文中提到的EOF的代码就别看了(属于是早年的腊鸡代码),现在推荐直接使用eofs(Package description — eofs 1.4.0 documentation (ajdawson.github.io))来完成EOF的计算。
2 c# I) T$ |7 g z* ]9 h. ]+ D 2021.01.31更新:
; C3 \& l! y2 \4 e2 s6 { 创建新环境时请将Python版本指定为3.6,否则会冲突导致无法安装。 / w% e3 `+ [4 t
一点感想:PyNgl和PyNio也不再更新了,美帝又去更新自己的Geocat包了,可惜啊Geocat我自己初次尝试,连安装都很大困难,还是学学人家cartopy吧,要不是cartopy画图太丑。 ( ?0 Q5 r& B. G6 W" H. O
更新:
) B2 `: [( v* x9 @4 [1 ` conda create --name pyncl --channel conda-forge/label/cf201901 pynio pyngl netcdf4 xarray ncl scipy 1 M5 q. t) {% s& g2 O9 g
此安装代码不建议在使用,速度慢,容易掉线。
, }. o4 X' }$ r# |* _* n: s 输入命令:vi ~/.condarc
3 B. g% `) [! A1 I9 { 讲最后一行的 -default 直接删除(前提是已经增加了清华或中科大源) 0 ?- ]+ [+ u, M# x- X
然后在新环境下使用conda install xxx 依次安装pynio pyngl netcdf4 xarray ncl scipy即可快速安装 3 _) p6 G% ?, z: `4 p7 r$ s
如果换源之后还是不行
4 l6 ?3 D, E% p+ n 数据可视化的软件和库有很多,但是对于大气和海洋数据的可视化,因为涉及到投影、海岸线的绘制,支持的还非常少。 1 v; a) G8 Y* N Y0 h
主要的有: ' m9 P' s' [3 \& e# t, O/ N
①Grads:易用性很差,常常是数据处理计算出来再由Grads读取来绘制。同时语法也比较奇怪。对Win10的支持也不是很好。 0 X: ] ~ G' B3 A7 L) g- \/ I
②NCL:NCL(The NCAR Command Language)是一种专门为大气海洋科学数据处理以及数据可视化设计的高级语言。其实是PyNgl和PyNio的前代产品。
0 g4 ]: E( U1 j3 E ③MATLAB:很好用但是主攻方向不在地球科学,绘制出来的图片稍微欠缺没敢。(个人感觉)
" Y! v0 t2 `2 p ④Cartopy:Python下的一个库,由英国气象局开发,Basemap的继承产品(Basemap在2020年不再维护),主要基于matplotlib实现可视化。目前还不是很完善,容易出现奇奇怪怪的问题。
. Q/ E0 f$ ]0 P. |* K! w ⑤PyNgl:NCL的续作由NCAR负责开发。绘制美观,功能强大,绘图时允许高度自定义。放一张官网扒的图 " X/ v3 f. U2 n3 c
; a$ Z, F- C6 I: u% V {: d# |" Q
还是挺美观的。 ( \- Z3 d T) [4 f% u
为什么选择Python?因为 人生苦短,我用Python!!! 它语法简单高效,一般来说上手三天即可(但是精通永无止境),学习曲线很平滑。
- U5 ]# S+ y9 t' ~" J/ X; ] b 既然PyNgl是Python下面的一个库,那么直接用conda安装不就完了吗,理论上来说是的,但是关键在于: . ?, V+ Z+ c* X) f5 I F2 y# B
它不支持Win10!!! # M( c+ G g+ v1 B% E/ U7 H" x
它不支持Win10!!! & Q# ]2 c4 \* E9 G- {
它不支持Win10!!! / L; s9 S& b9 n5 q6 z
这对于不常用Linux的科学工作者或者学生来说是很麻烦的事情,大多数人的主力系统都是Win10,当然macos系统也不需要参考此教程,可以直接利用conda安装。
! _; P8 [+ g5 j+ I$ [+ h4 } 幸运的是微软家有一个很好用的东西:子系统(WSL),Windows Subsystem for Linux(简称 WSL)是一个为在 Windows 10 上能够原生运行 Linux 二进制可执行文件(ELF 格式)的兼容层。它是由微软与 Canonical 公司合作开发,目标是使纯正的 Ubuntu 14.04 “Trusty Tahr”映像能下载和解压到用户的本地计算机,并且映像内的工具和实用工具能在此子系统上原生运行。
: C7 M4 m; n- n5 k 相比于 VMware 等虚拟机,WSL 占用内存和 CPU 资源更少,在 WSL 上运行软件的消耗和直接在 Windows 上差不多。而且,Windows 下可以直接访问 WSL 的环境。 ; e$ _ x6 p. @% _, a0 l& [7 A
安装子系统步骤:
) N3 u- `+ O! A$ M ①开启 WSL 可选特性
( v- M% E4 I$ Q# u9 E! M3 @ 在控制面板的“启动或关闭 Windows 功能”中勾选“适用于 Linux 的 Windows 子系统”。 9 C3 V/ Y l8 j+ P" j: k, \% u
5 B" u. v+ ?- A, d/ z
或在 PowerShell 中运行下述代码: & Q4 T Q" i0 U8 j
Enable-WindowsOptionalFeature -Online -FeatureName Microsoft-Windows-Subsystem-Linux
$ t C- h+ e& y6 l- ]0 C$ D& h ②下载安装
7 L# {" U) f2 |4 |3 o8 z 打开 Microsoft Store,搜索 Linux,就会显示 Ubuntu、suse 等几个发行版,点击进行安装即可。这里选择了 Ubuntu(第一个)。
! n! K4 f! Y! Y# k
$ X+ L% G- _# z" ]$ R ^+ K9 J 下载之后启动菜单里就会出现Ubuntu的图标了。让我们启动它,按照上面的提示等待几分钟,就可以进入初次登陆设置账号的界面。
, q k) |; {+ q0 S* I( q) P6 Z+ n 图片来自https://wu-kan.cn/_posts/2018-12-14-Windows-Subsystem-for-Linux/设置账号密码就完成了安装
2 V# d7 Y4 v H c: A0 M 图片来自https://wu-kan.cn/_posts/2018-12-14-Windows-Subsystem-for-Linux/关于Linux的命令可以自行查阅,只需要会cd,ls,pwd等简单命令即可。 % O3 x4 h+ o- |6 {
需要注意:如无必要不要输入下面命令,因为更新后貌似会有些问题,导致PyNgl无法使用。 1 X) F9 ]2 k- e$ {& `
sudo apt update -y
0 ]+ }8 ]! y; u0 i- q% j sudo apt upgrade -y
1 A, \% ^- _% R9 J3 y 子系统下anaconda的安装:
- B: U1 L3 I T4 K p ①首先从清华镜像Tsinghua Open Source Mirror下载ananconda在linux下的镜像,并放到Win10系统的任意位置(这里假设是D盘) & b A( S7 q0 L
# V' f5 f$ n8 v: I
接着在子系统中输入下面命令
9 T7 e4 o, J% @9 U3 W bash /mnt/d/Anaconda3-2020.02-Linux-x86_64.sh#mnt其实是挂载到了本地的硬盘
! @: @7 k7 I& B+ j$ ? 一路回车,注意在正式安装前会弹出安装地址,记下这个位置。然后他会开始安装,等就行。 % o9 Y0 U: v A- Y4 P/ P6 L" R
②环境变量的设置(保证在子系统的任意位置均可使用Python和conda命令) ' X( O7 e1 H: c% }5 U4 S
1,输入 vi ~/.bashrc (打开了一个编辑器) 6 L0 Z! f5 @5 [, f4 H
2,按一下i键,进入编辑模式
j9 @; b* e$ ^+ i) R' q 3,在最后输入export PATH=/home/lishanliao/anaconda3/bin PATH 要换成你自己的目录,如果你是一路回车安装的一般只要把lishanliao换成你自己的用户名即可
! C* E7 D, S% o' D* Z 4,按下shift+ i旁边那个冒号键),输入wq 保存并退出。
6 Y' K& r1 t2 W! i 5,输入source ~/.bashrc 重新加载环境变量
1 d+ w) D# _7 ]) ~, f$ M$ f+ c 6,如果你害怕不会用这个编辑器,可以直接在命令行中输入下面命令,在执行5即可 0 r+ Z% }) {$ j# }/ m
echo export PATH ="/home/liang_yu/anaconda3/bin  PATH" >> ~/.bashrc#还是要换成你自己的路径
& T g3 ]9 j' }( D+ k J3 K ③conda下载源更换(加快库的下载速度) 7 X9 c. q* p% z/ t! [
直接在linux命令行中输入下面命令(复制粘贴即可)
( ~* O1 U6 O0 J# K+ y conda config --add channels https://mirrors.ustc.edu.cn/anaconda/pkgs/main/
/ I0 S6 a- N1 ^) v" p conda config --add channels https://mirrors.ustc.edu.cn/anaconda/pkgs/free/5 |* W8 F6 ^) X7 d# ]
conda config --add channels https://mirrors.ustc.edu.cn/anaconda/cloud/conda-forge/
: m+ E7 G" {# v* ~3 t* M* C conda config --add channels https://mirrors.ustc.edu.cn/anaconda/cloud/msys2/
* _2 L+ w$ r7 S$ p1 M$ o" q K conda config --add channels https://mirrors.ustc.edu.cn/anaconda/cloud/bioconda/$ i8 g$ \" ~' v/ v9 Y5 y' j
conda config --add channels https://mirrors.ustc.edu.cn/anaconda/cloud/menpo/
( S/ \# [$ o, h' ~/ R7 K conda config --set show_channel_urls yes
. y0 X/ V* W% z ④PyNgl的安装
6 k) d3 A2 K5 N! p, p' H 在linux命令行中输入下面命令,他会自行安装,可能会出现网络错误,要么多试几遍要么挂梯子
: y/ K# W* j. ]' R5 Z! g( \7 ]3 B conda create --name pyncl --channel conda-forge/label/cf201901 pynio pyngl netcdf4 xarray ncl scipy
. ?8 e+ K% ]5 t/ e$ p: U3 ~' k 我并没把库安装在base环境下,而是创建了一个新的环境叫pyncl,因为好像有某些冲突。 1 n$ o" `$ l `# T1 m7 ^- f- ~4 Y P# Y
安装完成后,输入python,在输入 import Ngl,如果没有报错就是成功了。
/ n. N4 n$ U) B& G8 E) k1 ^) g ⑤接下来就可以画图啦 ' s$ @9 k3 G5 {0 r
现在win10下写好代码,保存成.py文件
. X+ B4 G* g9 k4 ^* U [- G 首先激活环境,输入下面命令
+ g5 L: S: m7 A% e% E$ D) z conda activate pyncl % c% d1 ^9 ~. ~! c
接着输入,它会自动执行。
3 z# f( g& M* o* H% m0 T python /mnt/d/file.py " x. X: b) C6 ~: c, c: i8 T6 }% J
最后展示一下示例图和代码
8 @" ^* `7 F7 q4 q9 K 2 j. X: l6 {. B1 T" [, X# B
代码示例(前半部分是EOF的计算,可以不看) 2 d& | _! d+ h
"""实现了自由调整子图的位置,没有使用panel的面板绘图,使用了新的地图投影"""+ n9 t, [" v! I1 g7 ~
import Ngl
! ~ R2 I$ Z2 ^% p. p import numpy as np
4 B" H2 ~7 A; [% i6 I; x import math& r K: s% W2 Y
import netCDF4 as nc
" E0 T- z8 d# k1 [. _# ?" c def EOF_3d(data_3d,lat,lon,K):
2 k+ o$ V: J! a' b& d, d8 ` """data_3d表示三维数据
* \% b% v, O% K' h& e 例如某一时间段,某一区域范围6 p, m7 V% I9 A' m
的SST数据,需先做拉直运算
2 g( [9 g' V4 X: x9 ]# z8 X- V! m, d K是int表示去前K个模态
9 U8 H! a9 r3 ^3 e& ?1 | 同时考虑到时空转换K不能大于n0 H+ ^. T3 K/ t+ E: i4 ^# k) a) W
因为L_Q的形状为[n,n],看表达式就可以看出来"""+ M$ G1 W% P8 V8 z) P
lat_n,lon_n,time=data_3d.shape
9 p5 H5 x5 u0 R% T m=lat_n*lon_n;n=time#m为个点数,n为观测次数: ]+ G( B0 ^9 d9 }
m=m-np.count_nonzero(data_3d[:,:,0]!=data_3d[:,:,0])
) m0 D1 _! i2 V: p8 v9 j7 b1 u+ Y0 U X=np.empty((m,n))7 \4 H4 ]" q/ g) _5 b6 _2 M3 @
k=0
! f& \3 X' h7 H7 E* c, @& d for i in range(lat_n):
) \$ y1 e* @1 Q) f$ E for j in range(lon_n):
' ?' D: W9 W: T/ Z3 d2 ]8 @8 { if (np.isnan(data_3d[i,j,0])):#考虑陆地( y$ w; x3 K9 u5 f
print (land)5 S3 g1 [/ W4 b& v
else:
( C3 Y% |9 G- A0 x; P% p* _3 i #考虑地球因素乘于cos(纬度)
: K2 ~$ o! S" N X[k,:]=data_3d[i,j,:]*(math.cos(math.pi*lat[i]/180))**0.5) t2 M p9 O8 M$ B ?# Z: ]( E# }
k=k+1
6 S) r" P4 J6 _5 \8 ^ X_mean=X.mean(axis=1).reshape((m,1))
) d* Q; V3 N8 G& e X_d=X-X_mean#利用距平阵计算$ @, B/ x" h' a! h
if (m<=n):#不使用时空转换
. k9 X+ \4 p7 a! J6 B S=np.dot(X_d,X_d.T)/n
. A U+ c' W+ |8 U lambda_,L=np.linalg.eig(S)
" u3 p2 f6 x! o( t else:
1 A7 D( t- U! n6 \0 S, T8 Q S_Q=np.dot(X_d.T,X_d)/m: z7 m4 g5 D# ?% |4 _1 R% m6 S
lambda_Q,L_Q=np.linalg.eig(S_Q)#注意L_Q的形状8 D9 n; g7 [* s5 \* m8 Z
lambda_=lambda_Q*(m/n)
9 P8 }# s3 |! @- A L=np.empty((m,K))5 E6 t: v- S N* Y; M
for i in range(K):6 ~+ _2 `* X- ^0 u/ H7 P, m
l_Q=L_Q.real[:,i]8 S" c6 B% J6 s: T4 \3 C& \
L[:,i]=np.dot(X_d,l_Q.T)/(m*lambda_Q[i])**0.5. z& ~& P; R! }' j2 E* u3 Y
lambda_=lambda_.real[0:K];L=L.real[:,0:K]#避免复数的问题
% F! b. F& k# e0 C2 { Y=np.dot(L.T,X_d)8 w- V- F3 m9 }& E9 \
p=np.empty((K,n))" T8 @6 \$ Y5 x5 [
e=np.empty((m,K))( U: C( I" R; w; D3 ^2 m1 d5 q( ~' O
for i in range(K):
" r7 }0 d& z. l1 B' N/ h4 h e[:,i]=lambda_[i]**0.5*L[:,i]1 n" n6 Z u( Y! @9 {* A
p[i,:]=Y[i,:]/lambda_[i]**0.55 j( X. I' A7 M2 M5 h; o( a" j
#将e恢复为空间性
# F3 ^. V! T, [; m" I* |# k e_2d=np.empty((lat_n,lon_n,K))
' G9 n& b+ X7 P8 G for m in range(K):
4 n Y. r) \, y k=0
7 P2 T; z$ v2 G4 F8 E& h* X3 G- o for i in range(lat_n):
2 ?2 d; E% j$ j for j in range(lon_n):
% \1 t* h4 A( a( b+ { if (np.isnan(data_3d[i,j,0])):4 W7 t" d% ^) L/ O# A
e_2d[i,j,m]=np.nan; ~% J( L4 {. y% P& J% l/ d' y# p4 b
else:
" P+ }" j4 W- N: s. \# A, d e_2d[i,j,m]=e[k,m]/(math.cos(math.pi*lat[i]/180))**0.54 M' d0 {) r6 m0 z' C( i
k=k+1
+ a5 L S! b |5 ^+ j% A1 G2 S& ` return lambda_,e_2d,p,Y: f" Y$ g1 T3 n* f4 d
#------------------4 K" i/ r. _1 @' X
#时间:2000-2018年每年1月(624开始19个月)8 b" R" }; p; P6 k' C
#区域:0-85,120-245
: D" Y O" k/ {8 [7 k #lat[2:37] 35, h+ M$ d! D% t( I' b; r% ^6 s" s
#lon[48:99] 51: O$ J% E9 P E, R4 ]# n
file="/mnt/c/Users/59799/Desktop/slp.mon.mean.nc"0 t$ x8 L, c! L: ^" m
#file=rC:\Users\59799\Desktop\slp.mon.mean.nc
% Y9 A7 Q* i- g2 b data = nc.Dataset(file)
+ X3 I( C7 z C, Q0 h# u. p+ ?! m lat = np.array(data.variables[lat])* i: O Q7 c1 i0 L
lon = np.array(data.variables[lon])9 N; H, M, n1 K5 g: f& z' Y
time = np.array(data.variables[time])
8 @5 O! r: n- [/ u$ K5 G# r/ _ w slp = np.array(data.variables[slp])8 |& D1 g9 c5 g1 y8 L, F
data_3d=np.empty((35,51,19))
+ f8 M( ~/ \. ^8 \# N4 f0 m5 \% b% f time=6240 z; E) d, ?8 g/ g( T! i1 g& \9 q
for k in range(19):
) ^1 ~, q/ u$ H! M4 f* R for i in range(35):5 u/ ?* t) Y- r, A$ v
for j in range(51):0 D6 I% Z3 V. A6 D( a& ]
data_3d[i,j,k]=slp[time,2+i,48+j]
9 T# w$ d. j! `0 d X6 f/ f time=time+12# G4 G3 U! s) O% h8 U
lad,e_2d,p_t,y=EOF_3d(data_3d,lat[2:37],lon[48:99],10)
+ @8 I+ P- e3 O: u
, }5 q* E, k6 @3 _2 T #绘图- C5 l) }% W& G
#BlueWhiteOrangeRed1 U- u- \7 s% [, h* q
plot=[]
& `$ X0 f( H! P- D5 j! ]/ R wks_type = "png"#子系统没有可视化模块,输出成png图片在打开来看
* j# z ?8 u7 ]1 ^, ?/ ] wks = Ngl.open_wks(wks_type,"/mnt/c/Users/59799/Desktop/EOF_first_model")#图片保存到本地的桌面
8 G3 N; Y1 M. s; ` cnres = Ngl.Resources()
9 ^) e) H/ `- t% @ w cnres = Ngl.Resources()+ x5 c6 j9 ^3 P# Y i
cnres.nglDraw = False#因为要绘制叠加图先阻止画图
9 B4 C" b* N0 q, V# { cnres.nglFrame = False#关闭框架与上一条搭配使用9 A. |4 e1 r& L$ ?" l0 g6 |
cmap = Ngl.read_colormap_file("BlueWhiteOrangeRed")#读取colorbar,官网可查啊4 K% `6 S: I. Y5 ?% K/ Y
cnres.cnFillPalette = cmap[:,:]#不完整使用整个色图,只是用颜色定义值的第五行开始
6 C3 ~% g$ _. X. O cnres.cnFillOn = True #等值线填充. C. \: O" \7 B' T
cnres.cnLinesOn = False#不画等值线
C" |) a9 V2 I" a% [ cnres.cnLineLabelsOn = False#不画等值线标签(等值线数值)
( J5 `2 V; @5 u& |$ j r8 b cnres.lbOrientation = "horizontal"#水平放置colorbar' a7 e! `# i: u0 ?" e4 J7 X4 s
cnres.mpProjection = "Orthographic"
( I }0 r/ e6 w" }6 x3 `( y' K; O cnres.sfXArray = lon[48:99]#经度对应为x轴
" \3 E* B8 q- R F3 b) y l% F/ s* R cnres.sfYArray = lat[2:37]#纬度对应为y轴
4 H M. H& @8 E1 I) B cnres.mpCenterLonF = lon[48:99].mean()) {! Y3 d* L% t" K9 ~( v* F
cnres.mpCenterLatF = lat[2:37].mean()5 w( b& X; T9 l3 m$ \# r+ g. e1 C
#cnres.mpLimitMode = "LatLon"#规定经纬度范围时需要使用,否则规定无效
3 S9 A2 a* M; Y+ Q0 Y2 ` #cnres.mpMinLonF = lon[48:99].min()#不加范围限制则图片为所规定中心纬度经度显示的全球视角(数据只在规定范围有的话会出现无数据的空白区域(海岸线依旧存在)) W! K) E" [& V' z+ @4 \2 M
#cnres.mpMaxLonF = lon[48:99].max()#不加范围限制则图片为所规定中心纬度经度显示的全球视角(数据只在规定范围有的话会出现无数据的空白区域)
% g( i7 f& F! O0 l #cnres.mpMinLatF = lat[2:37].min()#不加范围限制则图片为所规定中心纬度经度显示的全球视角(数据只在规定范围有的话会出现无数据的空白区域)
% v5 N0 l3 z3 I0 }: Q6 g9 ~2 R #cnres.mpMaxLatF = lat[2:37].max()#不加范围限制则图片为所规定中心纬度经度显示的全球视角(数据只在规定范围有的话会出现无数据的空白区域)
3 i$ Z9 v& C, G1 `5 U( H$ @ cnres.nglMaximize = False#必须关闭最大化绘图不然下面的调整plot box的参数是无效的
|2 A3 S. `$ e) H7 ^9 v+ v3 C cnres.vpXF = 0.08 # The left side of the box
; Z; u! D. l9 E3 T) B cnres.vpYF = 0.75 # The top side of the plot box
9 A' Z# N1 b" c ?: o cnres.vpWidthF = 0.4 # The Width of the plot box& P" |4 b q" |0 T
cnres.vpHeightF = 0.5 # The height of the plot box
) e, v6 Z9 Z. J$ c cnres.mpGeophysicalLineThicknessF =3.3 #海岸线宽度(粗细程度) 默认的太小了7 p# t i( O* T( r, P5 J
cnres.mpFillOn = False#不开启地图填充,关于陆地海洋内陆水的填充自然不需要3 O% ]; U9 n% @, F
cnres.tiMainString = "e_first_model"
" E+ d3 R. P* M p1= Ngl.contour_map(wks,e_2d[:,:,0],cnres)# c: L1 n9 ^! e4 B
/ Z5 K& }# e1 z xyres = Ngl.Resources()# N9 N, W' B8 J* i: {
xyres.nglDraw = False#因为要绘制叠加图先阻止画图$ _; [6 I6 ~, u" k% a
xyres.nglFrame = False#关闭框架与上一条搭配使用
5 z! i, O5 u" U xyres.nglMaximize = False#必须关闭最大化绘图不然下面的调整plot box的参数是无效的
% Z O4 } b) C3 T$ N6 |" y xyres.vpXF = 0.59 # The left side of the box; p2 N* X @4 E! O" |
xyres.vpYF = 0.645 # The top side of the plot box
# [# {3 ]8 G! d4 T1 I' d; | xyres.vpWidthF = 0.4 # The Width of the plot box9 ?0 j/ P2 H% H) N2 z
xyres.vpHeightF = 0.32 # The height of the plot box
) x0 u6 j) g) h6 _* N( B% D# B1 p xyres.tiXAxisString = "Jan of Each Year", H0 Z7 L' l5 E" F# z% l3 z
xyres.tmXBMode = "Explicit" # Define your own tick mark labels.
4 E; T8 H0 x/ v" L& W xyres.tmXBValues = list(range(1,20,1))# Values from 1 to 19.
+ }: c5 M2 h' v8 c xyres.tmXBLabels = ["00","01","02","03","04","05","06","07","08","09","10","11","12","13","14","15","16","17","18"]
. X: n8 c' u- Q% Y* G$ ?' j: t6 e xyres.tiMainString = "p_first_model"5 i3 U) y6 x+ J4 V" c8 a ~2 L
x=np.arange(1,21,1)
! N1 g# d' \; v& p) ^$ W% X p2 = Ngl.xy(wks,x,p_t[0],xyres)
2 g2 B3 h$ x( ]
& ]' U, V* q4 d
M: A# _7 a2 \& I Ngl.draw(p1)
& O4 t3 ]' w6 { Ngl.draw(p2)
, y/ @- F+ L7 P H: d' o: ^ S+ H Ngl.frame(wks)
2 P- M( o, q: \6 q9 ]0 n& i- U9 F Ngl.end()
4 c* F' s, x, ]
: C! J x* n/ V! o9 K- }
- ^) b' L9 x9 |4 [/ m( U& [, k% D; x& O7 n$ |
9 G; g3 }3 Z0 c" ` |