马上加入,结交更多好友,共享更多资料,让你轻松玩转电力研学社区!
您需要 登录 才可以下载或查看,没有账号?立即加入
×
s函数是system Function的简称,用它来写自己的simulink模块。(够简单吧,^_^,详细的概念介绍大伙看帮助吧)可以用MATLAB、C、C++、Fortran、Ada等语言来写,这儿我只介绍怎样用matlab语言来写吧(主要是它比较简单)。 先讲讲为什么要用s函数,我觉得用s函数可以利用matlab的丰富资源,而不仅仅局限于simulink提供的模块,而用c或c++等语言写的s函数还可以实现对硬件端口的操作,还可以操作windowsAPI等。 先介绍一下simulink的仿真过程(以便理解s函数),simulink的仿真有两个阶段:一个为初始化,这个阶段主要是设置一些参数,像系统的输入输出个数、状态初值、采样时间等;第二个阶段就是运行阶段,这个阶段里要进行计算输出、更新离散状态、计算连续状态等等,这个阶段需要反复运行,直至结束。 在matlab的workspace里打edit sfuntmpl(这是matlab自己提供的s函数模板),我们看它来具体分析s函数的结构。它的第一行是这样的: function [sys,x0,str,ts]=sfuntmpl(t,x,u,flag) 先讲输入与输出变量的含义:t是采样时间,x是状态变量,u是输入(是做成simulink模块的输入),flag是仿真过程中的状态标志(以它来判断当前是初始化还是运行等);sys输出根据flag的不同而不同(下面将结合flag来讲sys的含义),x0是状态变量的初始值,str是保留参数(mathworks公司还没想好该怎么用它,嘻嘻,一般在初始化中将它置空就可以了,str=[]),ts是一个1×2的向量,ts(1)是采样周期,ts(2)是偏移量。 下面结合sfuntmpl.m中的代码来讲具体的结构: switch flag,%判断flag,看当前处于哪个状态( D% j' P: L/ j- i& A4 g
8 a* N4 }# p8 {! x2 E7 a
case 0, [sys,x0,str,ts]=mdlInitializeSizes; flag=0表示处于初始化状态,此时用函数mdlInitializeSizes进行初始化,此函数在 sfuntmpl.m的149行
5 A6 z; r& w0 b. X% W) G
9 a) W' F8 r" ]/ C4 ?3 x4 s我们找到他,在初始化状态下,sys是一个结构体,用它来设置模块的一些参数,各个参数详细说明如下0 a4 U: L& l$ }# q8 [
/ x5 Z0 b7 Q& Q7 _* Y- a+ ?0 b% ~
size=simsizes;%用于设置模块参数的结构体用simsizes来生成" Q1 U9 E% ?. \* t
5 V' k: J6 i4 M. V! xsizes.NumContStates=0;%模块连续状态变量的个数
1 ]! b# Q. _/ B0 z( \9 [5 J) N
; h f' U, m4 ~- Bsizes.NumDiscStates=0;%模块离散状态变量的个数( G' q, `$ I h4 K: z
. p& t$ p- A! T, ~# i6 t/ q
sizes.NumOutputs=0;%模块输出变量的个数+ ], c, a# ?; y C+ s, q3 y: O, j
! p% |* w" h" P
sizes.NumInputs=0;%模块输入变量的个数. }9 l0 V/ G$ k0 q8 O4 J
" S+ e: a# b9 }" g
sizes.DirFeedthrough=1;%模块是否存在直接贯通(直接贯通我的理解是输入能直接控制输出)# t8 L1 P/ F* o! y* m
) Q' M* R* c: F
sizes.NumSampleTimes=1;%模块的采样时间个数,至少是一个5 q; M" v5 j; M3 X* q; w, n
7 O9 k$ `: N% G9 \sys = simsizes(sizes); %设置完后赋给sys输出4 X& l L1 S! _6 F
: v, \7 A7 @1 L2 C举个例子,考虑如下模型:5 ^2 m' U9 H3 q9 U8 V) k" N" w
. c$ B& G7 ?3 V' a dx/dt=fc(t,x,u) 也可以用连续状态方程描述:dx/dt=A*x+B*u* s1 ^2 r/ f% O
' ?* S! e" W' l+ ^8 S' P/ m x(k+1)=fd(t,x,u)也可以用离散状态方程描述:x(k+1)=H*x(k)+G*u(k)
1 u1 _- a$ r1 ^! {, G+ C7 g) z# ?1 b
y=fo(t,x,u)也可以用输出状态方程描述:y=C*x+D*u; r+ u) S+ O2 [' ^# x( N3 X2 w% q
/ i* a( s4 A# {) [- a% r设上述模型连续状态变量、离散状态变量、输入变量、输出变量均为1个,我们就只需改上面那一段代码为:" r/ n/ T0 s9 d& k) s3 d: f7 t% {( v; Z
; K8 _9 @7 ~9 m* O/ o. Q j( R4 x(一般连续状态与离散状态不会一块用,我这儿是为了方便说明)) S1 c: m; V& s5 ?
0 M c; q7 \6 D( Q" y+ o4 r- O$ Dsizes.NumContStates=1; sizes.NumDiscStates=1; sizes.NumOutputs=1; sizes.NumInputs=1; ts=1; 其他的可以不变。继续在mdlInitializeSizes函数中往下看: x0 = []; %状态变量设置为空,表示没有状态变量,以我们上面的假设,可改 %为x0=[0,0](离散和连续的状态变量我们都设它初值为0)$ K* p6 Y. _" I; r" u+ J3 R
$ e+ E4 k3 R4 ]9 r0 Rstr = []; %这个就不用说了,保留参数嘛,置[]就可以了,反正没什么用,可能7.0会给它一些意义
3 G" T8 ?2 ~% Q7 w0 D% g% Z+ V0 |% ~6 ]+ B0 p |
ts = [0 0]; %采样周期设为0表示是连续系统,如果是离散系统在下面的mdlGet%TimeOfNextVarHit函数中具体介绍: y& N. X4 _ x) ]8 n
; U) D2 \2 e1 |' Q& y
嘻嘻,总算讲完了初始化,后面的应该快了. x0 i" w9 Y. J/ r
0 ~5 K; k6 M& Y g b) N
在sfuntmpl的106行继续往下看:6 l( s& |9 V1 v5 z# @6 ` g2 S) ?& Y
1 `8 r( r, E) y9 Lcase 1,$ {2 b4 B9 e0 t- J! I; @
& f( j8 N o7 i$ J7 F Z1 ksys=mdlDerivatives(t,x,u);
7 Y# b; @9 ~( m& [1 D1 ?/ _+ @5 G4 Q8 ^& N6 `* R7 K' Y
flag=1表示此时要计算连续状态的微分,即上面提到的dx/dt=fc(t,x,u)中的dx/dt,找到mdlDerivatives函数(在193行)如果设置连续状态变量个数为0,此处只需sys=[]; 就可以了(如sfuntmpl中一样),按我们上述讨论的那个模型,此处改成 sys=fc(t,x(1),u)或sys=A*x(1)+B*u %我们这儿x(1)是连续状态变量,而x(2)是离散的,这儿只用到连续的,此时的输出sys就是微分
# S0 ]0 P' ~9 [3 V' h& R( R) N# _1 n$ Q: x6 F3 h, T& t, c! s
继续,在sfuntmpl的112行:: b" O3 J/ h+ k) j! i/ H) R
8 C$ V& v0 \0 o5 C1 L
case 2,% @1 k: l! z# H( ?# P
! z& Z/ N) w, x: jsys=mdlUpdate(t,x,u);
4 O: z. W j' Q5 F7 j) ~
! Z5 y6 n4 g: M/ H. x) Qflag=2表示此时要计算下一个离散状态,即上面提到的x(k+1)=fd(t,x,u),找到mdlUpdate函数(在206行)它这儿sys=[];表示没有离散状态,我们这而可以改成sys=fd(t,x(2),u)或sys=H*x(2)+G*u;%sys即为x(k+1)看来后面几个一两句话就可了,呵呵,在sfuntmpl的118行3 N3 r7 U6 _6 q8 Y
4 l: D% A7 D+ U, L, [7 l9 q
case 3,
' _5 I0 @ w1 O) l+ g+ K9 O) n. S5 H, q+ \, C8 }
sys=mdlOutputs(t,x,u);4 a& R# A2 e; D6 T+ h& h
0 q: P) @7 @' X4 \0 uflag=3表示此时要计算输出,即y=fo(t,x,u),找到mdlOutputs函数(在218行),如上,如果sys=[]表示没有输出,我们改成sys=fo(t,x,u)或sys=C*x+D*u %sys此时为输出y
; q# A. r2 j+ }- v0 M" ~% g- G
/ l3 Q# S! h* q# N" l好像快完了,嘻嘻,在sfuntmpl的124行1 u O& ?# c5 n. x0 Q3 l7 n
( H2 F2 r" i" q# H ^$ J* ~+ hcase 4, X# S8 l' [6 U( R. m
. p! f2 B! Q e: b7 a
sys=mdlGetTimeOfNextVarHit(t,x,u);7 ^7 M3 s6 h8 ]" d I* c5 ]
0 e7 `4 ~! b0 e) z8 A0 F+ U6 Mflag=4表示此时要计算下一次采样的时间,只在离散采样系统中有用(即上文的mdlInit ializeSizes中提到的ts设置ts(1)不为0)
& v' l/ Y+ u0 s2 V! j6 }% J A9 O) y. ^# w2 m3 W# e# A% }3 q- F
连续系统中只需在mdlGetTimeOfNextVarHit函数中写上sys=[];这个函数主要用于变步长的设置,具体实现大家可以用editvsfunc看vsfunc.m这个例子
+ i$ l9 d' |# e: d
$ y3 j- I1 }' Q7 h最后一个,在sfuntmpl的130行
5 C% t. f: c$ T# D# p
% E& f( u4 s: ~case 9,! @1 E# _, ]! T- b1 H! l8 B2 v
0 ~, X& E( q0 g# G
sys=mdlTerminate(t,x,u);2 a2 g* |0 t3 u" x, R
4 K6 ~' D) P4 |; }. S; m& wflag=9表示此时系统要结束,一般来说写上在mdlTerminate函数中写上sys=[]就可,如果你在结束时还要设置什么,就在此函数中写( \/ Q# O1 @+ w8 R. e
6 j) V% h0 S( o8 D关于sfuntmpl这个s函数的模板讲完了。 s函数还可以带用户参数,下面给个例子,和simulink下的gain模块功能一样,大伙自己看吧,我睡觉去了,累了2 ^$ r8 }$ J& a1 w
; }( s6 M4 P4 {* C$ c% l: ]function [sys,x0,str,ts] = sfungain(t,x,u,flag,gain)
/ a8 {% y: W3 T0 X5 f- y) Q
0 J9 a: W) P' t. Y3 y1 a4 H9 \switch flag,
$ D2 ~( \0 N% Z0 L; M6 F: W* Z7 F
# f/ h8 F1 c+ v8 z0 _- P; H: d" s- i3 _case 0,& \. i9 ~- a7 I" T8 G j0 s
+ {% [* V# Y) n' ^ sizes = simsizes;1 A- D0 a- a- L7 `
: X) M4 }0 z/ D. P1 ^
sizes.NumContStates = 0;
: N4 |& N" b2 d+ o+ P
) S& @% t; U, l V3 P& V sizes.NumDiscStates = 0;( M( G5 A v$ G, E3 O
* I0 d2 P& m3 O9 ~
sizes.NumOutputs = 1;/ h6 `( m+ b9 z L
5 s2 j0 ~. R$ u8 r
sizes.NumInputs = 1;
/ i q8 V4 @- w4 ~. {5 e* q
8 d+ G, E' o, w x% q% a sizes.DirFeedthrough = 1;! ?9 `$ e! ~" S9 r0 v v" }
; Z. M- O, S, B' g sizes.NumSampleTimes = 1;
/ w( ?: B# b3 R6 h! `& F9 w; c+ M! h, R
sys = simsizes(sizes);
. a! {; I0 t- x1 m
- u* c% x- f. |& l) ?/ R& [& ~ x0=[];
/ P/ \/ V+ V. e* N1 ]$ L* p/ B7 Q7 J3 B; `
str=[];
. D1 a% h% q5 l1 w
& P2 t( j1 c* Y, k; a8 Y: i ts=[0,0];5 ]6 \& z/ _) s( e
, o6 X/ p; z" d: e9 @+ P
case 3,
- [# j! a0 K2 J, j% g6 _! n. q
2 H9 J: C# B) l sys=gain*u;$ |* l9 V3 {% a& M$ Y; j. O
: [6 m4 q4 I% C8 r3 `case {1,2,4,9},- { L. H2 ]( M- z
5 w9 s, Q; @7 M/ e$ W" l$ _) ^& isys = [];
" F# H3 z+ ?* | {' f! ]+ T) J% E6 H% B+ f( |5 j
end
1 n# q! S$ X! `8 T
3 F& q, X7 y$ d' Z L6 R做好了s函数后,simulink--user-definedfunction下拖一个S-Function到你的模型,就可以用了。在simulink----user-definedfunction还有个s-FunctionBuilder,他可以生成用c语/ d! i% X2 r7 n$ n" F5 O1 n, @
言写的s函数
' ^" |) c( K" P在matlab的workspace下打sfundemos,可以看到很多演示s函数的程序。 SIMULINK s-function的设计
) }3 h8 I/ O8 J1 {$ s6 t+ j4 ?) G/ p( s4 G4 k- S$ E
Simulink为用户提供了许多内置的基本库模块,通过这些模块进行连接而构成系统的模型。对于那些经常使用的模块进行组合并封装可以构建出重复使用的新模块,但它依然是基于Simulink原来提供的内置模块。
/ G7 }7 k' R& C5 I7 F" u% g6 {8 _# }2 k8 o" T- F z2 I6 t' ^
而Simulink s-function是一种强大的对模块库进行扩展的新工具。
( \" V0 o* ^8 i4 K, `5 V. r' }6 s3 _( ^
(一)、s-function的概念
' l% }, E9 n4 I* l3 ?" ~" Y
& g( ^0 _) ~( us-function是一个动态系统的计算机语言描述,在MATLAB里,用户可以选择用m文件编写,也可以用c或mex文件编写,在这里只给大家介绍如何用m文件编写s-function。; B1 C: O1 B! g. ^7 i q
& _7 g% o/ G+ l: g$ ~
S-function提供了扩展Simulink模块库的有力工具,它采用一种特定的调用语法,使函数和Simulink解法器进行交互。% S, l( t: ? e- {4 d
1 P, x. V# T" b2 ?1 J: Z; j' Y
S-function最广泛的用途是定制用户自己的Simulink模块。它的形式十分通用,能够支持连续系统、离散系统和混合系统
9 P+ m0 T/ X( o5 o
! M+ v6 }' d* M$ }+ j# n(二)、建立m文件s-function
3 x0 M& k1 Z4 J+ ]% d
. R+ m$ P8 J3 k6 J; x# f9 h 1、使用模板文件:sfuntmp1.m 格式: [sys,x0]=function(t,x,u,flag)
. p" W( L% N2 E4 }! A/ F2 f
7 T; k# Q5 \. s% X- D5 g) g- P# D" H该模板文件位于MATLAB根目录下toolbox/simulink/blocks目录下。* O7 A4 @% E, z) q
- ^2 \" z; W/ R. a1 |, m
模板文件里s-function的结构十分简单,它只为不同的flag的值指定要相应调用的m文件子函数。比如当flag=3时,即模块处于计算输出这个仿真阶段时,相应调用的子函数为sys=mdloutputs(t,x,u)。
! ^5 e, k: d5 I: y6 k9 t7 o2 c& ?: w4 E$ E C% ~
模板文件使用switch语句来完成这种指定,当然这种结构并不唯一,用户也可以使用if语句来完成同样的功能。而且在实际运用时,可以根据实际需要来去掉某些值,因为并不是每个模块都需要经过所有的子函数调用。
1 |, ]$ P. c. X f" B" I3 c3 l" B' D' A3 c8 b1 T% d
模板文件只是Simulink为方便用户而提供的一种参考格式,并不是编写s-function的语法要求,用户完全可以改变子函数的名称,或者直接把代码写在主函数里,但使用模板文件的好处是,比较方便,而且条理清晰。$ [4 Y! `$ D4 v+ K7 @' T
使用模板编写s-function,用户只需把s-函数名换成期望的函数名称,如果需要额外的输入参量,还需在输入参数列表的后面增加这些参数,因为前面的4个参数是simulink调用s-function时自动传入的。对于输出参数,最好不做修改。接下去的工作就是根据所编s-function要完成的任务,用相应的代码去替代模板里各个子函数的代码即可。
! u% T$ T/ P: w0 M8 s v s" y' F5 j& \* w' X
Simulink在每个仿真阶段都会对s-function进行调用,在调用时,Simulink会根据所处的仿真阶段为flag传入不同的值,而且还会为sys这个返回参数指定不同的角色,也就是说尽管是相同的sys变量,但在不同的仿真阶段其意义却不相同,这种变化由simulink自动完成。
9 W- E; s$ C9 L4 o1 x( x5 I. Z) i' d0 Z$ s, X. q
m文件s-function可用的子函数说明如下:
, j/ H" x0 L0 l) h# h, I0 u3 V1 @& D8 I+ e# l& q d
mdlInitializeSizes(flag=0):定义s-function模块的基本特性,包括采样时间、连续或者离散状态的初始条件和sizes数组。
) w# ]/ z% E& @+ J% X3 w# X" Z1 |3 T1 M% ^1 D6 y, J
mdlDerivatives(flag=1):计算连续状态变量的微分方程。( M' g3 r$ F4 ~1 @ n+ _) o& i9 A
4 i8 B6 I2 @% ?/ w- h( lmdlUpdate(flag=2):更新离散状态、采样时间和主时间步的要求。, k. v& k0 J# `- y/ V
6 V, I7 ^7 A5 g! m; L2 VmdlOutputs(flag=3):计算s-function的输出。) |3 @" z+ q7 K, q
# o& l5 i& I9 a. o$ p
mdlGetTimeOfNextVarHit(flag=4):计算下一个采样点的绝对时间,这个方法仅仅是在用户在mdlInitializeSizes 里说明了一个可变的离散采样时间。5 A! w% o$ I$ F; f7 T& u6 t; c
! p8 {* R( O/ z$ h d
概括说来,建立s-function可以分成两个分离的任务:
- r, d* u# }8 K6 q0 s# v0 J. v
( Y& F9 t/ Q; Z& d0 f6 S4 w4 ~! I( t初始化模块特性包括输入输出信号的宽度,离散连续状态的初始条件和采样时间。7 G' A( d! e, @& W# Q, K
8 @* O f: H* z* ]$ F1 G将算法放到合适的s-function子函数中去。
4 c" O& @1 a# n6 V) i J. o; d+ u/ [8 P
2 、定义s-function的初始信息( Z3 R8 g5 S$ f' ^. X0 o
& q7 W: [: |: q) S+ d6 G为了让Simulink识别出一个m文件s-function,用户必须在s-函数里提供有关s-函数的说明信息,包括采样时间、连续或者离散状态个数等初始条件。这一部分主要是在mdlInitializeSizes子函数里完成。 [' a( {6 C& b+ r; N
Sizes数组是s-function函数信息的载体,它内部的字段意义为:
: V0 d, c8 t% ]. `- J) ^! V+ t# d$ N0 _; b
NumContStates(sys(1)):连续状态的个数(状态向量连续部分的宽度)0 t1 H$ X( M* T0 h8 p# i
9 _: D) |: ?' f: KNumDiscStates(sys(2)):离散状态的个数(状态向量离散部分的宽度)
: h& a: U2 l; `9 M3 B2 o
; X4 r6 J8 G. p* H/ F& B$ iNumOutputs(sys(3)):
, I6 p# A/ m: O输出变量的个数(输出向量的宽度)
' v& y4 U" r. P
9 G1 Y4 K: I6 K; U. U! X+ ?NumInputs(sys(4)):输入变量的个数(输入向量的宽度)2 M# M/ J- S- I7 e
3 ^" f% O6 K2 H- Y, w; z0 @! w
DirFeedthrough(sys(5)):有不连续根的数量5 O/ J! d/ k0 G0 j& W _4 R
4 W- L. f# E* W8 d8 p2 s8 uNumSampleTimes(sys(6)):采样时间的个数,有无代数循环标志
: z( Y3 N2 r }7 Y5 M" g8 G3 l6 `4 i+ f% h% Q7 l
如果字段代表的向量宽度为动态可变,则可以将它们赋值为-1。
) Z9 L8 \' j6 C; h8 s; Y; I$ I
1 N! A) c2 {. E/ C6 x' h注意DirFeedthrough是一个布尔变量,它的取值只有0和1两种,0表示没有直接馈入,此时用户在编写mdlOutputs子函数时就要确保子函数的代码里不出现输入变量u;1表示有直接馈入。" j; ?. G) J5 C! ~
8 u0 S+ {% U0 e* C: R, ^6 ONumSampleTimes表示采样时间的个数,也就是ts变量的行数,与用户对ts的定义有关。# T' I$ b ?& ^) y) d& ]
9 r$ b, g/ ~- F% A& C }需要指出的是,由于s-function会忽略端口,所以当有多个输入变量或多个输出变量时,必须用mux模块或demux模块将多个单一输入合成一个复合输入向量或将一个复合输出向量分解为多个单一输出。
7 J4 C; k0 H/ a" W* P! H2 H, p M/ _2 X" R1 b
3、输入和输出参量说明: \2 j' D3 l8 y# ^- s1 c+ A
. M8 R: f' q* m7 x2 y$ R
S-function默认的4个输入参数为t、x、u和flag,它们的次序不能变动,代表的意义分别为:
; I! _. o" Q3 m* y* g3 Yt:代表当前的仿真时间,这个输入参数通常用于决定下一个采样时刻,或者在多采样速率系统中,用来区分不同的采样时刻点,并据此进行不同的处理。
- [0 h3 R. \7 y9 P& }9 |/ a9 G4 O0 m* B
x:表示状态向量,这个参数是必须的,甚至在系统中不存在状态时也是如此。它具有很灵活的运用。
; i' d( Z2 ?4 X5 W: f
" i, i" t0 V+ [0 X+ A' tu:表示输入向量。, m5 @. X& U6 L/ j4 { k+ I
0 ?/ Q4 d9 [) {, e3 F* w- A9 C
flag:是一个控制在每一个仿真阶段调用哪一个子函数的参数,由Simulink在调用时自动取值。
! G1 i2 G8 K/ [6 p% Y6 T( X) f% a2 b5 C- F
S-function默认的4个返回参数为sys、x0、它们的次序不能变动,代表的意义分别为:6 A( S) B1 g; }2 j) h: s
. o- o: K" @# v; m1 I3 f* p7 Isys:是一个通用的返回参数,它所返回值的意义取决于flag的值。& R/ Q% ]" W& }" A% f5 G
) m# J3 `2 z6 {' ` P8 ^' A( dx0:是初始的状态值(没有状态时是一个空矩阵[]),这个返回参数只在flag值为0时才有效,其他时候都会被忽略。/ `+ x" ?0 }( U- s% `* u
/ R; {: }: [) l" c) n一、有一系统如下:
! ?! K) b& y+ n- D& a' S4 }7 s0 T
dx1=x2
+ c9 ^% m4 A$ H
* w: e( ~! \. m: V s8 X; Zdx2=9.81*sin(x(1))-2*x(2)+u
# S6 W4 G% J5 i1 d- T. r$ ^) R
4 U- h2 B0 f/ K, Z6 }: L7 w5 L/ M, l. Z# J
求出系统在单位阶跃输入下的x1的状态变化曲线,假设x1,x2初值为0。( y9 w( ^( W6 N& b
5 X& i1 {3 `# n% f, x |, e
function [sys,x0]=dong(t,x,u,flag)5 X$ |. g, A" M
1 e' ~4 G9 I4 c6 U5 K5 z
if flag==0) u/ r# B! c- ?! [0 m
6 v6 U4 f" p( q$ e2 f
sys=[2;0;2;1;0;0];
- I% n( [3 S& a3 @0 h6 ]5 n4 S9 m7 o
x0=[0;0];
( m6 E6 q T5 \' O
& Z5 i, \; A7 Y8 _+ h3 Selseif flag==1 U: _4 b- U$ n T
1 r- x0 J* u& v# _# u0 zsys=[x(2);9.81*sin(x(1))-2*x(2)+u];: y1 M7 Z0 E1 w! } C1 k/ x4 w: H
8 z& c7 h G0 B% |7 qelseif flag==3
' j2 a7 K! Q6 y* _# F% U4 X
5 {. X9 S: \6 X; Usys=[x(1);x(2)];
9 m9 L/ l7 @( |+ \5 D h" |
+ E. a! B4 \- B( V- @else# x, x* I* U; Y* X2 v! ~/ e; o2 a
4 A7 ^ D% ^, v9 ?sys=[];' c; E$ _ [( g4 E5 u8 X2 F
+ j h j5 B( ]& f
end |