SAS生成表基于条件?

时间:2015-05-04 12:41:02

标签: sas conditional-statements sas-macro proc-sql

我正在尝试编写一个宏,它将根据某些条件创建一个新表。用户可以输入单个数字或一系列数字来过滤原始数据。

例如,如果起始数据集如下所示,最终用户可以选择用单个数字(即:其中Z = 1)或一系列数字(即:Z在1和1之间)过滤Col Z. 5)。

X    Y    Z
0    1    3
2    7    9
0     0   0
2     1   1

如果用户想要输入要过滤的数字范围,则会为& start_range和& end_range填充值。如果用户想要对单个数字进行过滤,则& end_range将等于null。

我想要检查& end_range是否等于null的代码。如果是,则where语句应为列Z =& start_range的所有行创建一个新表。如果& end_range不等于null(它有一个填充的值),where语句将为列Z在& start_range和& end_range之间的所有行创建一个新表。

对于单个值,(其中& start_range = 1和& end_range =''),结果表将为:

X    Y    Z
2     1   1

对于一系列值,(其中& start_range = 1和& end_range = 5),结果表将为:

X    Y    Z
0    1    3
2     1   1

这是我到目前为止的代码,不是100%确定语法是否正确所以对所有建议都开放,谢谢!!:

 %macro filter();

proc sql;   
            create table filtered_data AS
            select raw_data.* 
            from raw_data
            WHERE       

                  %if &end_range NE '' %then %do;
                        Z between '&start_range' AND '&end_range';
                  %end;

                  %else %do;
                        Z = '&start_range';
                  %end;

            quit;

%mend;

%filter();

1 个答案:

答案 0 :(得分:0)

我建议不要创建宏定义,而是通过为开始和结束范围创建2个简单的宏变量来解决问题。起始范围始终具有有效数字,但对于结束范围,您可以在逻辑中编码以获取起始范围值,以防结束范围值为空/空。请查看以下代码,如果有帮助请告诉我。

data raw_data;
infile datalines missover;
input X    Y    Z;
datalines4;
0    1    3
2    7    9
0     0   0
2     1   1
;;;;
run;
/*User enters the range*/
%let start_range=1;
%let end_range=5;

/*User entered data refining*/
%let end_range=%sysfunc(coalescec(&end_range,&start_range));

proc sql;
create table filtered_data AS
select *
from raw_data
where z between &start_range and &end_range
;
quit;
proc print data=filtered_data;run;

如果您仍然希望创建宏定义来解决问题,那么请尝试在宏定义中调整第一个%IF语句,并删除您在PROC SQL的WHERE语句中应用于宏变量的单引号。查看更新的宏定义。

%macro filter;
proc sql;   
            create table filtered_data AS
            select raw_data.* 
            from raw_data
            WHERE       
                  %if %length(&end_range) %then %do;
                        Z between &start_range AND &end_range;
                  %end;

                  %else %do;
                        Z = &start_range;
                  %end;
            quit;
%mend;
%filter;