-
Notifications
You must be signed in to change notification settings - Fork 96
/
Copy pathcaliscmp.sas
472 lines (416 loc) · 14.1 KB
/
caliscmp.sas
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
/*--------------------------------------------------------------*
* Name: caliscmp.sas *
* Title: Compare model fits from PROC CALIS *
Doc: http://www.datavis.ca/sasmac/caliscmp.html
*--------------------------------------------------------------*
* Author: Michael Friendly <[email protected]> *
* Created: 25 Oct 2000 08:55:11 *
* Revised: 8 Nov 2001 17:45:19 *
* Version: 2.0 *
* Added more fit indices for V8 *
* Revised to work in V9.3
* *
*--------------------------------------------------------------*/
/*=
=Description:
PROC CALIS can fit only one model at a time. It cannot, therefore,
provide the comparative model tests given by other SEM/CFA software,
(e.g., AMOS), which can fit multiple models together.
The CALISCMP macro extracts goodness of fit statistics from the
RAM or OUTFIT data sets produced in a series of models fit to a given data
set using PROC CALIS. The COMPARE= parameter allows pairs of
nested models to be tested.
THIS MACRO WORKS DIFFERENTLY IN SAS 9.3 DUE TO EXTENSIVE CHANGES IN
THE FORMAT OF OUTPUT DATA SETS.
=Usage:
For each model, use the OUTRAM= option (or OUTFIT= in SAS 9.3+) on the
PROC CALIS statement to same the model fit statistics in a separate data set.
For example,
proc calis data=lord cov summary outram=ram1;
lineqs ....
proc calis data=lord cov summary outram=ram2;
lineqs ....
...
The CALISCMP macro is defined with keyword parameters. The RAM=
parameter must specify a list of one or more OUTRAM data sets.
The arguments may be listed within parentheses in any order, separated
by commas. For example:
%caliscmp(ram=ram1 ram2 ram3 ram4,
models=%str(H1 rho=1/H2/H3 rho=1/H4),
compare=1 2 / 3 4 /1 3/ 2 4);
==Parameters:
* RAM= List of the names of two or more OUTRAM data sets from
CALIS runs, separated by spaces.
* MODELS= List of model descriptions or labels, separated by '/',
in the order corresponding to the RAM= list. If not
specified, the models are labeled 'Model 1', 'Model 2', ...
* STATS= List of statistics to be printed for each model.
[Default: STATS=NPARM DF CHISQUAR P_CHISQ RMR GFI AGFI AIC
CAIC SBC CENTRALI PARSIMON CNHOELT]
* COMPARE= List of pairs of models to be compared, separated by '/'.
Each pair consists of two integers, referring to the
models specified in the RAM= and MODELS= lists.
Each pair should refer to two nested models, where the
first is the more restrictive or more constrained.
For each pair, the macro calculates the difference in
Chi-squares, which is tested as a Chi-square on the
difference in degrees of freedom.
* OUT= Name of the outoput data set containing the fit statistics
for all models. [Default: OUT=RAMSTATS]
=Example:
The SAS/STAT User's Guide (``Intro to Structural Equations...'')
describes four models fit to Lord's data on four vocabulary tests,
two short tests with liberal time limits, two long, speeded tests.
These models posit one factor for the unspeeded tests, and a second factor
for the speeded tests. The interesting psychometric questions are
(a) whether each pair of tests can be considered parallel, and
(b) whether the two-factors could be collapsed into one (rho=1).
After fitting these models with separate PROC CALIS calls, compare
the fit statistics, and test for differences among pairs of models
as follows:
%caliscmp(ram=ram1 ram2 ram3 ram4,
models=%str(H1 rho=1/H2/H3 rho=1/H4),
compare=1 2 / 3 4 /1 3/ 2 4);
This produces the following output:
Model Comparison Statistics from 4 RAM data sets
RMS
Model Parameters df Chi-Square P>ChiSq Residual GFI Adj GFI
H1 rho=1 4 6 37.3412 0.00000 2.53409 0.97048 0.95081
H2 5 5 1.9320 0.85847 0.69829 0.99849 0.99699
H3 rho=1 8 2 36.2723 0.00000 2.43656 0.97122 0.85608
H4 9 1 0.7033 0.40168 0.27150 0.99946 0.99458
Schwarz Critical
Model AIC C_AIC BIC Centrality Parsimony N
H1 rho=1 25.3412 -7.5114 -1.5114 0.97614 0.97454 219.51
H2 -8.0680 -35.4452 -30.4452 1.00237 0.83224 3714.12
H3 rho=1 32.2723 21.3214 23.3214 0.97394 0.32509 108.04
H4 -1.2967 -6.7722 -5.7722 1.00023 0.16659 3540.52
Model Comparison ChiSq df p-value
---------------------------------------------------------
H1 rho=1 vs. H2 35.4092 1 0.00000 ****
H3 rho=1 vs. H4 35.5690 1 0.00000 ****
H1 rho=1 vs. H3 rho=1 1.0689 4 0.89918
H2 vs. H4 1.2287 4 0.87335
=*/
%macro caliscmp(
ram=, /* list of RAM data sets from CALIS runs */
models=, /* list of model descriptions */
stats=nparm df chisquar p_chisq rmr gfi agfi aic caic sbc centrali parsimon cnhoelt, /* list of statistics to be printed for each model */
compare=, /* list of pairs of models to be compared */
out=ramstats
);
%*-- Reset required global options;
%if %sysevalf(&sysver >= 7) %then %do;
%local o1 o2;
%let o1 = %sysfunc(getoption(notes));
%let o2 = %sysfunc(getoption(validvarname,keyword));
options nonotes validvarname=upcase;
/* This bit doesnt seem to work in SAS 9.3 */
data _null_;
set sashelp.vtitle(obs=2);
if _n_=2 then call symput('title2', text);
%end;
%else %do;
options nonotes;
%end;
%if %sysevalf(&sysver > 9.2) %then %do;
%goto v93;
%end;
%let abort=0;
%if %length(&ram)=0 %then %do;
%put ERROR: The RAM= parameter must specify a list of RAM data sets;
%let abort=1;
%goto done;
%end;
*-- Combine the OUTRAM data sets, select the _TYPE_='STAT' statistics;
data _stats_;
length model $40;
set
%let i=1;
%let dsn = %scan(&ram, 1, %str( ));
%do %while (&dsn ne );
&dsn (in=in&i)
%let i = %eval(&i+1);
%let dsn = %scan(&ram, &i, %str( ));
%end;
%let n = %eval(&i-1);
;
%do i=1 %to &n;
if in&i then do;
model = "%scan(&models, &i, %str(/))";
if model = ' ' then model = "Model &i";
end;
%end;
keep model _name_ _estim_;
if _type_='STAT';
run;
proc transpose data=_stats_ out=&out(drop=_label_ _name_);
by model notsorted;
var _estim_;
data &out;
set &out nobs=nmod end=eof;
label
model='Model'
n='N'
fit='Fit Criterion'
nparm='Parameters'
df='df'
chisquar ='Chi-Square'
p_chisq = 'P>ChiSq'
rmr='RMS Residual'
gfi='GFI'
agfi='Adj GFI'
chisqnul='Null model ChiSq'
aic='AIC'
caic='Consistent AIC'
sbc='Schwarz BIC'
centrali='Centrality'
parsimon='Parsimonious NFI'
cnhoelt='Critical N'
ztestwh = 'Wilson-Hilferty Z'
%if %sysevalf(&sysver >= 7) %then %do;
rmseaest = 'RMSEA Estimate'
rmsealob = 'RMSEA Lower bound'
rmseaupb = 'RMSEA Upper bound'
compfiti = 'Comparative fit index'
evci_est = 'ECVI Estimate'
ecvi_lob = 'ECVI lower-90% limit'
ecvi_upb = 'ECVI upper-90% limit'
bb_nonor = 'Bentler-Bonet non-normed index'
bb_normd = 'Bentler-Bonet normed fit index'
bol_rho1 = 'Bollen normed Rho1'
bol_del2 = 'Bollen non-normed Delta2'
%end;
;
if eof then call symput('nmod', trim(left(put(nmod,3.0))));
run;
%put nmod=&nmod;
title2 "Model Comparison Statistics from &nmod RAM data sets";
proc print label;
id model;
var &stats;
run;
%if %length(&compare) %then %do;
proc format;
value stars 0 - 0.0001 = '****'
0.0001<-0.001 = '*** '
0.001 <-0.01 = '** '
0.01 <-0.05 = '* '
other = ' ';
data _null_;
set &out end=eof;
file print;
length comp modcmp $200 pair $20;
array chi{&nmod} chisq1-chisq&nmod;
array dof{&nmod} df1-df&nmod;
array mod{&nmod} $40 model1-model&nmod;
retain chisq1-chisq&nmod df1-df&nmod model1-model&nmod;
chi[_n_] = chisquar;
dof[_n_] = df;
mod[_n_] = model;
if eof then do;
comp = "&compare";
sep = ' vs. ';
pair = scan(comp, 1, "/");
indent = 5;
statind = indent + 40;
* put +indent ' ChiSq df p-value Model Comparison' /
+indent '---------------------------------------------------------';
put +indent ' Model Comparison ' @(statind+5) 'ChiSq df p-value '/
+indent '---------------------------------------------------------';
do i=1 by 1 until (pair = ' ');
m1 = input(scan(pair, 1), 4.0);
m2 = input(scan(pair, 2), 4.0);
* put pair= m1= m2=;
if (m1^=. & m2^=.) then do;
chisq = chi[m1] - chi[m2];
ddf = dof[m1] - dof[m2];
modcmp = trim(mod[m1]) || sep || trim(mod[m2]);
if (chisq < 0) then do;
chisq = - chisq;
ddf = - ddf;
modcmp = trim(mod[m2]) || sep || trim(mod[m1]) || ' [R]';
end;
p = 1 - probchi(chisq, ddf);
stars = put(p, stars.);
* put +indent chisq 10.4 ddf 5. p 10.5 +1 stars $4. +5 modcmp;
put +indent modcmp @(statind) chisq 10.4 ddf 5. p 10.5 +1 stars $4.;
end;
pair = scan(comp, i+1, "/");
end;
end;
run;
%end;
%goto exit;
/***********************************************
Code for SAS V 9.3+: Still under development
************************************************/
%v93:
%put WARNING: This macro does not work the same with PROC CALIS in SAS 9.3+;
proc format;
value fitcode
101 = "N Observations"
103 = "N Variables"
104 = "N Moments"
105 = "N Parameters"
106 = "N Active Constraints"
111 = "Baseline Model Function Value"
113 = "Baseline Model Chi-Square"
114 = "Baseline Model Chi-Square DF"
115 = "Pr > Baseline Model Chi-Square"
201 = "Fit Function"
203 = "Chi-Square"
204 = "Chi-Square DF"
205 = "Pr > Chi-Square"
211 = "Z-Test of Wilson & Hilferty"
212 = "Hoelter Critical N"
213 = "Root Mean Square Residual (RMSR)"
214 = "Standardized RMSR (SRMSR)"
215 = "Goodness of Fit Index (GFI)"
301 = "Adjusted GFI (AGFI)"
302 = "Parsimonious GFI"
303 = "RMSEA Estimate"
304 = "RMSEA Lower 90% Confidence Limit"
305 = "RMSEA Upper 90% Confidence Limit"
306 = "Probability of Close Fit"
307 = "ECVI Estimate"
308 = "ECVI Lower 90% Confidence Limit"
309 = "ECVI Upper 90% Confidence Limit"
310 = "Akaike Information Criterion"
311 = "Bozdogan CAIC"
312 = "Schwarz Bayesian Criterion"
313 = "McDonald Centrality"
401 = "Bentler Comparative Fit Index"
402 = "Bentler-Bonett NFI"
403 = "Bentler-Bonett Non-normed Index"
404 = "Bollen Normed Index Rho1"
405 = "Bollen Non-normed Index Delta2"
406 = "James et al. Parsimonious NFI";
data _stats0_;
merge
%let i=1;
%let dsn = %scan(&ram, 1, %str( ));
%do %while (&dsn ne );
&dsn (rename=FitValue=Model&i)
%let i = %eval(&i+1);
%let dsn = %scan(&ram, &i, %str( ));
%end;
%let n = %eval(&i-1);
;
by IndexCode;
drop PrintChar;
label
%do i=1 %to &n;
Model&i = "%scan(&models, &i, %str(/))"
%end;
;
if IndexCode in (101,105,203,204,205,213,215,301,303,304,305,306, 310,311,312);
proc print data=_stats0_(drop=IndexCode) label;
id FitIndex;
by _type_ notsorted;
run;
data _stats_;
length model $40;
set
%let i=1;
%let dsn = %scan(&ram, 1, %str( ));
%do %while (&dsn ne );
&dsn (in=in&i)
%let i = %eval(&i+1);
%let dsn = %scan(&ram, &i, %str( ));
%end;
%let n = %eval(&i-1);
;
%do i=1 %to &n;
if in&i then do;
model = "%scan(&models, &i, %str(/))";
if model = ' ' then model = "Model &i";
end;
%end;
keep model _type_ IndexCode FitIndex FitValue;
if IndexCode in (105,203,204,205,213,215,301,303,304,305,306, 310,311,312);
proc transpose data=_stats_ out=&out(drop=_label_ _name_);
by model notsorted;
id FitIndex;
var FitValue;
data &out;
set &out nobs=nmod end=eof;
if eof then call symput('nmod', trim(left(put(nmod,3.0))));
run;
%put CALISCMP: Model Comparison Statistics from nmod=&nmod models;
*title2 "Model Comparison Statistics from &nmod RAM data sets";
/*
proc print data=&out;
id model;
run;
*/
%if %length(&compare) %then %do;
proc format;
value stars 0 - 0.0001 = '****'
0.0001<-0.001 = '*** '
0.001 <-0.01 = '** '
0.01 <-0.05 = '* '
other = ' ';
ods escapechar='^';
%let delta = ^{unicode 0394};
data _comp_;
set &out(keep=model chi_square chi_square_df) end=eof;
*file print;
length comp modcmp $200 pair $20;
array chi{&nmod} chisq1-chisq&nmod;
array dof{&nmod} df1-df&nmod;
array mod{&nmod} $40 model1-model&nmod;
retain chisq1-chisq&nmod df1-df&nmod model1-model&nmod;
drop chisq1-chisq&nmod df1-df&nmod model1-model&nmod;
drop comp sep pair m1 m2 i chi_square chi_square_df model;
chi[_n_] = chi_square;
dof[_n_] = chi_square_df;
mod[_n_] = model;
label modcmp='Model comparison'
chisq="&delta Chi Square"
p="Pr(>&delta Chi Square)"
ddf="&delta df";
if eof then do;
comp = "&compare";
sep = ' vs. ';
pair = scan(comp, 1, "/");
do i=1 by 1 until (pair = ' ');
m1 = input(scan(pair, 1), 4.0);
m2 = input(scan(pair, 2), 4.0);
* put pair= m1= m2=;
if (m1^=. & m2^=.) then do;
chisq = chi[m1] - chi[m2];
ddf = dof[m1] - dof[m2];
modcmp = trim(mod[m1]) || sep || trim(mod[m2]);
if (chisq < 0) then do;
chisq = - chisq;
ddf = - ddf;
modcmp = trim(mod[m2]) || sep || trim(mod[m1]) || ' [R]';
end;
p = 1 - probchi(chisq, ddf);
stars = put(p, stars.);
* put +indent chisq 10.4 ddf 5. p 10.5 +1 stars $4. +5 modcmp;
* put +indent modcmp @(statind) chisq 10.4 ddf 5. p 10.5 +1 stars $4.;
output;
end;
pair = scan(comp, i+1, "/");
end;
end;
run;
proc print data=_comp_ label;
id modcmp;
var chisq ddf p stars;
run;
%end;
%exit:
%*-- Restore global options;
%if %sysevalf(&sysver >= 7) %then %do;
options &o1 &o2;
*title2 "&title2";
%end;
%else %do;
options notes;
*title2;
%end;
%done:
%mend;