C++洛谷题解(34)——P1009

题目信息

题名:阶乘之和

题目:用高精度计算出 S = 1! + 2! + 3! + ··· + n! (n ≤ 50)其中 !  表示阶乘,定义为n! = n × (n - 1) × (n - 2) × ··· × 1。例如,5! = 5 × 4 × 3 × 2 × 1 = 120。

输入格式:一个正整数 n。

输出格式:一个正整数 S,表示计算结果。

输入样例:3

输出样例:9

说明提示:对于 100% 的数据,1≤n≤50。注,《深入浅出基础篇》中使用本题作为例题,但是其数据范围只有 n≤20,使用书中的代码无法通过本题。如果希望通过本题,请继续学习第八章高精度的知识。

分析题目

这道题如果光是阶乘没有高精度就很简单,只可惜它有高精度,是《深入浅出基础篇》这本书中的代码的升级版,这样难度就直接翻倍了!

由于这道题相对其他题目而言很难,涉及到后面高精度的知识,所以这次的分析会长一点:

  1. 创建a,b,c,sum四个数组
  2. 创建函数pplus,具体内容在下面
  3. 创建函数cheng,具体内容也在下面
  4. 主函数内具体内容还在下面

好了,先来说一下pplus函数的内容:

  1. 函数参数那里写的是int *a,int *c,形参上面创建的数组
  2. 声明变量jw,并初始化为0
  3. 遍历1~1000之间的数,用for循环
  4. 然后是高精度的处理,这里不多说了,直接亮一下代码:
  5. c[i] += a[i] + jw;

    jw = c[i] / 10;

    c[i] %= 10;

然后说一下cheng函数的内容,这个函数和pplus差不多:

  1. 函数参数写的是int *a,int c,注意,c前面不要加*
  2. 再次声明变量jw,并初始化为0
  3. 还是遍历1~1000之间的数,用for循环
  4. 仍就是高精度的处理,这里不多说了,直接亮一下代码:
  5. a[i] = a[i] * c + jw;

    jw = a[i] / 10;

    a[i] %= 10;

代码与上面虽然差不多,但还是不同。最后是主函数部分:

  1. 声明变量n并输入
  2. 特别设置a[1]=1
  3. 遍历1~n之间的所有数,用cheng(a,i);和pplus(a,c);输入
  4. 创建bool类型的flag,初始化为0,即False
  5. 倒叙遍历1~1000之间的数
  6. 如果c[i]!=0就将flag设为1,即True
  7. 否则直接输出c[i]

至此,代码分析完成!

示例代码

注释版:

#include <bits/stdc++.h>//万能头
using namespace std;
int a[2000];//声明的
int b[2000];//a,b,c,sum
int c[2000];//四个数组并
int sum[2000];//设为2000
void pplus(int *a, int *c)//创建pplus函数
{
    int jw = 0;//声明jw并初始化为0
    for (int i = 1; i <= 1000; i++)//遍历1~1000
    {
        c[i] += a[i] + jw;//高精度
        jw = c[i] / 10;//数据
        c[i] %= 10;//操作
    }
}
void cheng(int *a, int c)//创建的cheng函数
{
    int jw = 0;//还是声明jw
    for (int i = 1; i <= 1000; i++)//遍历1~1000
    {
        a[i] = a[i] * c + jw;//依旧是
        jw = a[i] / 10;//高精度
        a[i] %= 10;//数据操作
    }
}
int main()
{
    int n;//声明n
    cin >> n;//输入n
    a[1] = 1;//a[1]设为1
    for (int i = 1; i <= n; i++)//遍历1~n
    {
        cheng(a, i);//用函数
        pplus(a, c);//用函数x2
    }
    bool flag = 0;//创建bool的flag,初始化为0
    for (int i = 1000; i >= 1; i--)//倒叙
    {
        if (c[i] != 0)//如果c[i]!=0
            flag = 1;//flag=1
        if (flag)//否则
            cout << c[i];//直接输出
    }
    system("pause");
    return 0;//结束
}

复制版:

#include <bits/stdc++.h>
using namespace std;
int a[2000];
int b[2000];
int c[2000];
int sum[2000];
void pplus(int *a, int *c)
{
    int jw = 0;
    for (int i = 1; i <= 1000; i++)
    {
        c[i] += a[i] + jw;
        jw = c[i] / 10;
        c[i] %= 10;
    }
}
void cheng(int *a, int c)
{
    int jw = 0;
    for (int i = 1; i <= 1000; i++)
    {
        a[i] = a[i] * c + jw;
        jw = a[i] / 10;
        a[i] %= 10;
    }
}
int main()
{
    int n;
    cin >> n;
    a[1] = 1;
    for (int i = 1; i <= n; i++)
    {
        cheng(a, i);
        pplus(a, c);
    }
    bool flag = 0;
    for (int i = 1000; i >= 1; i--)
    {
        if (c[i] != 0)
            flag = 1;
        if (flag)
            cout << c[i];
    }
    system("pause");
    return 0;
}
  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
过河卒是一个典型的动态规划问题。首先,我们将整个棋盘看作一个二维数组,数组的每个元素表示到达该位置的路径数目。然后,我们根据题目给出的条件,逐步更新数组中的元素,直到计算出到达目标位置的路径数目。 具体的解题思路如下: 1. 首先,我们可以将马的位置设置为0,表示无法经过该位置。 2. 然后,我们根据马的位置,更新数组中的元素。对于二维数组中的每个位置,我们根据左边和上边的位置来计算到达当前位置的路径数目。具体地,如果左边和上边的位置都可以经过,那么到达当前位置的路径数目就等于左边和上边位置的路径数目之和。如果左边或上边的位置无法经过,那么到达当前位置的路径数目就等于左边或上边位置的路径数目。 3. 最后,我们输出目标位置的路径数目。 下面是洛谷p1002过河卒题解C++代码: ```cpp #include <bits/stdc++.h> using namespace std; int main() { long long a[21][21]; int x1, y1, x2, y2; cin >> x1 >> y1 >> x2 >> y2; // 初始化数组,马的位置设置为0 for(int i=0; i<=20; i++) { for(int k=0; k<=20; k++) { a[i][k] = 1; } } a[x2][y2] = 0; // 根据马的位置更新数组中的元素 if(x2 >= 2 && y2 >= 1) a[x2-2][y2-1] = 0; if(x2 >= 1 && y2 >= 2) a[x2-1][y2-2] = 0; if(x2 <= 18 && y2 >= 1) a[x2+2][y2-1] = 0; if(x2 <= 19 && y2 >= 2) a[x2+1][y2-2] = 0; if(x2 >= 2) a[x2-2][y2+1] = 0; if(x2 >= 1) a[x2-1][y2+2] = 0; if(y2 >= 1) a[x2+2][y2-1] = 0; if(y2 >= 2) a[x2+1][y2-2] = 0; // 动态规划计算路径数目 for(int i=1; i<=20; i++) { for(int k=1; k<=20; k++) { if(a[i][k] != 0) { a[i][k] = a[i-1][k] + a[i][k-1]; } } } // 输出目标位置的路径数目 cout << a[x1][y1] << endl; return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Mine Shin

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值