zoj3905 Cake
Time Limit: 4 Seconds Memory Limit: 65536 KB
Alice and Bob like eating cake very much. One day, Alice and Bob went to a bakery and bought many cakes.
Now we know that they have bought n cakes in the bakery. Both of them like delicious cakes, but they evaluate the cakes as different values. So they decided to divide those
cakes by following method.
Alice and Bob do n / 2 steps, at each step, Alice choose 2 cakes, and Bob takes the cake that he evaluates it greater, and Alice take the rest cake.
Now Alice want to know the maximum sum of the value that she can get.
Input
The first line is an integer T which is the number of test cases.
For each test case, the first line is an integer n (1<=n<=800). Note that n is always an even integer.
In following n lines, each line contains two integers a[i] and b[i], where a[i] is the value of ith cake that Alice evaluates,
and b[i] is the value of ith cake that Bob evaluates. (1<=a[i], b[i]<=1000000)
Note that a[1], a[2]..., a[n] are n distinct integers and b[1], b[2]..., b[n] are n distinct integers.
Output
For each test case, you need to output the maximum sum of the value that Alice can get in a line.
Sample Input
1
6
1 6
7 10
6 11
12 18
15 5
2 14
Sample Output
28
题意:有n个球,两个人玩游戏,每次选出两个球,B会选出他认为球的价值比较大的,会挑走一个球,然后A会得到另一个球的价值,问怎样取能使A获得的价值最大。
思路:可以先按B所认为的价值从大到小排序,那么对于任意前m件物品,A可以得到m/2件,然后设dp[i][j]表示前i件物品A取出j件最多获得的价值,那么当循环到前第i件时有两种转移形式,一种是这一件不取,那么前i-1件就要取j件,否则就要取j-1件,即dp[i][j]=max(dp[i-1][j],dp[i-1][j-1]+c[i].a);
#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<vector>
#include<map>
#include<set>
#include<queue>
#include<stack>
#include<string>
#include<algorithm>
using namespace std;
#define ll long long
#define inf 0x7fffffff
#define maxn 810
struct node{
int a,b;
}c[maxn];
bool cmp(node a,node b){
return a.b>b.b;
}
int dp[maxn][maxn];
int main()
{
int n,m,i,j,T;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
for(i=1;i<=n;i++){
scanf("%d%d",&c[i].a,&c[i].b);
}
sort(c+1,c+1+n,cmp);
memset(dp,-1,sizeof(dp));
dp[1][0]=0;
for(i=2;i<=n;i++){
dp[i][0]=0;
for(j=1;j<=i/2;j++){
dp[i][j]=max(dp[i-1][j],dp[i-1][j-1]+c[i].a);
}
}
printf("%d\n",dp[n][n/2]);
}
}
还有两种是贪心的解法,一个是从前往后贪心,还有一个是从后往前贪心,两者的本质是一样的。我们考虑从后往前贪心,因为不管A所选择的两个数是什么,都是B先选,所以我们可以先把n个数据按b为第一关键字从大到小排序,按a从大到小排序。那么不管我们取出哪两个数,B都是取排在前面的数,题目等价于有n个数,B按顺序取对其来说价值最大的数,A尽可能多的取得对它来说价值大的数。那么我们考虑B取法的规律,前1个数,B至少取1个,前3个数,B至少取2个,前5个数,B至少取3个....前n(n为奇数)个数,B至少取n/2+1个,所以我们可以推出前1个数必由B取,前3个数A至多取1个,前5个数,A至多取3个...那么我们便可进一步推出,后1个数,A至少取1个,后3个数,A至少取2个,后5个数,B至少取3个,那么我们可以从后往前每次把2个数放入set里,然后把set里最大的数取出,加上该值就行了。
代码一:(从后往前推)
#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<vector>
#include<map>
#include<set>
#include<queue>
#include<stack>
#include<string>
#include<algorithm>
using namespace std;
typedef long long ll;
#define inf 99999999
#define pi acos(-1.0)
#define maxn 1000
struct node1{
int num;
}temp,temp1;
bool operator<(node1 c,node1 d){
return c.num>d.num;
}
set<node1>myset;
set<node1>::iterator it;
struct node{
int a,b;
}c[maxn];
bool cmp(node c,node d){
return c.b>d.b;
}
int main()
{
int n,m,i,j,T;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
for(i=1;i<=n;i++){
scanf("%d%d",&c[i].a,&c[i].b);
}
sort(c+1,c+1+n,cmp);
if(n==2){
printf("%d\n",c[2].a);
continue;
}
int sum=c[n].a;
myset.clear();
for(i=n-1;i>=1;i--){
node1 temp;
temp.num=c[i].a;
myset.insert(temp);
if(i%2==0){
it=myset.begin();
sum+=(*it).num;
myset.erase(*it);
}
}
printf("%d\n",sum);
}
return 0;
}
/*
1
6
1 6
7 10
6 11
12 18
15 5
2 14
*/
代码二:
#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<vector>
#include<map>
#include<set>
#include<queue>
#include<stack>
#include<string>
#include<algorithm>
using namespace std;
typedef long long ll;
#define inf 99999999
#define pi acos(-1.0)
#define maxn 1000
set<int>myset;
set<int>::iterator it;
struct node{
int a,b;
}c[maxn];
bool cmp(node c,node d){
return c.b>d.b;
}
int main()
{
int n,m,i,j,T,sum1;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
sum1=0;
for(i=1;i<=n;i++){
scanf("%d%d",&c[i].a,&c[i].b);
sum1+=c[i].a;
}
sort(c+1,c+1+n,cmp);
if(n==2){
printf("%d\n",c[2].a);
continue;
}
int sum=0;
myset.clear();
for(i=1;i<=n;i++){
myset.insert(c[i].a);
if(i%2==1){
it=myset.begin();
sum+=*it;
myset.erase(it);
}
}
printf("%d\n",sum1-sum);
}
return 0;
}
/*
1
6
1 6
7 10
6 11
12 18
15 5
2 14
*/
zoj3905 Cake的更多相关文章
- Windows 7上执行Cake 报错原因是Powershell 版本问题
在Windows 7 SP1 电脑上执行Cake的的例子 http://cakebuild.net/docs/tutorials/getting-started ,运行./Build.ps1 报下面的 ...
- 2015暑假多校联合---Cake(深搜)
题目链接:HDU 5355 http://acm.split.hdu.edu.cn/showproblem.php?pid=5355 Problem Description There are m s ...
- Scalaz(15)- Monad:依赖注入-Reader besides Cake
我们可以用Monad Reader来实现依赖注入(dependency injection DI or IOC)功能.Scala界中比较常用的不附加任何Framework的依赖注入方式可以说是Cake ...
- uva10167 Birthday Cake
Lucy and Lily are twins. Today is their birthday. Mother buys a birthday cake for them. Now we put t ...
- HDU 4762 Cut the Cake(公式)
Cut the Cake Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Tota ...
- Brute Force --- UVA 10167: Birthday Cake
Problem G. Birthday Cake Problem's Link:http://uva.onlinejudge.org/index.php?option=com_onlinejudg ...
- 2015-2016 ACM-ICPC, NEERC, Southern Subregional Contest, B. Layer Cake
Description Dasha decided to bake a big and tasty layer cake. In order to do that she went shopping ...
- hdu acmsteps 2.1.3 Cake
Cake Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submissi ...
- ZOJ 3905 Cake ZOJ Monthly, October 2015 - C
Cake Time Limit: 4 Seconds Memory Limit: 65536 KB Alice and Bob like eating cake very much. One ...
随机推荐
- 【Vue】Vue框架常用知识点 Vue的模板语法、计算属性与侦听器、条件渲染、列表渲染、Class与Style绑定介绍与基本的用法
Vue框架常用知识点 文章目录 Vue框架常用知识点 知识点解释 第一个vue应用 模板语法 计算属性与侦听器 条件渲染.列表渲染.Class与Style绑定 知识点解释 vue框架知识体系 [1]基 ...
- LeetCode872. 叶子相似的树
题目 1 class Solution { 2 public: 3 vector<int>ans1; 4 vector<int>ans2; 5 bool leafSimilar ...
- C++:I/O流的概念和流类库的结构
一.C++输入输出包含以下三个方面的内容: 对系统指定的标准设备的输入和输出.即从键盘输入数据,输出到显示器屏幕.这种输入输出称为标准的输入输出,简称标准I/O. 以外存磁盘文件为对象进行输入和输出, ...
- Zju1100 Mondriaan
题目描述 有一个m行n列的矩阵,用1*2的骨牌(可横放或竖放)完全覆盖,骨牌不能重叠,有多少种不同的覆盖的方法? 你只需要求出覆盖方法总数mod p的值即可. 输入格式 三个整数数n,m,p,m< ...
- SpringBoot 好“吃”的启动原理
原创:西狩 编写日期 / 修订日期:2020-12-30 / 2020-12-30 版权声明:本文为博主原创文章,遵循 CC BY-SA-4.0 版权协议,转载请附上原文出处链接和本声明. 不正经的前 ...
- 24v转3.3v稳压芯片,高效率DC-DC变换器3A输出电流
PW6206系列是一个高精度,高输入电压低静态电流,高速,低功耗降线性稳压器具有高纹波抑制.输入电压高达40V,负载电流为在VOUT=5V和VIN=7V时高达300mA.该设备采用BCD工艺制造.PW ...
- pytest:通过scope控制fixture的作用范围
一.fixture里面有个参数scope,通过scope可以控制fixture的作用范围,根据作用范围大小划分:session>module>class>function,具体作用范 ...
- Scrapy———反爬蟲的一些基本應對方法
1. IP地址驗證 背景:有些網站會使用IP地址驗證進行反爬蟲處理,檢查客戶端的IP地址,若同一個IP地址頻繁訪問,則會判斷該客戶端是爬蟲程序. 解決方案: 1. 讓Scrapy不斷隨機更換代理服務器 ...
- C++ /Python 将视频中的片段转为图片
配置OpenCV :项目名称->右击->属性 VC++目录 包含目录 放 ...\build\include ...\build\include\opencv ...\build\ ...
- https://hbase.apache.org/devapidocs/org/apache/hadoop/hbase/util/MurmurHash.html
https://hbase.apache.org/devapidocs/org/apache/hadoop/hbase/util/MurmurHash.html https://github.com/ ...