zoukankan      html  css  js  c++  java
  • UVA

    题目大意:有一些老师,每一位都有自己的工资以及教授的课程。共s<=8个课程。其中的一些老师必须选择,问你保证每节课至少有一个老师的最少总工资。

    题解:

    首先很容易想到状态压缩,搞一个3进制的数,分别表示每一门课程的情况,一共38=6561。但是这样是不行的,相当于暴力啊!
    一个套路:三进制转化为二进制*2。也就是搞一个216的数,1~8和9~16表示每门课程,这样就可以利用位运算了。
    然后知道这个就很显然的,一个背包问题。要注意,这样转化为二进制之后,要定义一条规则,每次添加课程,优先1~8位,有了就再加到9~16位即可。
    处理出所有强制选择的老师的授课状态S,令dp[S]=sum{c[]}。其他的全是INF。然后从All=(1<<(s*2)-1到S。当成01背包做即可。

     1 #include<queue>
     2 #include<cstdio>
     3 #include<vector>
     4 #include<cstring>
     5 #include<iostream>
     6 #include<algorithm>
     7 #define RG register
     8 #define LL long long
     9 #define fre(a) freopen(a".in","r",stdin);//freopen(a".out","w",stdout);
    10 using namespace std;
    11 const int MAXN=1000;
    12 int s,n,m,S,sum,All;
    13 int c[MAXN],a[MAXN],dp[1<<16];
    14 char ch;
    15 int main()
    16 {
    17    while(scanf("%d%d%d",&s,&n,&m)!=EOF)
    18       {
    19          if(s==0)break;
    20          All=(1<<(s*2))-1;
    21          for(int i=1,flag,x;i<=n+m;i++)//鬼里鬼气的输入
    22             {
    23                a[i]=0;
    24                scanf("%d",&c[i]);
    25                flag=0;
    26                while(1)
    27                   {
    28                      ch=getchar();
    29                      while(ch<'0'||ch>'9')
    30                         {
    31                            if(ch=='
    '||ch=='
    ') { flag=1; break; }
    32                            ch=getchar();
    33                         }
    34                      if(flag)break;
    35                      x=0;
    36                      while('0'<=ch&&ch<='9')x=x*10+(ch-'0'),ch=getchar();
    37                      a[i]|=(1<<(x-1));
    38                      if(ch=='
    '||ch=='
    ')break;
    39                   }
    40             }
    41          S=sum=0;
    42          for(int i=1;i<=n;i++)//按照规则,优先填后面的
    43             {
    44                sum+=c[i];
    45                int p=S&a[i];
    46                S|=(p<<s);
    47                S|=a[i];
    48             }
    49          memset(dp,0x3f3f3f3f,sizeof dp);
    50          dp[S]=sum;
    51          for(int i=n+1;i<=n+m;i++)
    52             {
    53                for(int j=All;j>=S;j--)//按照规则,优先填后面的
    54                   {
    55                      int p=a[i]&j;
    56                      p=a[i]|(p<<s);
    57                      dp[j|p]=min(dp[j|p],dp[j]+c[i]);
    58                   }
    59             }
    60          printf("%d
    ",dp[All]);
    61       }
    62    return 0;
    63 }
  • 相关阅读:
    算法25-----位运算(2)-----案例
    算法24----重复子字符串
    算法23-------岛屿的最大面积 LeetCode 695
    算法22-----托普利茨矩阵leetcode766
    算法21----重塑矩阵 LeetCode566
    Python笔记25-----------创建二维列表【浅copy】和转置
    TensorFlow+实战Google深度学习框架学习笔记(7)-----队列与多线程
    TensorFlow+实战Google深度学习框架学习笔记(6)----神经网络经典损失函数
    Testing
    Spark简介
  • 原文地址:https://www.cnblogs.com/D-O-Time/p/7717803.html
Copyright © 2011-2022 走看看