SAS Programming

DATA Step, Macro, Functions and more
BookmarkSubscribeRSS Feed
🔒 This topic is solved and locked. Need further help from the community? Please sign in and ask a new question.
wizkid2050
Fluorite | Level 6

data test;
input accounts;
datalines;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
;
run;

data test2;
input managers$ count;
datalines;
a 48
b 45
c 49
;
run;

 

 

i have these two data sets i want an output of 

 

1 a

2 b

3 c

4 a

5 b

6 b

7 b

8 b

 

after every merge step the value for count should increase for each variable and if the count reaches 50 then

it should not merge for that step.

plz help me with any logic if possible.

Free to use macros or sql.

1 ACCEPTED SOLUTION

Accepted Solutions
gamotte
Rhodochrosite | Level 12
data test3;
	set tr_test2;
	/*keep managers;*/
	array a_mngrs {*} _NUMERIC_;

	if _N_=1 then index=1;

	do i=1 to &n_accounts.;
		if min(of a_mngrs{*}) lt 50 then do;
			do j=1 to dim(a_mngrs);
				if a_mngrs{index} lt 50 then do;
					a_mngrs{index}=a_mngrs{index}+1;
					managers=vname(a_mngrs{index});
					j=dim(a_mngrs); /* Exit loop on j*/
					output;
				end;

				/* Probably a better way to do this */
				index=mod(index+1,dim(a_mngrs));
				if index=0 then index=dim(a_mngrs);
			end;
		end;
		else do;
			i=&naccounts.; /* Exit loop on i */
		end;
	end;
run;

View solution in original post

23 REPLIES 23
Kurt_Bremser
Super User

Looks like the perfect example for using a hash object:

data want (keep=accounts managers);
set test;
length managers $8 count 8;
retain managers;
if _n_ = 1
then do;
  declare hash test2(dataset:'test2', ordered: 'yes');
  declare hiter iter('test2');
  test2.definekey('managers');
  test2.definedata('managers','count');
  test2.definedone();
  call missing(managers,count);
  iter.first();
end;
put 'Start of step';
put managers=;
put count=;
runover = 0;
do while (count >= 50);
  put 'while iteration';
  rc = iter.next();
put managers=;
put count=;
  if rc ne 0
  then do;
    if runover then leave;
    runover = 1;
    rc = iter.first();
  end;
end;
if runover and rc
then do;
  put 'No managers left';
  stop;
end;
put 'after while iteration';
put managers=;
put count=;
count + 1;
test2.replace();
output;
rc = iter.next();
if rc then rc = iter.first();
put 'end of step';
put managers=;
put count=;
run;

Note that the many PUTs in there are for debugging and clarification how it works.

The RETAIN for managers is necessary, or the key value will be set to missing at the start of each data step iteration.

wizkid2050
Fluorite | Level 6
Can't it be done in more simplified manner than this?
Kurt_Bremser
Super User

Addendum:

if it were simple, you wouldn't have needed to come here.

And you just have a fine opportunity to learn about one of the more arcane, but very useful features of the SAS data step language (hash objects).

 

wizkid2050
Fluorite | Level 6

Thanks for the Solution Sir.

Could you also point me in the direction regarding what and where to read about hash objects so that i could understand the code better.

Kurt_Bremser
Super User

@wizkid2050 wrote:

Thanks for the Solution Sir.

Could you also point me in the direction regarding what and where to read about hash objects so that i could understand the code better.


A very good (and quite extensive) paper can be found here:

http://www2.sas.com/proceedings/sugi31/241-31.pdf

This is the relevant part of the Data Step Concepts part of the SAS documentation:

Using DATA Step Component Objects

This is the reference for the language elements (statements and methods):

Hash and Hash Iterator Object Language Elements

 

You can find all this, and more, with a google search for "sas hash objects basics"

Kurt_Bremser
Super User

To help you a little in understanding what happens, I've added some comments:

data want (keep=accounts managers);
set test;
length managers $8 count 8;
* necessary so we don't get into trouble because of wrong variable types;
retain managers;
* because of the retain statement, the key of the hash object
  is kept from one data step iteration to the next;
