T1 音量调节
问题描述
一个吉他手准备参加一场演出。他不喜欢在演出时始终使用同一个音量,所以他决定每一首歌之前他都要改变一次音量。在演出开始之前,他已经做好了一个列表,里面写着在每首歌开始之前他想要改变的音量是多少。每一次改变音量,他可以选择调高也可以调低。
音量用一个整数描述。输入文件中给定整数beginLevel,代表吉他刚开始的音量,以及整数maxLevel,代表吉他的最大音量。音量不能小于0也不能大于maxLevel。输入文件中还给定了n个整数c_1,c_2,…,c_n,表示在第i首歌开始之前吉他手想要改变的音量是多少。
吉他手想以最大的音量演奏最后一首歌,你的任务是找到这个最大音量是多少。
输入
第一行依次为三个整数:n, beginLevel, maxlevel。
第二行依次为n个整数:c_1,c_2,…,c_n。
输出
输出演奏最后一首歌的最大音量。如果吉他手无法避免音量低于0或者高于maxLevel,输出-1。
样例1
输入
3 5 10
5 3 7
输出
10
样例2
输入
4 8 20
15 2 9 10
输出
-1
数据规模
1≤n≤50,1≤c_i≤maxLevel,1≤maxLevel≤1000, 0≤beginLevel≤maxLevel。
到达型的01背包问题
dp[i][j]:前i首歌曲能否达到音量j,dp[i][j]=0不能达到,dp[i][j]=1表示可以达到
音量调高表示取第i件物品,音量调低表示不取第i件物品
音量为背包容量,01背包模板题(调高调低带约束)
初始条件:dp[0][beginlevel]=1,没演奏前可以到达beginlevel
#include<bits/stdc++.h>
using namespace std;
int n,maxx,begin,a[55],dp[55][1055],qj,sum1,sum2;
int main()
{
cin>>n>>begin>>maxx;
sum1=begin;
sum2=begin;
for(int i=1;i<=n;i++)
{
cin>>a[i];
}
memset(dp,0,sizeof(dp));
dp[0][begin]=1;
for(int i=1;i<=n;i++)
{
for(int j=maxx;j>=0;j--)
{
if(j>=a[i]&&dp[i-1][j-a[i]])
dp[i][j]=1;
if(j+a[i]<=maxx&&dp[i-1][j+a[i]])
dp[i][j]=1;
}
}
for(int i=maxx;i>=0;i--)
{
if(dp[n][i]==1)
{
cout<<i;
return 0;
}
}
cout<<-1;
return 0;
}
T2 蛋糕塔
【问题描述】
Hl高中要举行一场蛋糕塔比赛。注意,不是蛋糕比赛,而是蛋糕塔比赛。
学校会提供N种不同类型的蛋糕,第i种蛋糕的高度为Hi(5 <= H_i <= T),营养价值为Vi(1 <= Vi<= 1,000,000),并且保证所有蛋糕的高度为5的整数倍,每种类型的蛋糕没有数量限制。
蛋糕塔比赛的规则就是要求按照提供的蛋糕,垒成一个高度不超过T(1 <= T <= 1,000)的蛋糕塔,并且要求这个蛋糕塔所有蛋糕的营养价值累加和最高。
因为蛋糕不是很结实,参加比赛的小x发现一个现象,如果某块蛋糕的高度超过K,那么这块蛋糕下面的所有蛋糕的高度都将被压缩为自己高度的4/5,但是营养价值不会丢失。发现这个情况后的小x很兴奋,现在他想知道,如何安排自己的蛋糕塔,能让营养价值最高。
【输入】
第一行三个整数:N,T和K
接下来N行,每行两个整数:Vi 和 Hi。
【输出】
一个整数,表示蛋糕塔最大营养价值。
【输入输出样例1】
cheese.in cheese.out
3 53 25
100 25
20 5
40 10 240
【样例解释】
有3种蛋糕,蛋糕塔的高度限制为53,高度必须超过25的蛋糕才能将其下面的蛋糕高度压缩。我们按照下面的方法来垒蛋糕:
个数 高度 价值 高 -> [1] 25 100 [2] 4 20 [3] 8 40 [3] 8 40 底 -> [3] 8 40
最上的蛋糕将下方的蛋糕都压缩为4/5.最大价值为240
【数据范围】
40% 数据保证 N<=20
100% 数据保证 N<=100 其他数据看题目描述
我们先假设没有压扁,处理一次完全背包,得到一个最优解。
我们再处理压扁,如果要压扁,必然会把高度大于k的那个蛋糕放在最上面。我们枚举放在最上面 的蛋糕,然后对剩下的高度进行完全背包,再求最优值。当然,这里所有蛋糕的高度都被压扁了。。
这样问题得到解决。
这#include<bits/stdc++.h>
using namespace std;
int sum,n,t,k,vi[10500],hi[10500],dp[10500]={};
int main()
{
cin>>n>>t>>k;
for(int i=1;i<=n;i++)
{
cin>>vi[i]>>hi[i];
}
for(int i=1;i<=n;i++)
{
for(int j=hi[i];j<=t+t;j++)
{
dp[j]=max(dp[j],dp[j-hi[i]]+vi[i]);
}
}
sum=dp[t];
for(int i=1;i<=n;i++)
{
if(hi[i]>=k)
sum=max(sum,vi[i]+dp[(t-hi[i])*5/4]);
}
cout<<sum;
return 0;
}
T3 格斗
【问题描述】
格斗俱乐部是格斗爱好者的一个组织,在这里,格斗者们能通过与别的成员进行格斗来释放自己的压力与轻松自己的情绪。最近俱乐部举行了一场比赛,该比赛有N位选手参加,他们将围成一个圆圈,每一场比赛圈内任意的两位相邻的选手均可进行相互的格斗,胜利者将留在圈内进入下轮比赛而失败者则直接被送往医院(没有平局)。比赛是残酷的,最后圈内将只剩下一位选手,他将是总冠军。
我们做个奇怪的假设,两位选手进行格斗,他们比赛的结果总是确定的。虽然俱乐部的成员们都很喜欢格斗,但是他们仍然很希望能获得总冠军。现在你通过统计已经知道了任意两位选手格斗的结果,你有责任告诉每位选手,如果赛程合适安排的话,他是否可能成为总冠军。
【输入】
数据第一行是一个整数N,(1<=N<=40),表示比赛的选手数量。
接下来给出一个N*N的“0”、“1”矩阵A(行内用空格隔开),第i行第j列为 1表示选手i能战胜选手j,否则选手j能战胜选手i。
你可以假定Aij与Aji(i≠j)均是不同的且Aii=0。比赛开始时所有选手按顺时针方向由编号1到编号N站成一个圈,初始时编号1与编号N的选手是相邻的。
【输出】
输出包含N行,每行为一个整数“0”或“1”,“1”表示第i号选手有可能成为冠军,“0”表示不可能。
【输入输出样例1】
data.in
3
0 1 1
0 0 1
0 0 0
data.out
1
0
0
这题没有做过
大佬博客:https://blog.csdn.net/Algor_pro_king_John/article/details/69249109
T4 lcs
没做过
Good Game
OlccK