NOIP2006题解

本文详细解析了三道算法竞赛题目:能量项链问题通过动态规划求解最大能量和;金明的预算方案采用依赖背包算法优化;作业调度方案则通过直接模拟来解决。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

能量项链:
题目大意:有n个珠子串成一个环,每个珠子有头标记和尾标记,每次可以合并任意相邻的两颗珠子i,j,所得的能量为head[i]*head[j]*tail[j],求将n颗珠子合并成一颗的最大能量和。n<=100.
题解:
对于有环的题目,我们一般先拆环为链,变成一条长度为2n的链。然后我们发现最终答案就是在这条长度为2n的链上找到一段长度为n的区间进行操作使得能量和最大。不难想到对于长度为n的区间我们可以看作是找到一段长度为n-1的区间进行操作使得能量和最大,再将这个能量和加上最后两颗珠子合并所得到的能量。同理可以推出n-1,n-2,n-3….2。典型的具有无后效性,于是我们采取DP解题。
由于我们刚才推出的结论与长度有关,因此我们将长度作为第一维循环,于是当我们循环到i时,我们必定知道1~i-1时的每个最优值,于是就转化为经典的区间DP了。
设f[j][j+i-1]代表从第j位到第j+i-1位的最大能量和,则
f[j][j+i-1]=max(f[j][k]+f[k+1][j+i-1]+a[j]*a[k+1]+a[j+i])(j<=k)

#include<cstdio>
#include<algorithm>
using namespace std;
int n,i,j,k,f[210][210],a[210],ans;
int main(){
    scanf("%d",&n);
    for(i=1;i<=n;i++)scanf("%d",&a[i]);
    for(i=n+1;i<=n*2;i++)a[i]=a[i-n];
    for(i=2;i<=n;i++)
        for(j=1;j<=2*n-i;j++)
            for(k=j;k<j+i-1;k++)
                f[j][j+i-1]=max(f[j][j+i-1],f[j][k]+f[k+1][j+i-1]+a[j]*a[k+1]*a[j+i]);
    for(i=1;i<=n;i++)ans=max(ans,f[i][i+n-1]);
    printf("%d",ans);
    return 0;
}

金明的预算方案:
题目大意:有n个物品,每个物品有一个价格和重要度,并且有可能有附件(即买了该物品才能买附件),求用不超过m的钱使得所有购买物品的价格*重要度之和最大,n<=60,m<=32000,所有价格(包括m)均为10的倍数,每个物品最多为两个物品的必需品。
题解:
显然可以看出这是一道裸的背包,因为有附件的存在,我们可以用树依赖背包或者普通的带依赖品的背包做,由于这里每个物品最多为两个物品的必需品,我们可以选择用普通的依赖背包做。
由于所有价格都是10的倍数,我们在DP的时候先把所有价格都除以10,最后输出的时候将答案乘10即可。
其实所谓的带依赖品的背包就是分情况讨论一下……分当前物品有一个附件,当前物品有两个附件,和当前物品没有附件三种情况讨论,然后就按普通的01背包做即可。由于找到该物品的附件需要O(n),复杂度太高(虽然这题n才60,是可以过去的),我们选择离散化。时间复杂度:O(n*m/10),空间复杂度:O(m/10)。

