原题链接

题目大意

\(n\times n\)的带权方阵,选一个权值最大的连通块

Solution

一眼连通性DP,然后就没了

转移很好想的啦,简单讨论一下就行了

有一个坑点,就是不能一个格子都不选,特判一下

注释还算详细QwQ

#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <string>
#include <vector>
#include <cmath>
#include <ctime>
#include <queue>
#include <map>
#include <set> using namespace std; #define ull unsigned long long
#define pii pair<int, int>
#define uint unsigned int
#define mii map<int, int>
#define lbd lower_bound
#define ubd upper_bound
#define INF 0x3f3f3f3f
#define IINF 0x3f3f3f3f3f3f3f3fLL
#define DEF 0x8f8f8f8f
#define DDEF 0x8f8f8f8f8f8f8f8fLL
#define vi vector<int>
#define ll long long
#define mp make_pair
#define pb push_back
#define re register
#define il inline #define N 10
#define MOD 114511 int n;
int val[N+5][N+5], ans = DEF, bin[N+5]; namespace HashTable { //哈希表所需
int lst, cur, head[MOD+5], nxt[MOD+5], tot[2];
int f[2][MOD+5], a[2][MOD+5];
void insert(int x, int v) { // 把x状态的最优值与v取max
int x0 = x%MOD, i;
for(i = head[x0]; i; i = nxt[i]) if(a[cur][i] == x) break;
if(!i) nxt[++tot[cur]] = head[x0], f[cur][tot[cur]] = DEF, a[cur][tot[cur]] = x, head[x0] = i = tot[cur];
f[cur][i] = max(f[cur][i], v);
}
} using namespace HashTable; // 最多有5(9/2向上取整)个联通块
// 采用8进制压缩 int decode(int s0, int v) { // 最小表示所需
static int tf[8];
int cnt = 0, s = 0;
memset(tf, 0, sizeof tf);
for(int i = 0; i < n; ++i) {
int x = (s0>>(3*i))%8;
if(!x) continue;
if(!tf[x]) tf[x] = ++cnt;
s += tf[x]*bin[i];
}
if(cnt == 1) ans = max(ans, v); // 顺便更新全局最优解
return s;
} int main() {
scanf("%d", &n);
bin[0] = 1;
for(int i = 1; i <= n; ++i) bin[i] = bin[i-1]<<3;
for(int i = 1; i <= n; ++i)
for(int j = 1; j <= n; ++j) scanf("%d", &val[i][j]), ans = max(ans, val[i][j]);
insert(0, 0); // 初始化
for(int i = 1; i <= n; ++i) {
for(int j = 1; j <= n; ++j) {
lst = cur, cur ^= 1, tot[cur] = 0;
memset(head, 0, sizeof head);
for(int k = 1; k <= tot[lst]; ++k) {
int s = a[lst][k], v = f[lst][k], p1, p2 = (s>>(3*(j-1)))%8;
if(j == 1) p1 = 0;
else p1 = (s>>(3*(j-2)))%8;
if(!p1 && !p2) { // 左和上都没有选
insert(decode(s, v), v); // 当前不选
insert(decode(s+7*bin[j-1], v+val[i][j]), v+val[i][j]); // 当前选
}
else if(p1 && !p2) { // 只有左选了
insert(decode(s, v), v); // 当前不选
insert(decode(s+p1*bin[j-1], v+val[i][j]), v+val[i][j]); // 当前选
}
else if(!p1 && p2) { // 只有上选了
int cnt = 0;
for(int p = 0; p < n; ++p) if((s>>(3*p))%8 == p2) cnt++; // 只要这个连通块还有另一处与轮廓线相交就可以不选当前的
if(cnt >= 2) insert(decode(s-p2*bin[j-1], v), v); // 当前不选
insert(decode(s, v+val[i][j]), v+val[i][j]); // 当前选
}
else { // 左和上都选了
int cnt = 0;
for(int p = 0; p < n; ++p) if((s>>(3*p))%8 == p2) cnt++; // 与上一种情况同理
if(cnt >= 2) insert(decode(s-p2*bin[j-1], v), v); // 当前不选
if(p1 != p2) for(int p = 0; p < n; ++p) if((s>>(3*p))%8 == p2) s += p1*bin[p]-p2*bin[p]; // 合并两个连通块
insert(decode(s, v+val[i][j]), v+val[i][j]); // 当前选
}
}
}
}
printf("%d\n", ans);
return 0;
}

