在宏内部使用变量内部数据集名称时,SAS语法错误22和200

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了在宏内部使用变量内部数据集名称时,SAS语法错误22和200相关的知识,希望对你有一定的参考价值。

我正在尝试使用宏中的循环在SAS中对几个数据集进行排序,使用数据列表(其中一些在数据集名称中有前导零)(例如,在示例代码中,01,02列表),以及此代码还将指导我想构建的其他一些循环。

我使用SAS指南循环遍历一个非顺序的值列表,以宏DO循环代码作为起点:http://support.sas.com/kb/26/155.html

data dir1201;
   input compid directorid X;
   format ;
   datalines;
01 12 11  
02 15 5  
;
run;

data dir1202;
   input compid directorid X;
   format ;
   datalines;
01 12 1  
03 18 8  
;
run;

%macro loops1(values);                                                                                                         
     /* Count the number of values in the string */                                                                                                                                   
     %let count=%sysfunc(countw(&values)); 
     /* Loop through the total number of values */                                                                                         
     %do i = 1 %to &count;
      %let value=%qscan(&values,&i,%str(,)); 
      proc sort data=dir12&value out=dir12sorted&value nodupkey;
      by directorid compid;
      run;
      %end;
%mend;
options mprint mlogic;
%loops1(%str(01,02))  

我假设非顺序列表需要str,但是当我想保留前导零时这也很有用;

我看到宏变量似乎在日志中包含01或02,但之后我收到错误22和200。以下是使用此示例代码的日志错误的片段:

339  %macro loops1(values);
340       /* Count the number of values in the string */
341       %let count=%sysfunc(countw(&values));
342       /* Loop through the total number of values */
343       %do i = 1 %to &count;
344        %let value=%qscan(&values,&i,%str(,));
345        proc sort data=dir12&value out=dir12sorted&value nodupkey;
346        by directorid compid;
347        run;
348        %end;
349  %mend;
350  options mprint mlogic;
351  %loops1(%str(01,02))
MLOGIC(LOOPS1):  Beginning execution.
MLOGIC(LOOPS1):  Parameter VALUES has value 0102
MLOGIC(LOOPS1):  %LET (variable name is COUNT)
MLOGIC(LOOPS1):  %DO loop beginning; index variable I; start value is 1; stop value is 2; by
      value is 1.
MLOGIC(LOOPS1):  %LET (variable name is VALUE)
NOTE: Line generated by the macro variable "VALUE".
1    dir1201
          --
          22
           --
           200
ERROR: File WORK.DIR12.DATA does not exist.

我不明白为什么dir1201显示,但然后错误是引用数据集work.dir12(忽略01

答案

宏报价让解析器误以为宏报价是新标记开始的信号。您可以添加%unquote()以删除宏引用。

proc sort data=%unquote(dir12&value) out=%unquote(dir12sorted&value) nodupkey;

或者只是不要添加宏引用开头。

%let value=%scan(&values,&i,%str(,));

如果您将宏设计为采用空格分隔值而不是逗号分隔,则使用宏会更容易。然后,无需在调用中添加宏引用。

%macro loops1(values);
%local i value ;
%do i = 1 %to %sysfunc(countw(&values,%str( )));
  %let value=%scan(&values,&i,%str( ));
proc sort data=dir12&value out=dir12sorted&value nodupkey;
  by directorid compid;
run;
%end;
%mend loops1;
%loops1(values=01 02)
另一答案

宏声明选项/PARMBUFF用于使自动宏变量SYSPBUFF可用于扫描作为参数传递的任意数量的逗号分隔值。

%macro loops1/parmbuff;
  %local index token;
  %do index = 1 %to %length(&syspbuff);
    %let token=%scan(&syspbuff,&index);
    %if %Length(&token) = 0 %then %goto leave1;
    proc sort data=dir12&token out=dir12sorted&token nodupkey;
    by directorid compid;
    run;
  %end;
  %leave1:
%mend;
options mprint nomlogic;
%loops1(01,02)

SYSPBUFF MACRO / PARMBUFF

另一答案

由于你在循环日期,我认为这样的事情从长远来看可能更有帮助:

%macro date_loop(start, end);
    %let start=%sysfunc(inputn(&start, anydtdte9.));
    %let end=%sysfunc(inputn(&end, anydtdte9.));
    %let dif=%sysfunc(intck(month, &start, &end));

    %do i=0 %to &dif;
        %let date=%sysfunc(intnx(month, &start, &i, b), yymmdd4.);
        %put &date;
    %end;
%mend date_loop;

%date_loop(01Jan2012, 01Jan2015);

这是SAS文档中的略微修改版本(宏附录,示例11)。

http://documentation.sas.com/?docsetId=mcrolref&docsetTarget=n01vuhy8h909xgn16p0x6rddpoj9.htm&docsetVersion=9.4&locale=en

以上是关于在宏内部使用变量内部数据集名称时,SAS语法错误22和200的主要内容,如果未能解决你的问题,请参考以下文章

在宏中传递变量 args 时出现语法错误

交织 SAS 数据集(按常见患者编号)

SAS数据步骤视图和数据包装在宏for循环中

第二章 导入数据到SAS | 逻辑库和SAS数据集

SAS;TABULATE制作报表

我的SAS菜鸟之路3