/* File CExamples/Recursion/combin2.c */
#include <stdio.h>
long int combin(int i, int k)
/* Binomial coefficient of i choose k;
Note: (i choose k) = 0 for k < 0,
= 1 for k == 0,
= pow(-1, k) (k-i-1 choose k) for i < 0, k > 0
= (i * (i-1) * ... * (i-k+1))/
(k * (k-1) * ... * (1)) for i >= 0, k > 0 */
/* This function remembers previous calls */
{long int combinat; /* intermediate function result */
#define MAXARG 50
static long int memo[MAXARG][MAXARG]; /* statics are initialized to 0 */
#ifdef DEBUG
/* code within #ifdef DEBUG prints the trace */
/* compile with -DDEBUG to get trace */
static int invoke = 0; /* level of invocation */
static int invblanks; /* loop index in debug statement */
invoke = invoke+1; /* raise level of invocation */
for (invblanks = 1; invblanks <= invoke-1; invblanks++)
printf("%1d ", invblanks % 10); /* last digit only */
/* indent properly, writing level numbers also */
printf("combin(%3d,%3d)\n",i,k); /* indicate call */
#endif
/* Check arguments */
if (k < 0) combinat = 0L;
else if (k == 0) combinat = 1L;
/* Now k > 0 */
else if (i < 0)
{if (k & 1) /* test lowest order bit of k */
/* k is odd */ combinat = -combin(k-i-1, k);
else /* k is even */ combinat = combin(k-i-1, k);
}
/* Now k > 0 and i >= 0 */
else if (i == 0 || k > i) combinat = 0L;
/* Now i > 0 and 0 < k <= i */
else if (i == k) combinat = 1L;
/* Now i > 0 and 0 < k < i */
else /* check if space to remember */
if(i <= MAXARG && k <= MAXARG)
/* check if already computed */
{if(memo[i-1][k-1] > 0)
/* already computed, don't recur */
combinat = memo[i-1][k-1];
else /* compute and memorize */
combinat = memo[i-1][k-1] = combin(i-1,k-1)+combin(i-1,k);
}
/* no space to memorize */
else combinat = combin(i-1,k-1)+combin(i-1,k);
#ifdef DEBUG
for(invblanks=1; invblanks<=invoke-1; invblanks++)
printf("%1d ",invblanks % 10);
/* indent properly, writing level numbers also */
printf("value=%ld\n",combinat);
invoke=invoke-1; /* lower level of invocation before return */
#endif
return(combinat);
}/*combin*/