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

先求出每个数的得分情况,分数和得分状态,(1<<4)种状态

按分数从大到小排序 然后每种状态取一个数(如果有的话)

然后对 dp[i][j] 进行背包 dp[i][j] 表示的是选了i个数选的总状态为j情况下的最大值

然后根据每个 dp[i][j] 对数组剩余的数进行最优选择(在不破坏 最终状态 j 的情况下,尽量选单位分数高的)

最后求最大的情况

代码:

#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<cmath>
#include<set>
#include<map>
#include<stack>
#include<vector>
#include<algorithm>
#include<queue>
#include<stdexcept>
#include<bitset>
#include<cassert>
#include<deque>
#include<numeric> using namespace std; typedef long long ll;
typedef unsigned int uint;
const double eps=1e-12;
const int INF=0x3f3f3f3f;
const ll MOD=1000000007;
const int H=1000005;
const int K=10005;
const int N=1005;
const int M=(1<<4);
struct node
{
int a,b;
int point,k;
}in[N];
bool prime(int x)
{
if(x==1) return false;
for(int i=2;i*i<=x;++i)
if(x%i==0)
return false;
return true;
}
void get(node &x)
{
int a=x.a;
int num=0,sum=0;
for(int i=1;i*i<=a;++i)
if(a%i==0)
{
++num;
sum+=i;
int j=a/i;
if(i!=j)
{++num;sum+=j;}
}
x.k=0;x.point=0;
if(num==2)
{x.point++;x.k|=1;}
if(prime(num))
{x.point++;x.k|=2;}
if(prime(sum))
{x.point++;x.k|=4;}
int h=(int)(sqrt(1.0*a)+0.5);
int h1=(int)(sqrt(1.0*h)+0.5);
if(a==1||((num&1)==0&&((num>>1)&1)==0)||((num&1)==1&&h1*h1==h))
{x.point++;x.k|=8;}
}
bool cmp(node x,node y)
{
return x.point>y.point;
}
int main()
{
//freopen("data.in","r",stdin);
//freopen("1011.in","r",stdin);
//freopen("1011.out","w",stdout);
int T;
scanf("%d",&T);
while(T--)
{
int n,k;
scanf("%d %d",&n,&k);
for(int i=0;i<n;++i)
{
scanf("%d %d",&in[i].a,&in[i].b);
get(in[i]);
if(i>0) printf(" ");
printf("%d",in[i].point);
}
printf("\n");
int bit[4];
for(int i=0;i<4;++i)
scanf("%d",&bit[i]);
sort(in,in+n,cmp);
int dp[20][20];
memset(dp,-1,sizeof(dp));
dp[0][0]=0;
bool had[20];
memset(had,false,sizeof(had));
for(int i=0;i<n;++i)
if(had[in[i].k]==false)
{
had[in[i].k]=true;
node &w=in[i];
for(int i=M-1;i>=0;--i)
for(int j=0;j<M;++j)
if(dp[i][j]!=-1)
{
int l=i+1;
int r=(j|w.k);
dp[l][r]=max(dp[l][r],dp[i][j]+w.point);
}
w.b--;
}
int ans=-INF;
for(int i=0;i<=16;++i)
for(int j=0;j<16;++j)
if(i<=k&&dp[i][j]!=-1)
{
int sum=dp[i][j];
for(int l=0;l<4;++l)
if((j&(1<<l))==0)
sum+=bit[l];
int w=k-i;
for(int l=0;w>0&&l<n;++l)
if((j|in[l].k)==j)
{
if(w>=in[l].b)
{
w-=in[l].b;
sum+=(in[l].b*in[l].point);
}else
{
sum+=(w*in[l].point);
w=0;
}
if(w==0)
break;
}
if(w==0)
ans=max(ans,sum);
}
printf("%d\n",ans);
}
return 0;
}

