BookmarkSubscribeRSS Feed
amandahe
Calcite | Level 5

Hello,

I have one table which is a fixed table has 4 columns, ID, date, hour, product, the table name as Product

and I have another 10 tables, which has more columns, ID, date, hour, product, category, subcategory, region. And the 10 tables name as YEAR2021_WW01, YEAR2021_WW02, YEAR2021_WW03...YEAR2020_WW02.

 

I would like these 10 tables (YEAR2021_WW01...) inner join with Product on a.id=b.id and a.date=b.date and a.hour=b.hour.

 

and i want to keep all the columns from table a, and one column "product" from table PODUCT, then output the 10 tables with the same table name(YEAR2021_WW01, YEAR2021_WW02, YEAR2021_WW03...YEAR2020_WW02).

 

I write a proc sql just for joining one table,

proc sql;

create table YEAR2021_WW01 as

select a*, b.Product

from YEAR2021_WW01 a inner join PRODUCT b on (a.id=b.id and a.date=b.date and a.hour=b.hour)

quit;

 

How to write a macro to join the 10 tables at once?

 

PRODUCT

IDDATEHOURPRODUCT
100101JAN20218CAMBRIDGE
100215JAN202116SUMMERSET
    

 

 

YEAR2021_WW01

IDDATEHOURSTATUSMOODREGIONCATEGORYSUBCATEGORY
100101JAN20218NORMALTRUEPARISAAA
1002 15JAN202116NORMALTRUEPARISBBB
        

 

 

 

5 REPLIES 5
andreas_lds
Jade | Level 19

Assuming that the year-tables have the same variables. this looks like suboptimal data-design, so i would append them into one table (or view). Afterwards you need only one merge. If you really want to write a macro, start with working code to do one merge and you should not use proc sql, but a data step to avoid the warning:

WARNING: This CREATE TABLE statement recursively references the target table. A consequence of this is a possible data integrity 
         problem.

Note: Overwriting an existing dataset during merge etc. is a bad idea, because you can't repeat the step if something goes wrong, you have to repeat all steps necessary to re-create the overwritten dataset.

ChrisNZ
Tourmaline | Level 20

No macro needed. Something like

proc sql;
  create table WANT as
  select a.*, b.PRODUCT
  from (select * from YEAR2021_WW01
            union
             ....
       select * from YEAR2021_WW10)  a
    inner join PRODUCT b 
  on (a.ID=b.ID and a.DATE=b.DATE and a.HOUR=b.HOUR);
quit;

 

 

amandahe
Calcite | Level 5

Hi Chris,

 

Thank you Chris, how can I get 10 separate tables after inner join with the PRODUCT table? 

ChrisNZ
Tourmaline | Level 20

Why use 10 tables rather than a where clause?

Kurt_Bremser
Super User

So you do a basic lookup on three keys to get the name of product; Do this with a hash object:

data product;
input ID $ DATE :date9. HOUR  PRODUCT :$10.;
format date yymmdd10.;
datalines;
1001  01JAN2021 8 CAMBRIDGE
1002  15JAN2021 16  SUMMERSET
;

data YEAR2021_WW01;
input ID $ DATE :date9. HOUR STATUS $ MOOD $ REGION $ CATEGORY $ SUBCATEGORY $;
format date yymmdd10.;
datalines;
1001  01JAN2021 8 NORMAL  TRUE  PARIS A AA
1002   15JAN2021  16  NORMAL  TRUE  PARIS B BB
;

data want;
set YEAR2021_WW01;
if _n_ = 1
then do;
  length product $10;
  declare hash p (dataset:"product");
  p.definekey("ID","DATE","HOUR");
  p.definedata("product");
  p.definedone();
  call missing(product);
end;
if p.find() = 0;
run;

You simply add all table names to the SET statement.

hackathon24-white-horiz.png

The 2025 SAS Hackathon has begun!

It's finally time to hack! Remember to visit the SAS Hacker's Hub regularly for news and updates.

Latest Updates

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
  • 5 replies
  • 1876 views
  • 0 likes
  • 4 in conversation