BookmarkSubscribeRSS Feed
🔒 This topic is solved and locked. Need further help from the community? Please sign in and ask a new question.
Len18
Fluorite | Level 6

I am trying to create a table where column A = 'Y', if column A does not exist then where column B= 'Y'. Some months the source table has column A and other months it has column B.

 

I attempted to use the coalescec function but this does no work if the both columns are not present. 

PROC SQL;
CREATE TABLE TBL_&DATE. AS
	SELECT 
		NAME,	
		ADDRESS,
		PRODUCT
	FROM 
		SRC_&DATE.
	WHERE 
		COALESCEC(COL_A,COL_B) = 'Y'
;QUIT;

 

I am using SAS EG 7.1 

1 ACCEPTED SOLUTION

Accepted Solutions
ChrisNZ
Tourmaline | Level 20

SQL's rigid/declarative structure is unsuited to dealing with varying inputs.

 

data TBL_&date.;
  length COL_A COL_B $1;
  set SRC_&date.; 	
  COL=coalescec(COL_A, COL_B)
  if COL = 'Y';
  keep NAME ADDRESS PRODUCT COL;
run;

 

View solution in original post

12 REPLIES 12
Jagadishkatam
Amethyst | Level 16

try below code

 

PROC SQL;
CREATE TABLE TBL_&DATE. AS
	SELECT 
		NAME,	
		ADDRESS,
		PRODUCT
	FROM 
		SRC_&DATE.
	WHERE 
		COL_A='Y' or COL_B= 'Y'
;QUIT;
Thanks,
Jag
Len18
Fluorite | Level 6
Hi,
This gives the column not found error for either Column A or B depending on the month, as both columns do not exist in the source table each month.
s_lassen
Meteorite | Level 14

A simple macro could do the trick:

%macro extract(date);
  %local dsid varname;
  %let dsid=%sysfunc(open(SRC_&date,i));
  %if &dsid=0 %then %do;
    %put ERROR: TABLE SRC_&date could not be opened;
    %return;
    %end;
  %if %sysfunc(varnum(&dsid,COL_A)) %then 
    %let varname=COL_A;
  %else 
    %let varname=COL_B;
  %let dsid=%sysfunc(close(&dsid));
  PROC SQL;
    CREATE TABLE TBL_&DATE. AS
	SELECT 
		NAME,	
		ADDRESS,
		PRODUCT
	FROM 
		SRC_&DATE.
	WHERE 
		&varname='Y';
  QUIT;
%mend;

%extract(02JUL2020);

The macro checks if COL_A is present on the input table and uses that if it is, otherwise COL_B. 

ballardw
Super User

There is a big difference between "Column A does not exist" and "the value of column a is missing".

"Does not exist" to me means that the variable is not in the data set and any reference attempting to use the variable will be an error, and rightfully so.

 

Your process should be fixed so that the variable name is consistent. I would spend more time tracking down why and fixing it much earlier in the process. When I see one issue like this I usually see multiples. And each causes a different set of headaches.

 

If you are "reading" external data from a source that cannot have consistent column headers, typical examples are spreadsheets with poor control over contents, with Proc Import then you need to address how the data is read. Proc Import guesses each and every time it is called to read a file. And differences will appear such as variable types changing, variable names as you have seen, lengths of variables. All of these make programming difficult. Since you have mentioned sometimes one name and sometimes another then that strongly implies you need to control the data read as you go forward to prevent problems.

Len18
Fluorite | Level 6
Hi, yes, the variable is not always in the dataset. The source table is not within my control and although either column can be used for my purpose, the two columns are not identical. Ideally, I would have liked both columns to be in the dataset.
AhmedAl_Attar
Ammonite | Level 13

Depending on your SAS version, this code may work for you

%let date=&sysdate;
data src_&date;
	NAME='Some Name';
	ADDRESS='Some Address';
	PRODUCT='Some Product';
	COL_A='Y';
	*COL_B='Y';
run;
%let g_dynWhere=;
options mprint;