#include<cstdio>
#include<algorithm>
using namespace std;
int n,m,v[70][10],p[70][10],a,f[3300],i,j,k,w[70],b,c,cnt,xh[70];
int main(){
    scanf("%d%d",&m,&n);m/=10;
    for(i=1;i<=n;i++){
        scanf("%d%d%d",&a,&b,&c);
        if(c)v[xh[c]][++w[xh[c]]]=a/10,p[xh[c]][w[xh[c]]]=b;
        else v[++cnt][0]=a/10,p[cnt][0]=b,xh[i]=cnt;
    }
    for(i=1;i<=cnt;i++)
        for(k=m;k>=v[i][0];k--){
            f[k]=max(f[k],f[k-v[i][0]]+v[i][0]*p[i][0]);
            if(w[i]>=1){
                if(k>=v[i][0]+v[i][1])f[k]=max(f[k-v[i][0]-v[i][1]]+v[i][0]*p[i][0]+v[i][1]*p[i][1],f[k]);
            }
            if(w[i]>=2){
                if(k>=v[i][0]+v[i][2])f[k]=max(f[k],f[k-v[i][0]-v[i][2]]+v[i][0]*p[i][0]+v[i][2]*p[i][2]);
                if(k>=v[i][0]+v[i][1]+v[i][2])f[k]=max(f[k],f[k-v[i][0]-v[i][1]-v[i][2]]+v[i][0]*p[i][0]+v[i][1]*p[i][1]+v[i][2]*p[i][2]);
            }
        }
    printf("%d",f[m]*10);
    return 0;
}

作业调度方案:
题目大意:给你n项作业,m台机器,每项作业都有m个部分,每项作业的每个部分都必须在规定的机器上完成,每项作业的每个部分需要花费一定的时间,每项作业的第i个部分当且仅当第i-1个部分完成了才可进行。每项作业必须安排在规定的机器上可用的时间段中最早的一段,求这么安排的总时间。
题解:
直接模拟即可,它说什么就做什么。

#include<cstdio>
#include<algorithm>
using namespace std;
int n,m,i,j,b[50],jiqi[50][50],shij[50][50],xh[410],cnt[410],k,ans;
bool f[50][300],flag;
int main(){
    scanf("%d%d",&m,&n);
    for(i=1;i<=n*m;i++)
    scanf("%d",&xh[i]);
    for(i=1;i<=n;i++)
        for(j=1;j<=m;j++)scanf("%d",&jiqi[i][j]);
    for(i=1;i<=n;i++)
        for(j=1;j<=m;j++)scanf("%d",&shij[i][j]);
    for(i=1;i<=n*m;i++){
        cnt[xh[i]]++;
        for(j=b[xh[i]]+1;;j++){
            flag=1;
            for(k=j;k<=j+shij[xh[i]][cnt[xh[i]]]-1;k++)
                if(f[jiqi[xh[i]][cnt[xh[i]]]][k]){
                    flag=0;
                    break;
                }
            if(flag){
                for(k=j;k<=j+shij[xh[i]][cnt[xh[i]]]-1;k++)f[jiqi[xh[i]][cnt[xh[i]]]][k]=1;
                b[xh[i]]=j+shij[xh[i]][cnt[xh[i]]]-1;
                ans=max(ans,j+shij[xh[i]][cnt[xh[i]]]-1);
                break;
            }
        }
    }
    printf("%d",ans);
    return 0;
}

