上个月写的题qwq……突然想写篇博客

题目:

洛谷4294

分析:

斯坦纳树模板题。

简单来说,斯坦纳树问题就是给定一张有边权(或点权)的无向图,要求选若干条边使图中一些选定的点连通(可以经过其他点),且边权(或点权)之和最小。很明显,这样最终形成的是一棵树。

通常,斯坦纳树问题规模都比较小。考虑状压DP。用\(dp[u][S]\)表示让点\(u\)与集合\(S\)中所有关键点连通的最小花费。有如下两种转移:

第一,把两条到\(u\)的路径拼在一起,减去重合点\(u\)的点权,即(\(w_u\)表示点\(u\)的点权,\(S'\)表示\(S\)的一个真非空子集,\(S-S'\)表示以\(S'\)相对于\(S\)的补集,下同):

\[dp[u][S]=min(dp[u][S']+dp[u][S-S']-w_u)
\]

第二,延伸一条路径,即(\(v\)与\(u\)之间存在一条边):

\[dp[u][S]=min(dp[v][S]+w_u)
\]

第二种存在循环更新的问题。但是它长得很像最短路,于是大力跑最短路算法即可。

注意更新顺序,要从小到大枚举集合\(S\),先更新第一种再更新第二种。

代码:

把上面的点\(u\)换成坐标就好了……

dp的时候记一下从哪个状态转移来的。

#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cctype>
#include <queue>
#include <functional>
using namespace std;
#define _ 0 namespace zyt
{
template<typename T>
inline void read(T &x)
{
char c;
bool f = false;
x = 0;
do
c = getchar();
while (c != '-' && !isdigit(c));
if (c == '-')
f = true, c = getchar();
do
x = x * 10 + c - '0', c = getchar();
while (isdigit(c));
if (f)
x = -x;
}
template<typename T>
inline void write(T x)
{
static char buf[20];
char *pos = buf;
if (x < 0)
putchar('-'), x = -x;
do
*pos++ = x % 10 + '0';
while (x /= 10);
while (pos > buf)
putchar(*--pos);
}
const int N = 10, ST = 1 << N, INF = 0x3f3f3f3f;
struct _pre
{
int x, y, st;
_pre(const int _x = -1, const int _y = -1, const int _st = -1)
: x(_x), y(_y), st(_st) {}
}pre[N][N][ST];
struct point
{
int x, y;
point(const int _x = 0, const int _y = 0)
: x(_x), y(_y) {}
bool operator < (const point &b) const
{
return x == b.x ? y < b.y : x < b.x;
}
};
int f[N][N][ST], arr[N][N], n, m, k;
const int dx[] = {0, 0, 1, -1};
const int dy[] = {1, -1, 0, 0};
void Dijkstra(const int s)
{
typedef pair<int, point> pip;
static priority_queue<pip, vector<pip>, greater<pip> > q;
static bool vis[N][N];
while (!q.empty())
q.pop();
for (int i = 0; i < n; i++)
for (int j = 0; j < m; j++)
{
q.push(make_pair(f[i][j][s], point(i, j)));
vis[i][j] = false;
}
while (!q.empty())
{
point u = q.top().second;
q.pop();
vis[u.x][u.y] = true;
for (int i = 0; i < 4; i++)
{
point v = point(u.x + dx[i], u.y + dy[i]);
if (v.x < 0 || v.x >= n || v.y < 0 || v.y >= m || vis[v.x][v.y])
continue;
if (f[v.x][v.y][s] > f[u.x][u.y][s] + arr[v.x][v.y])
{
f[v.x][v.y][s] = f[u.x][u.y][s] + arr[v.x][v.y];
pre[v.x][v.y][s] = _pre(u.x, u.y, s);
q.push(make_pair(f[v.x][v.y][s], v));
}
}
}
}
bool mark[N][N];
void dfs(const _pre p)
{
mark[p.x][p.y] = true;
_pre nxt = pre[p.x][p.y][p.st];
if (nxt.x == -1 && nxt.y == -1 && nxt.st == -1)
return;
dfs(nxt);
if (nxt.st != p.st)
{
nxt.st ^= p.st;
dfs(nxt);
}
}
int work()
{
read(n), read(m);
for (int i = 0; i < n; i++)
{
memset(f[i], INF, sizeof(int[m][ST]));
for (int j = 0; j < m; j++)
{
read(arr[i][j]);
if (!arr[i][j])
f[i][j][1 << (k++)] = 0;
}
}
for (int i = 0; i < (1 << k); i++)
{
for (int j = (i - 1) & i; j; j = (j - 1) & i)
for (int x = 0; x < n; x++)
for (int y = 0; y < m; y++)
if (f[x][y][i] > f[x][y][j] + f[x][y][i ^ j] - arr[x][y])
{
f[x][y][i] = f[x][y][j] + f[x][y][i ^ j] - arr[x][y];
pre[x][y][i] = _pre(x, y, j);
}
Dijkstra(i);
}
int ans = INF;
_pre pans;
for (int i = 0; i < n; i++)
for (int j = 0; j < m; j++)
if (ans > f[i][j][(1 << k) - 1])
{
ans = f[i][j][(1 << k) - 1];
pans = _pre(i, j, (1 << k) - 1);
}
write(ans), putchar('\n');
dfs(pans);
for (int i = 0; i < n; i++)
{
for (int j = 0; j < m; j++)
if (!arr[i][j])
putchar('x');
else if (mark[i][j])
putchar('o');
else
putchar('_');
putchar('\n');
}
return (0^_^0);
}
}
int main()
{
return zyt::work();
}

【BZOJ2595_洛谷4294】[WC2008]游览计划(斯坦纳树_状压DP)的更多相关文章

  1. 洛谷4294 [WC2008]游览计划——斯坦纳树

    题目:https://www.luogu.org/problemnew/show/P4294 大概是状压.两种转移,一个是以同一个点为中心,S由自己的子集拼起来:一个是S相同.中心不同的同层转移. 注 ...

  2. Luogu 4294 [WC2008]游览计划 | 斯坦纳树

    题目链接 Luogu 4294 (我做这道题的时候BZOJ全站的SPJ都炸了 提交秒WA 幸好有洛谷) 题解 这道题是[斯坦纳树]的经典例题.斯坦纳树是这样一类问题:带边权无向图上有几个(一般约10个 ...

  3. [bzoj4006][JLOI2015]管道连接_斯坦纳树_状压dp

    管道连接 bzoj-4006 JLOI-2015 题目大意:给定一张$n$个节点$m$条边的带边权无向图.并且给定$p$个重要节点,每个重要节点都有一个颜色.求一个边权和最小的边集使得颜色相同的重要节 ...

  4. BZOJ 4006 Luogu P3264 [JLOI2015]管道连接 (斯坦纳树、状压DP)

    题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=4006 (luogu)https://www.luogu.org/probl ...

  5. 【BZOJ2595】[Wc2008]游览计划 斯坦纳树

    [BZOJ2595][Wc2008]游览计划 Description Input 第一行有两个整数,N和 M,描述方块的数目. 接下来 N行, 每行有 M 个非负整数, 如果该整数为 0, 则该方块为 ...

  6. bzoj2595: [Wc2008]游览计划 斯坦纳树

    斯坦纳树是在一个图中选取某些特定点使其联通(可以选取额外的点),要求花费最小,最小生成树是斯坦纳树的一种特殊情况 我们用dp[i][j]来表示以i为根,和j状态是否和i联通,那么有 转移方程: dp[ ...

  7. BZOJ2595: [Wc2008]游览计划(斯坦纳树,状压DP)

    Time Limit: 10 Sec  Memory Limit: 256 MBSec  Special JudgeSubmit: 2030  Solved: 986[Submit][Status][ ...

  8. BZOJ 2595 [Wc2008]游览计划 ——斯坦纳树

    [题目分析] 斯坦纳树=子集DP+SPFA? 用来学习斯坦纳树的模板. 大概就是用二进制来表示树包含的点,然后用跟几点表示树的形态. 更新分为两种,一种是合并两个子集,一种是换根,换根用SPFA迭代即 ...

  9. bzoj2595 [Wc2008]游览计划——斯坦纳树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2595 今天刚学了斯坦纳树,还不太会,写一道题练习一下: 参考了博客:http://www.c ...

随机推荐

  1. git帮助网址

    http://rogerdudler.github.io/git-guide/index.zh.html https://help.github.com/

  2. python3返回值中的none

    浏览器返回null,python3返回none,懵了. google了很多资料,不明就里,这就是没基础的后果啊呀呀呀. 上阮一峰的截图,就这么理解下凑合吧:

  3. OS X中微信双开

    1.打开系统终端: 2.下载插件,输入命令(不包括括号) git clone https://github.com/Sunnyyoung/WeChatTweak-macOS.git 并回车: 3.进入 ...

  4. software collection

    software software Table of Contents 1. Privacy 2. GFW 2.1. google search 2.2. 修改 DNS 服务器 2.2.1. 修改ip ...

  5. 【DIP, OpenCV】Some Kinds Of Image Smoothing Methodologies

    In digital image processing(DIP), many methods are used in smoothing images in order to suppress noi ...

  6. RequestMapping_HiddenHttpMethodFilter 过滤器

    [REST] 1.REST:即Representational State Transfer.(资源)表现层状态转化.是目前最流行的一种互联网软件架构.它结构清晰.符合标准.易于理解.扩展方便,所以得 ...

  7. scrapy的User-Agent中间件、代理IP中间件、cookies设置、多个爬虫自定义settings设置

    在scrapy的反爬中,常用的几个配置,简单总结了下: User-Agent中间件: from fake_useragent import UserAgent class RandomUserAgen ...

  8. 用xtrabackup+binlog恢复误删除的数据库

    关键技术,数据库产生的二进制文件,在主库就是binlog在从库就是relay-log,用最新的物理备份可以新启动个新实例,可以模拟个从库,把主库的binlog复制到新的数据库实例上,利用主从复制和物理 ...

  9. 链表中倒数第N个元素——剑指Offer

    https://www.nowcoder.net/practice/529d3ae5a407492994ad2a246518148a?tpId=13&tqId=11167&tPage= ...

  10. 【CV论文阅读】Faster R-CNN: Towards Real-Time Object Detection with Region Proposal Networks

    由RCNN到FAST RCNN一个很重要的进步是实现了多任务的训练,但是仍然使用Selective Search算法来获得ROI,而FASTER RCNN就是把获得ROI的步骤使用一个深度网络RPN来 ...