PROC SQL NOPRINT;
	SELECT STRIP(name)||"='Y'"
	INTO	:g_dynWhere separated by ' OR ' 
	FROM	dictionary.columns
	WHERE	libname='WORK'
	AND		memname=upcase("SRC_&DATE")
	AND		name LIKE 'COL_%'
	AND		memtype='DATA';

	/*%put &=g_dynWhere;*/

	CREATE TABLE TBL_&DATE. AS
	SELECT 
		NAME,	
		ADDRESS,
		PRODUCT
	FROM 
		SRC_&DATE.
	%if (%superq(g_dynWhere) NE ) %then
	%do;
		WHERE &g_dynWhere
	%end;
	;
QUIT;	

Back in 2018, @ChrisHemedinger posted the following Blog about SAS 9.4 M5. The code above would examine the structure of your source table and dynamically build the condition.

That condition then in turn, would conditionally be used in the Create Table Query.

 

Hope this helps,

Ahmed  

Len18
Fluorite | Level 6
Hi I just tried it, it's giving me the full dataset and not just those containing 'Y' in the column.

Just to ensure I follow what the code is doing though. You firstly created a dummy dataset. Then you created a null variable g_dynWhere. Then it's removing leading and trailing spaces and putting data from columns A & B into the null variable created previously. Since the 2 columns can now be referenced by the single variable we use it in the where clause by saying if it is not blank then we apply it to where clause.
AhmedAl_Attar
Ammonite | Level 13

@Len18 

I think you got the most of it. Here is my break down

1. Create sample/dummy data to simulate your initial situation, and provide working code.

2. dictionary.columns is a table that contains the records about the structure of the table. Think of it, as a table representation of the SQL Describe tablename statement.

3. Create Macro variable (g_dynWhere) that would hold a 'OR' separated string values of Col_A='Y' OR Col_B='Y'. Col_A and Col_B are values stored in the name column.

4. The condition

%if (%superq(g_dynWhere) NE ) %then
%do;
WHERE &g_dynWhere
%end;

Would only add the WHERE clause if the resolved value of the  macro variable (g_dynWhere) was not blank. i.e. at least one of the two columns (Col_A, Col_B) existed in the Source Table with the required value.

 

Hope this clarify it for you

Ahmed

Len18
Fluorite | Level 6
Thanks a lot, yes this explains it perfectly.
PGStats
Opal | Level 21

I would suggest this kind of approach (a view and coalescec with a default value) :

 

data t1;
do id = "a", "b", "c", "";
    a = id;
    output;
    end;
run;

data t2;
do id = "a", "b", "c", "";
    b = id;
    output;
    end;
run;

proc sql;
create table tt
   (
    table char(2),
    id char(1),
    c char(12),
    whichColumn char(12)
   );
quit;

%macro insert(t);

data _t / view=_t;
length a b $12;
set &t;
run;

proc sql;
insert into tt (table, id, c, whichColumn) 
select 
    "&t." as table,
    id, 
    coalescec(a, b, "x") as c,
    choosec(1 + whichc(calculated c, a, b), "None", "Col a", "Col b") as whichColumn
from _t;
drop view _t;
quit;

%mend insert;

%insert(t1);
%insert(t2);

proc print data=tt noobs; run;
table 	id 	c 	whichColumn
t1 	a 	a 	Col a
t1 	b 	b 	Col a
t1 	c 	c 	Col a
t1 	  	x 	None
t2 	a 	a 	Col b
t2 	b 	b 	Col b
t2 	c 	c 	Col b
t2 	  	x 	None
PG
ChrisNZ
Tourmaline | Level 20

SQL's rigid/declarative structure is unsuited to dealing with varying inputs.

 

data TBL_&date.;
  length COL_A COL_B $1;
  set SRC_&date.; 	
  COL=coalescec(COL_A, COL_B)
  if COL = 'Y';
  keep NAME ADDRESS PRODUCT COL;
run;

 

Len18
Fluorite | Level 6
This worked! Thank you

SAS Innovate 2025: Save the Date

 SAS Innovate 2025 is scheduled for May 6-9 in Orlando, FL. Sign up to be first to learn about the agenda and registration!

Save the date!

How to Concatenate Values

Learn how use the CAT functions in SAS to join values from multiple variables into a single value.

Find more tutorials on the SAS Users YouTube channel.

SAS Training: Just a Click Away

 Ready to level-up your skills? Choose your own adventure.

Browse our catalog!

Discussion stats
  • 12 replies
  • 5147 views
  • 4 likes
  • 7 in conversation