http://acm.hdu.edu.cn/showproblem.php?pid=1789

这是道贪心题:虽说简单。但是对于刚接触的人来讲,看懂都有点困难,因为不知道从哪下手!就拿我来讲吧!就是这样的概念。还是找到学长讲解了一下,才略懂点。这不就赶紧把思路整理下。

讲到这道题就必须说下题意,为什么n??因为当时我就因为题意不明确,才看别人的代码好模糊。

题意是:给的第一个数;测试n 组数据,接下来一行就是m ,指的是有m们功课。接下来的两行就是每门功课必须完成的期限,和每门功课完成后的分数。要求就是要我们在规定期限内完成所学的功课。(每门功课都需要一天来完成)。还有一层意思就是:就算一天完成一门,也必须在规定时间内完成所有功课。

举个例子:例如第一组数据;每门功课的期限是3天,因此是可以完成的,为啥呢么??因为每门功课要一天完成,有三门功课,因此一天完成一门,所以刚好三天就完成了。所得的答案也就没有罚分。这里题目要求得就是:最小的罚分是多少??

讲到这里大概题目意思懂了吧??呵呵,,懂了意思就好办!接下来就是贴代码了::

详情请看代码:(我当初也是看别人的思路,然后在敲的);

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
struct point
{
    int day;
    int score;
};
bool camp(const point a,const point b)
{
   // if(a.score!=b.score)
   // {
        return a.score>b.score;
   // }
   // else
   // return a.day<b.day;//这里为什么可以不排序呢??因为是定义了结构体,只要分数排序了,则分数对应的时间也会变化。。
}
int main()
{
    int i,j,t,n;
    scanf("%d",&t);
    while(t--)
    {
        point s[10005];
          int flag[10005]={0};
          int ans=0;
          scanf("%d",&n);
          for(i=0;i<n;i++)
          {
              scanf("%d",&s[i].day);
          }
          for(i=0;i<n;i++)
          {
              scanf("%d",&s[i].score);
          }
          sort(s,s+n,camp);
          int t;
          for(i=0;i<n;i++)
          {
              t=s[i].day;
            while(flag[t]) t--;
            if(t==0) ans+=s[i].score;
            else
                flag[t]=1;
          }
          printf("%d\n",ans);
    }
    return 0;

}

 

这段代码是解决一个问题的递推算法。问题的描述是:给定一个闭区间 [m, n],其中 m 和 n 是两个正整数,统计该区间内满足特定条件的数字个数。 首先,代码中的 init 函数用于初始化一个二维数组 dp。dp[i][j] 表示第一位为 j 的 i 位数中满足条件的数字数量。初始化时,将 dp[0][0] 设置为 1,表示只有一个位数且为 0,满足条件的数字个数为 1。 然后,通过嵌套循环来计算 dp 数组的其他元素。外层循环遍历位数 i,内层两个循环遍历第 i 位数的可能取值 j 和前一位数的可能取值 k。在遍历过程中,根据特定条件判断,如果满足条件,则将 dp[i][j] 累加上 dp[i-1][k] 的值。 接下来,solve 函数用于计算闭区间 [0, n) 中满足条件的数字个数。首先,将数字 n 拆分成位数,并保存在 digit 数组中。然后,从高位到低位遍历 digit 数组。对于第 i 位数 digit[i],通过嵌套循环来计算满足条件的数字个数。内层循环遍历从 0 到 digit[i]-1 的可能取值 j,根据特定条件判断,如果满足条件,则将答案 ans 加上 dp[i][j] 的值。 在循环过程中,如果第 i 位数 digit[i] 等于 4,则表示以 4 开头的数字后面的数字不满足条件,因此可以直接跳出循环。如果第 i 位数 digit[i] 等于 2 且下一位数 digit[i+1] 等于 6,则表示以 62 开头的数字后面的数字也不满足条件,可以直接跳出循环。 最后,在主函数中,通过调用 init 函数来初始化 dp 数组。然后,通过循环读入输入的 m 和 n 的值,直到 m 和 n 都为 0 时结束循环。在每次循环中,计算闭区间 [m, n] 内满足条件的数字个数,即 solve(m+1) - solve(n),并输出结果。 这段代码利用动态规划的思想,通过递推关系计算满足条件的数字个数,从而高效地解决了给定的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值