传送门

题目大意:

有n(\(\le 15\))个作业,每个作业有个name, deadline(截止日期),cost(做作业花的时间),如果没有按时完成某个作业,惩罚分数为超出的时间,求一个合理的顺序使得惩罚分数最小,如果有多个方案分数相同,输出字典序最小的。

题目分析

看到\(n \le 15\)可知状压dp:dp[S]表示完成S状态的最小惩罚分数,转移也较为简单:$$dp[i | (1 << (j - 1))] = min(dp[S | (1 << (j - 1))], dp[i] + getTime(i) + cost[j] - deadline[j])$$。因为要输出方案,于是记录一个from数组表示这个状态是从哪个状态转移来的。又因为要求字典序最小,现将name排序,这样在转移时如果\(dp[i | (1 << (j - 1))] == dp[i] + getTime(i) + cost[j] - deadline[j])\),并且\(i | (1 << (j - 1)) < from[i]\),那么\(from[i] = i | (1 << (j - 1))\),来保证字典序最小。

code

#include<bits/stdc++.h>
using namespace std; const int N = 20, S = 33000, OO = 0x3f3f3f3f;
int n, dp[S], T, from[S];
vector<int> ans;
struct node{
string name;
int dl, cost;
inline bool operator < (const node &b) const {
return name < b.name;
}
}task[N]; inline int getTime(int s){
int ans = 0;
for (int i=0; (1<<i)<=s; i++)
if ((1<<i)&s)
ans += task[i + 1].cost;
return ans;
} inline void init(){
memset(dp, OO, sizeof dp), dp[0] = 0;
memset(from, 0, sizeof from);
ans.clear();
} inline int find(int x){
int l = 1, r = 15;
while (l<=r) {
int mid = (l + r) >> 1;
if (1<<(mid-1) == x) return mid;
else if (1<<(mid-1) > x) r = mid - 1;
else l = mid + 1;
}
return 0;
} int main(){
ios::sync_with_stdio(false);
cin.tie(NULL), cout.tie(NULL);
cin >> T;
while(T--){
init();
cin >> n;
for (int i=1; i<=n; i++) cin >> task[i].name >> task[i].dl >> task[i].cost;
sort(task + 1, task + n + 1);
for (int i=0; i<(1<<n); i++) {
for(int j=1; j<=n; j++) {
if (i&(1<<(j-1))) continue;
if (dp[i|(1<<(j-1))] > dp[i] + max(0, getTime(i) + task[j].cost - task[j].dl)) {
from[i|(1<<(j-1))] = i;
dp[i|(1<<(j-1))] = dp[i] + max(0, getTime(i) + task[j].cost - task[j].dl);
}
else if(dp[i|(1<<(j-1))] == dp[i] + max(0, getTime(i) + task[j].cost - task[j].dl)) {
if(i < from[i|(1<<(j-1))]) from[i|(1<<(j-1))] = i;
}
}
}
cout << dp[(1<<n)-1] << endl;
int now = (1<<n)-1;
while(now) {
int diff = now ^ (from[now]);
int pos = find(diff);
// cout << now << " " << from[now] << " " << diff << " " << pos << endl;
ans.push_back(pos);
now = from[now];
}
for(int i=ans.size()-1; i>=0; i--) cout << task[ans[i]].name << endl;
}
return 0;
}

最新文章

  1. Python2.7-异常和工具
  2. 2015 年 JavaScript 开发者调查报告
  3. JSON和GSON操作json数据
  4. Redis 源码解析
  5. equals方法,hashcode()方法
  6. (转)javaScript插件开发
  7. 配置超级用户口令(Cisco IOS系统)
  8. libpq中调用prepared statement:
  9. tomcat 配置https (单向认证)
  10. mybatis配置sql超时时间
  11. TCP数据包结构
  12. MariaDB多源复制环境搭建(多主一丛)
  13. Go的基本环境配置
  14. bootrom脚本的创建
  15. django之ajax补充
  16. HDU - 3006 The Number of set(状态压缩位运算)
  17. springboot系列一、springboot产生背景及介绍
  18. CentOS 7.x,不重新编译 PHP,动态安装 imap 扩展
  19. Ubuntu 安装软件方法
  20. LIS问题(DP解法)---poj1631

热门文章

  1. Redis原理(二)
  2. openGLES(二)
  3. php字符串函数分类总结
  4. winform最大化后不遮挡任务栏
  5. 什么是MVC,什么是WCF
  6. Day2:PYC
  7. Java BigDecimal的基本使用方法
  8. Machine-learning of Andrew Ng
  9. [Angular HTML] Implementing The Input Mask Cursor Navigation Functionality -- setSelectionRange
  10. Linux环境编程之共享内存区(一):共享内存区简单介绍