[JLOI2009]神秘的生物——轮廓线DP的更多相关文章

  1. Luogu P3886 [JLOI2009]神秘的生物 最小表示法,轮廓线DP,插头DP,动态规划

    亲手写掉的第一道最小表示法!哈哈哈太开心啦~ 不同于以往的几个插头\(dp\),这个题目的轮廓线是周围的一圈\(n\)个格子.而其所谓"插头"也变成了相邻格子的所属连通分量编号,并 ...

  2. [JLOI2009]神秘的生物

    题目链接 题目大意 给定一个\(n*n\)的矩阵,从其中选取恰好一个连通块,使选取的格子所对应的权值和最大. \(n\leq 9\) 解题思路 由于\(n\)特别小,考虑插头dp. 和一般的插头dp不 ...

  3. 轮廓线DP POJ3254 && BZOJ 1087

    补了一发轮廓线DP,发现完全没有必要从右往左设置状态,自然一点: 5 6 7 8 9 1 2 3 4 如此设置轮廓线标号,转移的时候直接把当前j位改成0或者1就行了.注意多记录些信息对简化代码是很有帮 ...

  4. HDU4804 Campus Design 轮廓线dp

    跟上面那篇轮廓线dp是一样的,但是多了两个条件,一个是在原图上可能有些点是不能放的(即障碍),所以转移的时候要多一个判断color[i][j]是不是等于1什么的,另外一个是我们可以有多的1*1的骨牌, ...

  5. POJ2411 Mondriaan's Dream 轮廓线dp

    第一道轮廓线dp,因为不会轮廓线dp我们在南京区域赛的时候没有拿到银,可见知识点的欠缺是我薄弱的环节. 题目就是要你用1*2的多米诺骨排填充一个大小n*m(n,m<=11)的棋盘,问填满它有多少 ...

  6. UVA - 11270 轮廓线DP

    其实这题还能用状压DP解决,可是时间达到2000ms只能过掉POJ2411.状压DP解法详见状压DP解POJ2411 贴上POJ2411AC代码 : 2000ms 时间复杂度h*w*(2^w)*(2^ ...

  7. [UOJ422][集训队作业2018]小Z的礼物——轮廓线DP+min-max容斥

    题目链接: [集训队作业2018]小Z的礼物 题目要求的就是最后一个喜欢的物品的期望得到时间. 根据$min-max$容斥可以知道$E(max(S))=\sum\limits_{T\subseteq ...

  8. 【UOJ#422】【集训队作业2018】小Z的礼物(min-max容斥,轮廓线dp)

    [UOJ#422][集训队作业2018]小Z的礼物(min-max容斥,轮廓线dp) 题面 UOJ 题解 毒瘤xzy,怎么能搬这种题当做WC模拟题QwQ 一开始开错题了,根本就不会做. 后来发现是每次 ...

  9. BZOJ.4572.[SCOI2016]围棋(轮廓线DP)

    BZOJ 洛谷 \(Description\) 给定\(n,m,c\).\(Q\)次询问,每次询问给定\(2*c\)的模板串,求它在多少个\(n*m\)的棋盘中出现过.棋盘的每个格子有三种状态. \( ...

随机推荐

  1. 使用 pycharm调试docker环境运行的Odoo

    2019日 星期一 安装docker windows系统,参考 docker官方文档 Mac系统,参考 docker官方文档 构建自定义ODOO镜像 标准ODOO镜像可能不包含特别的python模块, ...

  2. docker镜像内没有vim

    问题: 我们在容器中找不到vim等命令   原因: 镜像制作的时候没把这些东西加进去   解决: 用apt update更新源之后再安装vim apt update apt-get install - ...

  3. mysql中的反引号``

    [1]反引号`,数字1左边的符号.tab键上面的符号. 它是为了区分MYSQL的保留字与普通字符而引入的符号. 不加反引号建的表不能包含MYSQL保留字,否则出错 如上图,很明显的,如果我们直接建立名 ...

  4. 【LOJ】#2239. 「CQOI2014」危桥

    LOJ#2239. 「CQOI2014」危桥 就是先把每条边正着连一条容量为2的边,反着连一条容量为2的边 显然如果只有一个人走的话,答案就是一个源点往起点连一条容量为次数×2的边,终点往汇点连一个次 ...

  5. java微信token校验

    1.微信验证接口 package com.park.utils.wechatUtil; import org.springframework.web.bind.annotation.RequestMa ...

  6. 模块和包,logging模块

    模块和包,logging日志 1.模块和包 什么是包? 只要文件夹下含有__init__.py文件就是一个包. 假设文件夹下有如下结构 bake ├── test.py ├── __init__.py ...

  7. Git安装使用秘籍

    首先Git的功能,是用于帮助用户实现版本控制的软件,GIT一般和GitHub配套使用.Git是个软件,GitHub是个网站,它们的关系就像雷锋与雷峰塔一样,没什么关系.本文只提供Git安装方法,其它请 ...

  8. springboot application.properties配置大全

    springboot application.properties配置大全 官方文档 https://docs.spring.io/spring-boot/docs/current/reference ...

  9. C# 7.0 语法

    C# 7.0的语法主要是优化了之前的写法,使得更加简洁方便.try catch when  这个使用场景很少,正常的开发无业务处理的时候不建议使用 . #region 2.字符串嵌入值 Console ...

  10. No database provider has been configured for this DbContext

    var context = ((IInfrastructure<IServiceProvider>)set).GetService<DbContext>(); 在EF Core ...