Codeforces 1155F Delivery Oligopoly dp(看题解)
看别人写的才学会的。。。
我们考虑刚开始的一个点, 然后我们枚举接上去的一条一条链,
dp[mask]表示当前已经加进去点的状态是mask所需的最少边数。
反正就是很麻烦的一道题, 让我自己写我是写不出来的。。。 我好菜啊。
#include<bits/stdc++.h>
#define LL long long
#define LD long double
#define ull unsigned long long
#define fi first
#define se second
#define mk make_pair
#define PLL pair<LL, LL>
#define PLI pair<LL, int>
#define PII pair<int, int>
#define SZ(x) ((int)x.size())
#define ALL(x) (x).begin(), (x).end() using namespace std; const int N = 1e6 + ;
const int inf = 0x3f3f3f3f;
const LL INF = 0x3f3f3f3f3f3f3f3f;
const int mod = 1e6 + ;
const int p = 1e6 + ;
const double eps = 1e-;
const double PI = acos(-); template<class T, class S> inline void add(T& a, S b) {a += b; if(a >= mod) a -= mod;}
template<class T, class S> inline void sub(T& a, S b) {a -= b; if(a < ) a += mod;}
template<class T, class S> inline bool chkmax(T& a, S b) {return a < b ? a = b, true : false;}
template<class T, class S> inline bool chkmin(T& a, S b) {return a > b ? a = b, true : false;} int n, m;
int G[][];
int vtomask[][ << ];
int dp[ << ], pre[ << ][];
vector<int> road[ << ][][];
vector<PII> ans; int main() {
memset(dp, inf, sizeof(dp));
scanf("%d%d", &n, &m);
for(int i = ; i < m; i++) {
int u, v; scanf("%d%d", &u, &v);
u--; v--;
G[u][v] = G[v][u] = ;
}
for(int i = ; i < n; i++) {
for(int mask = ; mask < ( << n); mask++) {
for(int j = ; j < n; j++) {
if(mask >> j & ) {
if(G[i][j]) vtomask[i][mask]++;
}
}
}
}
for(int i = ; i < n; i++) road[ << i][i][i].push_back(i);
for(int mask = ; mask < ( << n); mask++) {
for(int u = ; u < n; u++) {
for(int v = ; v < n; v++) {
if(!SZ(road[mask][u][v])) continue;
for(int z = ; z < n; z++) {
if(mask >> z & ) continue;
if(!G[v][z]) continue;
int nxtmask = mask | ( << z);
if(SZ(road[nxtmask][u][z])) continue;
road[nxtmask][u][z] = road[mask][u][v];
road[nxtmask][u][z].push_back(z);
}
}
}
}
dp[] = ;
for(int mask = ; mask < ( << n); mask++) {
if(dp[mask] >= inf) continue;
for(int u = ; u < n; u++) {
if(mask >> u & ) continue;
if(!vtomask[u][mask]) continue;
if(vtomask[u][mask] >= ) {
int nxtmask = mask | ( << u);
if(chkmin(dp[nxtmask], dp[mask] + )) {
pre[nxtmask][] = mask;
pre[nxtmask][] = u;
pre[nxtmask][] = -;
}
}
for(int v = u + ; v < n; v++) {
if(mask >> v & ) continue;
if(!vtomask[v][mask]) continue;
int amask = (( << n) - ) ^ mask ^ ( << u) ^ ( << v);
for(int smask = amask; ; smask = (smask - ) & amask) {
int tmask = smask | ( << u) | ( << v);
if(SZ(road[tmask][u][v])) {
if(chkmin(dp[mask | tmask], dp[mask] + SZ(road[tmask][u][v]) + )) {
pre[mask | tmask][] = mask;
pre[mask | tmask][] = u;
pre[mask | tmask][] = v;
}
}
if(!smask) break;
}
}
}
}
printf("%d\n", dp[( << n) - ]);
int mask = ( << n) - ;
while(mask != ) {
int pmask = pre[mask][];
int u = pre[mask][];
int v = pre[mask][];
if(~v) {
int tmask = mask ^ pmask;
for(int i = ; i < SZ(road[tmask][u][v]); i++)
ans.push_back(mk(road[tmask][u][v][i - ], road[tmask][u][v][i]));
for(int i = ; i < n; i++) {
if((pmask >> i & ) && G[u][i]) {
ans.push_back(mk(i, u));
break;
}
}
for(int i = ; i < n; i++) {
if((pmask >> i & ) && G[v][i]) {
ans.push_back(mk(i, v));
break;
}
}
} else {
for(int i = , c = ; c; i++) {
if(G[u][i] && (pmask >> i & )) {
ans.push_back(mk(u, i));
c--;
}
}
}
mask = pmask;
}
for(auto& t : ans) printf("%d %d\n", t.fi + , t.se + );
return ;
} /*
*/
Codeforces 1155F Delivery Oligopoly dp(看题解)的更多相关文章
- Codeforces 513E2 Subarray Cuts dp (看题解)
我们肯定要一大一小间隔开来所以 把式子拆出来就是类似这样的形式 s1 - 2 * s2 + 2 * s3 + ...... + sn 然后把状态开成四个, 分别表示在顶部, 在底部, 在顶部到底部的中 ...
- Codeforces 750E New Year and Old Subsequence 线段树 + dp (看题解)
New Year and Old Subsequence 第一感觉是离线之后分治求dp, 但是感觉如果要把左边的dp值和右边的dp值合起来, 感觉很麻烦而且时间复杂度不怎么对.. 然后就gun取看题解 ...
- Codeforces 1017F The Neutral Zone (看题解)
这题一看就筛质数就好啦, 可是这怎么筛啊, 一看题解, 怎么会有这么骚的操作. #include<bits/stdc++.h> #define LL long long #define f ...
- Codeforces 865C Gotta Go Fast 二分 + 期望dp (看题解)
第一次看到这种骚东西, 期望还能二分的啊??? 因为存在重置的操作, 所以我们再dp的过程中有环存在. 为了消除环的影响, 我们二分dp[ 0 ][ 0 ]的值, 与通过dp得出的dp[ 0 ][ 0 ...
- Codeforces 442D Adam and Tree dp (看题解)
Adam and Tree 感觉非常巧妙的一题.. 如果对于一个已经建立完成的树, 那么我们可以用dp[ i ]表示染完 i 这棵子树, 并给从fa[ i ] -> i的条边也染色的最少颜色数. ...
- Codeforces 1101F Trucks and Cities dp (看题解)
Trucks and Cities 一个很显然的做法就是二分然后对于每个车贪心取check, 这肯定会TLE, 感觉会给人一种贪心去写的误导... 感觉有这个误导之后很难往dp那个方向靠.. dp[ ...
- Codeforces 596D Wilbur and Trees dp (看题解)
一直在考虑, 每一段的贡献, 没想到这个东西能直接dp..因为所有的h都是一样的. #include<bits/stdc++.h> #define LL long long #define ...
- Codeforces 983C Elevator dp (看题解)
Elevator 怎么今天写啥题都不会写啊, 我是傻了吗.. 把电梯里面四个人的目标点当作状态, 然后暴力转移. #include<bits/stdc++.h> #define LL lo ...
- Codeforces 744C Hongcow Buys a Deck of Cards 状压dp (看题解)
Hongcow Buys a Deck of Cards 啊啊啊, 为什么我连这种垃圾dp都写不出来.. 不是应该10分钟就该秒掉的题吗.. 从dp想到暴力然后gg, 没有想到把省下的红色开成一维. ...
随机推荐
- 升级AndroidStudio3.4问题汇总
1.Could not get unknown property 'bootClasspath' for object of type org.gradle.api.tasks.compile.Com ...
- palacehoder的自定义样式【输入框input /文本域textarea】
7.palacehoder的自定义样式[输入框input /文本域textarea] 因为每个浏览器的CSS选择器都有所差异,所以需要针对每个浏览器做单独的设定(可以在冒号前面写input和texta ...
- luasocket编译安装遇到的坑
由于需要获得本机的IP地址,所以需要 : local socket = require('socket') local server_hostname = socket.dns.gethostname ...
- ElasticSearch query_string vs multi_match cross_fields query
ElasticSearch query_string vs multi_match cross_fields query 本文记录以字段为中心的查询和以词为中心的查询这两种查询方式的区别以及在Elas ...
- APP数据的爬取
前言 App 的爬取相比 Web 端爬取更加容易,反爬虫能力没有那么强,而且数据大多是以 JSON形式传 输的,解析更加简单.在 Web 端,我们可以通过浏览器的开发者工具监听到各个网络请求和响应过程 ...
- day 19 - 1 模块
collections 模块 在内置数据类型(dict.list.set.tuple)的基础上,collections 模块还提供了几个额外的数据类型:Counter.deque.defaultdic ...
- HDU - 1050
wa了5遍?!! (1)前4遍,思路没简化,企图模拟整个过程,但是调用sort函数时由于没有把奇数的屋子和偶数的屋子统一,排序出了问题. 思路:遍历n段,每次只扫未被标记过的一段,ans++并且从该段 ...
- 20175333曹雅坤MyCP(课下作业,必做)
MyCP(课下作业,必做) 要求 编写MyCP.java 实现类似Linux下cp XXX1 XXX2的功能,要求MyCP支持两个参数: java MyCP -tx XXX1.txt XXX2.bin ...
- docker中的镜像中运行Django项目
首先要在镜像中 安装python3 以及 django2.0.4 目前我用的是这两个版本. 进入镜像 创建项目 进入项目中修改setting文件 将引号和星号添加进括号中 ALLOWED_HOSTS ...
- 记一次简单的PHP代码审计(SSRF案例)
题目链接: http://oj.momomoxiaoxi.com:9090/ 用dirsearch对网址进行扫描,发现robots.txt 命令行: python3 dirsearch.py -u & ...