Calculate sum of 1+(1/2!)+…+(1/n!) n number in C language

后端 未结 4 1366
情歌与酒
情歌与酒 2021-01-24 08:44

Like the title say, how I calculate the sum of n number of the form: 1+(1/2!)+⋯(1/n!)? I already got the code for the harmonic series:

#include 

         


        
4条回答
  •  野趣味
    野趣味 (楼主)
    2021-01-24 09:08

    The number n! is equal to the product of n and the preceding factorial, that is, (n - 1)!.
    If you calculate n! in an iteration, you are doing n products.
    In the next step, say n+1, you repeat again these n products followed by the multiplication by n+1.
    This means that you are repeating the same operations again and again.

    It is a better strategy to hold the previous factorial that was calculated in the step n, and then, in the step n+1, just to multiply the n! by n+1. This reduces the number of products to 1 in each iteration.

    Thus, you can calculate the series in the following way:

     int max_n = 20;     /* This value can come from another point of the program */
    
     int n;                  /* Initial value of the index */
     double factorial_n = 1; /* It has to be initialized to 1, since the factorial of 0 is 1 */
     double sum = 0.0;       /* It has to be initialized to 0, in order to calculate the series */
    
     for (n = 0; n <= max_n; )
      {
          sum += 1.0/factorial_n;
          n++;
          factorial_n *= n;
      }
    
     printf("Series result: %.20f\n", sum);
    

    There are some numerical issues with this approach, but this go beyond the scope of your question.

    About overflow: It is necessary to be carefull about the overflow of factorials after several iterations. However, I will not write code to handle overflow.

    EDIT

    I think that you have not to follow the suggestions of those people that advice to use a factorial function. This approach is very unefficient, since a lot of products are done in every iteration.
    IN comparisson with that approach, the mine is better.

    However, if you have plans to calculate these series very often, then my approach is not efficient anymore. Then, the right technique is that pointed out in the Bli0042's answer, that is: to hold the factorials in an array, and then just use them every time you need, without need to calculate them again and again in the future.

    The resulting program would be this:

    #include  
    
    #define MAX_N 100
    
    double factorial[MAX_N+1]; 
    
    void build_factorials(double *factorial, int max)
    {
        factorial[0] = 1.0;
        for (int j = 0;  j <= max; )
         {
           j++;
           factorial[j] = factorial[j-1] * j;
         }
    }
    
    double exp_series(int n)
    {
        int j;
        double sum;
        if (n > MAX_N)  /* Error */
           return 0.0;
    
        sum = 0.0;   
        for (j = n; j >= 0; j--)
           sum += 1.0/factorial[j];
        return sum;
    }
    
    int main(void) 
    {
    
     int n; 
     double sum; 
    
      build_factorials(factorial, MAX_N);
    
      printf("Series (up to n == 11): %.20f\n", exp_series(11));
      printf("Series (up to n == 17): %.20f\n", exp_series(17));
      printf("Series (up to n == 9):  %.20f\n", exp_series(9));
    
      getchar();    
    }
    

    The iteration is done in reverse order inside the function exp_series() in order to improve the numerical issues (that is, to amortiguate the loss of precision when summing small terms).

    The last code has side effects, because an external array is invoked inside the function exp_series().
    However, I think that handling this would become my explanation more obscure.
    Just, take it in account.

提交回复
热议问题