BookmarkSubscribeRSS Feed
Ksharp
Super User

Better post it at IML forum. @Rick_SAS is there. 

Hope I have time to review your question .

Good Luck.

3 REPLIES 3
kilasue
Calcite | Level 5

I rewrite the objective function using do:

nr=nrow(A);	
start ff(x) global(A,y,lambda,l,nr); xx=x[1:l-1]; sigma=x[l]; f=0; do i=1 to nr; f=f+sigma+huberf((A[i,]*xx-y[i])/sigma)*sigma; end; f=f+lambda*norm(x,"L2"); return(f); finish ff;

But I still can understand why my previous function caused an error. Maybe we cannot using apply() in objective function?

kilasue
Calcite | Level 5

Recently I see that sicikitlearn has a Huberregressor and I want to implement it using SAS. The loss function is in the picture:

2018-08-06_20-05-33.jpg

My code is:

proc iml;
	use mach.leaf_cla;
	read all var _all_ into data[colname=varNames]; 
	
	A=data[,1:ncol(data)-1];
	y=data[,ncol(data)];
	cv=j(nrow(A), 1, 1);
	A=insert(A,cv,0,ncol(A)+1);
*Using OLS estimator as the start value; b=solve(A`*A,A`*y); *Hm(x); start huberf(x) global(epsilon); if abs(x)<epsilon then f=x**2; else f=2*epsilon*abs(x)-epsilon**2; return(f); finish huberf;
*Objective function; start ff(x) global(A,y,lambda,l); xx=x[1:l-1]; sigma=x[l]; AX=(A*xx-y)/sigma; AXt=apply("huberf",AX); f=sum(sigma+AXt*sigma)+lambda*norm(x,"L2"); return(f); finish ff; epsilon=1.35; lambda=1; l=ncol(A)+1; opt={0 2}; sigma=50; b=insert(b,sigma,nrow(b)+1,0); temp=ff(b); print temp; call nlpnrr(rc, xres, "ff", b, opt);

My dataset is in the attachments.

But nlprr gives an error:

2018-08-06_20-09-51.jpg

I have tried nlpcg, nlpdd and nothing works. It's strange that I have tried ff(b) using start value and it give correct result.

2018-08-06_20-14-06.jpg

More details:

 The dataset has 64 explanatory variables and 1 target variable. The model has 64+1 constant +1 sigma =66 parameters.

Rick_SAS
SAS Super FREQ

I suspect you are correct that the APPLY function is causing the problem, although I don't know why. I will enter a bug report with SAS R&D so that someone can investigate.

 

An efficient alternative is to use the CHOOSE function. The APPLY function is not recommended because the APPLY function tends to be less efficient than vectorized computations.

 

In your case, you can avoid the loop by using the following:

start huberf(x) global(epsilon);
   f = choose( abs(x)<epsilon, x##2, 2*epsilon*abs(x)-epsilon##2 );
   return(f);
finish huberf;

 

PS. In your objective function, I think you meant to write NORM(xx) instead of NORM(x)? 

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

From The DO Loop
Want more? Visit our blog for more articles like these.
Discussion stats
  • 3 replies
  • 1625 views
  • 0 likes
  • 3 in conversation