if _n_ = 1
then do; * set up the hash object in the very first iteration of the data step;
  * declare the hash;
  declare hash test2(dataset:'test2', ordered: 'yes');
  * from the hash, declare the iterative hash object for sequential search;
  declare hiter iter('test2');
  * define hash object members;
  test2.definekey('managers');
  test2.definedata('managers','count');
  test2.definedone();
  * set to missing to avoid notes;
  call missing(managers,count);
  * intialize;
  iter.first();
end;
runover = 0; * this will flag if we already ran through the whole hash once;
do while (count >= 50); * find a manager with count < 50;
  rc = iter.next(); * next hash object;
  if rc ne 0 /* we have reached the "end" of the hash */
  then do;
    if runover then leave; * we've been here already, so exit the loop;
    runover = 1;
    rc = iter.first(); * reset the hash to its beginning;
  end;
end;
if runover and rc
then do; * obviously we've run out of managers with count <= 50;
  put 'No managers left';
  stop; * end the data step execution;
end;
count + 1; * increment;
test2.replace(); * and replace in the hash;
output; * output before stepping to the next hash object;
* if we don't do this now, the elements of the next hash iteration
  will end up in the output observation;
rc = iter.next(); * get next object;
if rc then rc = iter.first(); * if end, reset;
run;
gamotte
Rhodochrosite | Level 12

Hello,

 

I think this should work :

 


proc transpose data=test2 out=tr_test2;
id managers;
idlabel managers;
run;

	data want;
		set tr_test2 (drop=_NAME_);
		array a_mngrs {*}_NUMERIC_;
		
		if _N_=1 then index=1;

		do accounts=1 to 15;
			if a_mngrs{index} lt 50 then do;
				put index=;
				a_mngrs{index}=a_mngrs{index}+1;
				managers=vname(a_mngrs{index});
				output;
			end;
			
			/* Probably a better way to do this */
			index=mod(index+1,3);
			if index=0 then index=3;
		end;
	run;
gamotte
Rhodochrosite | Level 12

It doesn't, sorry 😕

wizkid2050
Fluorite | Level 6

The code seems to run fine but the issue here is you are not making use of first dataset named test.

although i have 1 to 15 there but what if i have that i some random order.

and i dont want to sort it. it has to be merged as it is

 

wizkid2050
Fluorite | Level 6

The code seems to work correctly.

gamotte
Rhodochrosite | Level 12

There was a proble with the accounts column. Here is a corrected version :

 

	data want;
		set tr_test2 (drop=_NAME_);
		array a_mngrs {*}_NUMERIC_;
		keep accounts managers;
		
		if _N_=1 then index=1;

		accounts=0;

		do i=1 to 15;
			if a_mngrs{index} lt 50 then do;
				a_mngrs{index}=a_mngrs{index}+1;
				managers=vname(a_mngrs{index});
				accounts+1;
				output;
			end;
			
			/* Probably a better way to do this */
			index=mod(index+1,dim(a_mngrs));
			if index=0 then index=dim(a_mngrs);
		end;
	run;
gamotte
Rhodochrosite | Level 12
data test;
input accounts;
datalines;
12
2
32
4
5
6
7
8
9
50
11
46
13
14
15
;
run;

data test2;
input managers$ count;
datalines;
a 48
b 45
c 49
;
run;

proc transpose data=test2 out=tr_test2;
id managers;
idlabel managers;
run;

proc sql;
	SELECT count(*)
	INTO :n_accounts
	FROM test;
quit;

data test3;
	set tr_test2;
	keep managers;
	array a_mngrs {*} a--c;

	if _N_=1 then index=1;

	do i=1 to &n_accounts.;
		if a_mngrs{index} lt 50 then do;
			a_mngrs{index}=a_mngrs{index}+1;
			managers=vname(a_mngrs{index});
			output;
		end;

		/* Probably a better way to do this */
		index=mod(index+1,dim(a_mngrs));
		if index=0 then index=dim(a_mngrs);
	end;
run;

data want;
	merge test test3 (in=a);
	if a;
run;
wizkid2050
Fluorite | Level 6

Great work Gamotte. adding a little more to the problem. this sticks for only a b and c what if we have n no. of managers in dataset with different counts. Just to make it dynamic.

sas-innovate-wordmark-2025-midnight.png

Register Today!

Join us for SAS Innovate 2025, our biggest and most exciting global event of the year, in Orlando, FL, from May 6-9. Sign up by March 14 for just $795.


Register now!

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
  • 23 replies
  • 2330 views
  • 4 likes
  • 4 in conversation