Doing Homework HDU - 1074 状态压缩

#include<iostream>
#include<cstring>
#include<cstdio>
#include<string>
#include<cmath>
using namespace std;
const int INF=0x3f3f3f3f;
const int MAXN=(1<<15)+5;
const int MOD=1e9+7;
int dead[20],cost[20];
int dp[MAXN],t[MAXN];
char s[20][105];
int pre[MAXN];
void print(int x)
{
    if(x==0)
        return;
    print(x^(1<<pre[x]));
    printf("%s\n",s[pre[x]] );
}

int main()
{
    int T;
    scanf("%d",&T);
    while(T--)
    {
        memset(pre,0,sizeof pre);
        int n;
        scanf("%d",&n);
        for(int i=0; i<n; i++)
            scanf("%s%d%d",s[i],&dead[i],&cost[i] );
        for(int i=1; i<(1<<n); i++)
        {
            dp[i]=INF;
            for(int j=n-1; j>=0; j--)
            {
                int tmp=1<<j;
                //如果没有做
                if(! (i&tmp) )
                    continue;
                //做这个题之后的分数+花费-期限=扣得分
                int score=t[i^tmp]+cost[j]-dead[j];
                //如果小于,说明不扣分
                if(score<0)
                    score=0;
                //如果当前的状态扣的分数   >  做这个题之后扣的分
                if(dp[i]>dp[i^tmp]+score)
                {
                    //分数
                    dp[i]=dp[i^tmp]+score;
                    //花的时间
                    t[i]=t[i^tmp]+cost[j];
                    //转移状态,这个状态下做的题
                    pre[i]=j;
                }
            }
        }
        printf("%d\n", dp[(1<<n)-1]);
        print((1<<n)-1);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/QingyuYYYYY/p/12238191.html

时间: 2024-08-28 09:06:03

Doing Homework HDU - 1074 状态压缩的相关文章

hdu 1074 状态压缩

http://acm.hdu.edu.cn/showproblem.php?pid=1074 我们可以断定状态的终止态一定是n个数全部选完的情况,那么它的前一个状态是什么呢,一定是剔除任一门课程后的n种状态. 例如 dp[全选了]=min{(dp[除了math都做了]+math的超时天数),(dp[除了computer都做了]+computer的超时天数),(dp[除了english都做了]+english的超时天数)}那么接下来的dp状态依然如此. 好了,接下来,我们该如何去思考了,这个题目共

HDU 1074 状态压缩DP 第一次写 多多指教

给你n个数表示有n门作业,下面n行  每行三个数 分别为学科名 截止时间  需要多久才能完成 如果逾期一天则扣掉一学分, 要你求出完成所有作业而被扣最小的学分, 并将完成作业的顺序输出.(时间相同时 按照单词字典数输出) 学分相同时按照字典数输出   目前对于从n-1到0不是很理解 #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> #include<map

hdu 5180 状态压缩 dp 打表

hdu 5180 状态压缩 dp 打表 题意: 在n*n的国际象棋中,放置若干个国王和k个车,使得国王之间不互相攻击,车之间不互相攻击,车不可攻击到国王(这并不代表国王不能攻击到车).国王能攻击到它上下左右,左上左下右上右下八个位置的棋子,车可以攻击到同一行或同一列中的棋子,求方案总数对1000000007取模后的值. 限制: 1 <= n <=15; 0 <= k <=15 思路: 状态压缩,dp,打表套打表 打表程序如下: 打表程序1: tab[a][b]表示a*b的棋盘王的放

hdu 4917Permutation(状态压缩DP)

hdu 4917Permutation(状态压缩DP) 题意:将1~n的n个数排列成序列(n<=40),但有m(m<=20)个限制条件,其中第i个限制条件的表示为ai,bi.表示该序列的第ai的数要小于第bi的.问有多少中排列?保证有解 解法:我们首先可以明确一点,这m个限制条件,所表示的关系会构成若干个DAG(有向无环图,我将其称之为拓扑图).我们只要将这n个数,填入到拓扑图上,使其满足拓扑关系,那么这样的序列就是可以的.而这若干个拓扑图之间,是不会相互影响的,因而我们可以单独考虑每一个拓扑

HDU 1074 Doing Homework(DP&#183;状态压缩)

题意  有n个作业要做   给你每个作业的最后期限  和做完这个作业需要的时间  作业每超过最后期限一天就会扣一分  只能把一个作业做完了再做另一个作业  问做完所有作业至少扣多少分 作业最多只有15个  看到这个数字容易想到是状态压缩  dp[i]表示i对应状态的最小扣分  i转换为二进制后为1的位表明该位对应的作业已经做了  为0的位没做  那么dp[i] = min{dp[k] + cost | k为将某一位变成1后等于 i 的状态} 由于要打印路径  所有还需要记录每个状态的上一个状态p

Doing Homework HDU - 1074

Ignatius has just come back school from the 30th ACM/ICPC. Now he has a lot of homework to do. Every teacher gives him a deadline of handing in the homework. If Ignatius hands in the homework after the deadline, the teacher will reduce his score of t

[HDU]5094Maze(状态压缩BFS)

状态压缩的题,第一次WA了,怎么改都不对,交了十几遍,之后重新写了一个,1A了 #include<cstdio> #include<queue> #include<cstring> #include<algorithm> using namespace std; const int maxn = 51; const int dir[4][2] = {{0,1},{-1,0},{0,-1},{1,0}}; int n,m,t; int door[maxn][m

hdu 3006(状态压缩)

The Number of set Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 1301    Accepted Submission(s): 795 Problem Description Given you n sets.All positive integers in sets are not less than 1 and n

hdu 5135 状态压缩dp

题意是给出最对12条边你     问最大可组成三角形面积是多大 (可有多个三角形)   典型的状态压缩  (最多12条边) 思路: 先暴力枚举所有可能三角形的面积   然后再根据状态压缩来做    最多可组成n/3个三角形  对每个三角形都记录三边或(运算)    和面积    然后判断进行合并  (根据且运算)   就行了 #include<stdio.h> #include<string.h> #include<cmath> #include<iostream