2k进制数
设r是个2k进制数,并满足以下条件:
1.r至少是个2位的2k进制数。
2.作为2k进制数,除最后一位外,r的每一位严格小于它右边相邻的那一位。
3.将r转换为2进制数q后,则q的总位数不超过w。
在这里,正整数k(1≤k≤9)和w(k< w≤30000)是事先给定的。
问:满足上述条件的不同的r共有多少个?
题解:
首先,我们需要把问题的条件转换一下:
1.r最多有w/k+1位。
2.r的取值方法相当于组合数(每一位严格小于它右边相邻的那一位相当于组合数)
所以我们可以发现这么一件事:
当我们确定r的第一位为i时,即可确定答案为C(2^k-1-i,w/k)。
而r的第一位最大值为2^(w%k)-1,所以累加就可以得到当r有w/k+1位时的方案数。
显然只要再求得r为2~w/k位时的方案数总和即可求得答案。
那么当r为i位时(i<=w/k),即是2^k-1个数中取i个数,即C(2^k-1,i),累加即可。
两个方案数累加即为答案,注意要用高精度。

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int n,i,j,c[210],f[520][210],k,w,f1[520][210],ans[210],m,mm;
void Plus(int a[],int b[]){
    a[0]=max(a[0],b[0]);
    for(int i=1;i<=a[0];i++){
        a[i]+=b[i];
        a[i+1]+=a[i]/10;
        a[i]%=10;
    }
    if(a[a[0]+1])a[0]++;
    while(a[a[0]]>9)a[a[0]+1]+=a[a[0]]/10,a[a[0]]%=10;
}
int main(){
    scanf("%d%d",&k,&w);
    m=w%k;
    mm=1<<m;
    n=1<<k;
    if(m)mm--;
    else mm=0;
    n--;
    f1[0][0]=f1[0][1]=1;
    for(i=1;i<=n-mm-1;i++){
        memset(f,0,sizeof(f));
        f[0][0]=f[0][1]=1;
        for(j=1;j<=i;j++)Plus(f[j],f1[j]),Plus(f[j],f1[j-1]);
        memcpy(f1,f,sizeof(f));
    }
    for(;i<n;i++){
        memset(f,0,sizeof(f));
        f[0][0]=f[0][1]=1;
        for(j=1;j<=i;j++)Plus(f[j],f1[j]),Plus(f[j],f1[j-1]);
        Plus(ans,f[min(n,w/k)]);
        memcpy(f1,f,sizeof(f));
    }
    memset(f,0,sizeof(f));
    f[0][0]=f[0][1]=1;
    Plus(f[1],f1[1]),Plus(f[1],f1[0]);
    for(j=2;j<=min(w/k,n);j++){
        Plus(f[j],f1[j]),Plus(f[j],f1[j-1]);
        Plus(ans,f[j]);
    }
    if(!ans[0])putchar('0');
    else for(i=ans[0];i;i--)putchar(ans[i]+'0');
    return 0;
}
### NOIP2024 T1 题解编程竞赛解题思路 #### 背景介绍 NOIP(全国青少年信息学奥林匹克联赛)作为国内重要的计算机科学赛事之一,旨在选拔优秀的程序设计人才。每年的比赛都会设置不同难度级别的题目来测试参赛者的算法能力和编程技巧。 #### 解析与策略制定 对于NOIP2024的第一道题目而言,通常这类题目会偏向基础概念的理解和简单应用,目的是让大部分选手能够入手并获得一定分数的同时也筛选出具备更深入思考能力的学生[^4]。 考虑到这一点,在面对T1这样的入门级挑战时,可以采取如下几种常见处理方式: - **直接求解**:如果问题本身相对直观,则可以直接通过观察数据特点找到规律进而得出结论。 - **暴力枚举**:当不确定最优解决方案时,可以通过遍历所有可能情况的方法尝试解决问题,虽然效率较低但对于小规模的数据集仍然适用。 - **模拟过程**:针对一些涉及具体操作流程的问题,按照给定条件逐步模仿实际执行步骤直至达到目标状态。 假设本年度的首题围绕着珠心算测验展开讨论,那么基于以往的经验来看,该类试题往往适合采用枚举的方式进行解答[^2]。下面给出一种具体的实现方案: ```cpp #include <iostream> using namespace std; int main() { int n; cin >> n; // 输入人数n bool flag[n+1]; memset(flag, false, sizeof(flag)); int a[n]; for(int i=1;i<=n;++i){ cin>>a[i]; for(int j=1;j<i;++j) for(int k=j+1;k<i;++k) if(a[j]+a[k]==a[i]) {flag[i]=true;break;} } int cnt=0; for(int i=1;i<=n;++i)if(!flag[i]) ++cnt; cout<<cnt<<"\n"; } ``` 上述代码实现了对每个数是否能由其他两个不同的数组合而成这一性质进行了判断,并统计满足特定条件的数量。这种方法不仅易于理解而且便于编码实现,非常适合初学者练习使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值