hdu 4610 Cards的更多相关文章

  1. HDU 4610 Cards (合数分解,枚举)

    Cards Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submi ...

  2. 【 2013 Multi-University Training Contest 1 】

    HDU 4602 Partition f[i]表示和为i的方案数.已知f[i]=2i-1. dp[i]表示和为i,k有多少个.那么dp[i]=dp[1]+dp[2]+...+dp[i-1]+f[i-k ...

  3. HDU 4876 ZCC loves cards(暴力剪枝)

    HDU 4876 ZCC loves cards 题目链接 题意:给定一些卡片,每一个卡片上有数字,如今选k个卡片,绕成一个环,每次能够再这个环上连续选1 - k张卡片,得到他们的异或和的数,给定一个 ...

  4. hdu 4876 ZCC loves cards(暴力)

    题目链接:hdu 4876 ZCC loves cards 题目大意:给出n,k,l,表示有n张牌,每张牌有值.选取当中k张排列成圈,然后在该圈上进行游戏,每次选取m(1≤m≤k)张连续的牌,取牌上值 ...

  5. HDU 1535 Invitation Cards(逆向思维+邻接表+优先队列的Dijkstra算法)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1535 Problem Description In the age of television, n ...

  6. POJ 1511 Invitation Cards / UVA 721 Invitation Cards / SPOJ Invitation / UVAlive Invitation Cards / SCU 1132 Invitation Cards / ZOJ 2008 Invitation Cards / HDU 1535 (图论,最短路径)

    POJ 1511 Invitation Cards / UVA 721 Invitation Cards / SPOJ Invitation / UVAlive Invitation Cards / ...

  7. HDU 1535 Invitation Cards(最短路 spfa)

    题目链接: 传送门 Invitation Cards Time Limit: 5000MS     Memory Limit: 32768 K Description In the age of te ...

  8. hdu 1535 Invitation Cards

    http://acm.hdu.edu.cn/showproblem.php?pid=1535 这道题两遍spfa,第一遍sfpa之后,重新建图,所有的边逆向建边,再一次spfa就可以了. #inclu ...

  9. HDU SPFA算法 Invitation Cards

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1535 分析: 题意:求1点到其它点的最短距离之和+其它点到1点的最短距离之和 前面一部分直接用SPFA ...

随机推荐

  1. hostapd源代码分析(一):网络接口和BSS的初始化

    [转]hostapd源代码分析(一):网络接口和BSS的初始化 原文链接:http://blog.csdn.net/qq_21949217/article/details/46004349 最近在做一 ...

  2. Android应用程序构成

    一个Android应用程序一般是由以下4个组件构成的: 活动(Activity) 意图(Intent) 服务(Service) 内容提供器(Content Provider) 这4个组件是构成andr ...

  3. FragmentPagerAdapter与FragmentStatePagerAdapter区别

    在一个 Android 应用中,我使用 FragmentPagerAdapter 来处理多 Fragment 页面的横向滑动.不过我碰到了一个问题,即当 Fragment 对应的数据集发生改变时,我希 ...

  4. GO语言中间的derfer

    defer Go语言中有种不错的设计,即延迟(defer)语句,你可以在函数中添加多个defer语句.当函数执行到最后时,这些defer语句会按照逆序执行, 最后该函数返回.特别是当你在进行一些打开资 ...

  5. JavaScript 同名方法的处理

    在JS中,如果存在同名同参的方法,它会先调用哪一个?先看两个例子: 例1: <html> <head> <title></title> <scri ...

  6. [转]Material Design Library 23.1.0的新变化与代码实战

    Design Library出来已经快有一个月了,当时大概看了一下介绍这个新版本变化的译文,内容不多,给我印象最深的就是Percent lib.AppBarLayout 和NavigationView ...

  7. 求出数组前面k个元素或数组中元素大于一半的元素(快速排序与堆排序的灵活运用)

    写这个的目的在于,说明快速排序的灵活运用.我们来看下关于快速排序中的一部分关键代码: 快速排序代码: int a[101],n;//定义全局变量,这两个变量需要在子函数中使用 void quickso ...

  8. 扩展Date的DateDiff方法--日期差

    Date.prototype.DateDiff = function(after){ var diffDay; var beforeDate = new Date(this).format(" ...

  9. 【CITE】DrawImage方法详解(转)

    Image和Bitmap类概述 GDI+的Image类封装了对BMP.GIF.JPEG.PNG.TIFF.WMF(Windows元文件)和EMF(增强WMF)图像文件的调入.格式转换以及简单处理的功能 ...

  10. 纯css3代码写无缝滚动效果

    主要用到css3中的动画 @keyframes, animation. 布局是外层一个div宽固定,然后overflow hidden 绝对定位,里面的ul 固定定位.通过对ul添加动画来实现效果.具 ...