题目大意:

网址:https://www.luogu.org/problemnew/show/3231
大意:a×b×c的三维空间里有a×b×c个点(x,y,z),其中有些点需要被消除。
消除的方法为:每次选定一个a1×b1×c1的三维区域,然后消除这个区域内的所有点。
消除的代价\(cost = min(a1,b1,c1);\)
现在询问消除此三维空间中所有需要消除点的最小代价为多少。
数据范围:\(a*b*c<=5000\)

题目解法:

显然题目可以转化为每次选择一个平面,然后消除该平面上的所有点。
先考虑二维空间,这不是超级无敌大水题吗。
二维不就是裸的二分图最小顶点覆盖吗?不明白怎么做的去AC一下这题:poj3041-Asteroids
三维咋办?
观察到\(a*b*c<=5000\),那么a、b、c中至少有一个是小于等于17的。
我们枚举最小的这一维是否切割,如果不切再跑最小顶点覆盖即可。
然后这题最要命的其实不是怎么做,而是怎么实现。
考虑一下建图怎么办,开个三维数组乱搞、旋转肯定是不行的。
其实可以把每一个点拆成三个坐标,然后连边(具体看代码,讲也讲不清)。

具体实现代码:

#include<bits/stdc++.h>
#define gi(x) scanf("%d",&x)
#define maxn 5005
#define INF 1e9+7
using namespace std;

int D,L[4],d,pos,g1,g2,g3,use[maxn];
struct Road{int to,next,blg;}t[2*maxn]; int head[maxn],cnt;
int mtc[maxn],vis[maxn],Ans;

void Add(int u,int v,int w){
    t[++cnt] = (Road){v,head[u]}; head[u] = cnt;
    t[++cnt] = (Road){u,head[v]}; head[v] = cnt;
    t[cnt-1].blg = t[cnt].blg = w;
}

//建图:
void Build(){
    gi(L[1]); gi(L[2]); gi(L[3]);
    pos = 1; cnt = 0;
    for(int i = 1; i <= 3; i ++)
        if(L[ i ] < L[ pos ])pos = i;
    if(pos == 1)g1 = L[1],g2 = L[2],g3 = L[3];
    if(pos == 2)g1 = L[2],g2 = L[1],g3 = L[3];
    if(pos == 3)g1 = L[3],g2 = L[1],g3 = L[2];
    for(int i = 1; i <= g2+g3; i ++)head[i] = 0;
    for(int i = 1; i <= L[1]; i ++)
        for(int j = 1; j <= L[2]; j ++)
            for(int k = 1; k <= L[3]; k ++){
                gi(d); if(!d)continue;
                if(pos == 1)Add(j , g2 + k , i );
                if(pos == 2)Add(i , g2 + k , j );
                if(pos == 3)Add(i , g2 + j , k );
            }
    return;
}

bool Hungarian(int u,int Vis){
    for(int i = head[u]; i; i = t[i].next){
        int v = t[i].to;
        if(vis[v] != Vis && !use[t[i].blg]){
            vis[v] = Vis;
            if(!mtc[v] || Hungarian(mtc[v],Vis)){
                mtc[v] = u; mtc[u] = v;
                return true;
            }
        }
    }return false;
}

inline int Solve(int ret){
    int Res = 0;
    for(int i = 1; i <= g2+g3; i ++)mtc[i] = 0;
    for(int i = 1; i <= g2+g3; i ++)vis[i] = 0;
    for(int i = 1; i <= g2; i ++){
        if(mtc[i])continue;
        if(Hungarian(i,i))Res++;
        if(Res + ret >= Ans)return Res + ret;
    }return Res + ret;
}

void Dfs(int nw,int ret){
    if(nw == L[pos]+1){Ans = min(Ans,Solve(ret)); return;}
    use[nw] = true; Dfs(nw+1,ret+1);
    use[nw] = false; Dfs(nw+1,ret);
}

int main(){
    gi(D);
    while(D--){
        Build();
        Ans = INF;
        Dfs(1,0);  printf("%d\n",Ans);
    }
    return 0;
}

[HNOI2013]消毒的更多相关文章

  1. [BZOJ3140][HNOI2013]消毒(二分图最小点覆盖)

    3140: [Hnoi2013]消毒 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1621  Solved: 676[Submit][Status] ...

  2. bzoj 3140: [Hnoi2013]消毒

    3140: [Hnoi2013]消毒 Description 最近在生物实验室工作的小T遇到了大麻烦. 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为a*b*c,a.b.c 均为正整数 ...

  3. P3231 [HNOI2013]消毒

    P3231 [HNOI2013]消毒 二维覆盖我们已经很熟悉了 扩展到三维,枚举其中较小的一维,这里定义为$a$ 以$a$为关键字状压,$1$表示该面全选 剩下的面和二维覆盖一样二分图匹配 如果还没接 ...

  4. 3140:[HNOI2013]消毒 - BZOJ

    题目描述 Description 最近在生物实验室工作的小 T 遇到了大麻烦. 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为 a*b*c,a.b.c均为正整数.为了实验的方便,它被划 ...

  5. bzoj3140: [Hnoi2013]消毒

    Description 最近在生物实验室工作的小T遇到了大麻烦. 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为a*b*c,a.b.c 均为正整数.为了实验的方便,它被划分为a*b*c ...

  6. 【刷题】BZOJ 3140 [Hnoi2013]消毒

    Description 最近在生物实验室工作的小T遇到了大麻烦. 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为abc,a.b.c 均为正整数.为了实验的方便,它被划分为abc个单位立 ...

  7. BZOJ3140:[HNOI2013]消毒——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=3140 https://www.luogu.org/problemnew/show/P3231 最近在 ...

  8. bzoj3140: [Hnoi2013]消毒(二分图)

    题目描述 最近在生物实验室工作的小T遇到了大麻烦. 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为a*b*c,a.b.c 均为正整数.为了实验的方便,它被划分为a*b*c个单位立方体区 ...

  9. [luogu3231 HNOI2013] 消毒 (二分图最小点覆盖)

    传送门 Description 最近在生物实验室工作的小T遇到了大麻烦. 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为abc,a.b.c 均为正整数.为了实验的方便,它被划分为abc ...

随机推荐

  1. Base64算法原理

    3个Byte (3 X 8 = 24 bits) 以3个字节为单位,依次取6位数据,并在前面补上2个0.这样就增加了一个字节的数据.

  2. centos7使用docker部署gitlab-ce-zh应用

    1.国内拉取镜像比较慢,所以这里采用DaoCloud源. # curl -sSL https://get.daocloud.io/daotools/set_mirror.sh | sh -s http ...

  3. go入门

    1.hello world 小程序 package main import "fmt" func main() { fmt.println("hello,世界" ...

  4. php提供的sapi有哪些?CGI、FastCGI、php-fpm、php-cgi解释

    一.前言 一直对PHP的sapi是什么东西好奇,在网上一查都是各种说fpm cgi fastcgi php-cgi 直到看了鸟哥的这篇文章介绍戳这里,看到源码下的sapi目录才有所了解. 二.sapi ...

  5. 使用JSONP进行跨域Ajax 调用

    JSONP 是啥 JSONP 全称是JSON with Padding. 当需要进行跨域Ajax 调用的时候, 需要用到JSONP 协议. 客户端 $.ajax({ url: 'http://xxx' ...

  6. Xen的虚拟化详解

    最近在看Xen在2003年发表在sosp上的论文<Xen and the Art of Virtualization>,中途遇到一些不理解的技术点,在网络上查找相关资料,发现大多数人都只是 ...

  7. java12 - 正则表达式

    正则表达式简介 常用搭配说明: ^a 表示这个位置只能是字母 a [1,2,3,4] 表示这个位置只能在 1,2,3,4 中取一个 [[a-z][A-Z]] 表示可以任意大小写字母 ([a-z]{1, ...

  8. 关于微信分享到朋友圈(Thinkphp框架下实现)

    PHP部分 扩展类代码部分: <?php namespace Think; class JsSdk {       private $appId;       private $appSecre ...

  9. 由select引发的思考

    一.前言 网络编程里一个经典的问题,selec,poll和epoll的区别?这个问题刚学习编程时就接触了,当时看了材料很不明白,许多概念和思想没有体会,现在在这个阶段,再重新回头看这个问题,有一种豁然 ...

  10. CodeForces - 796C Bank Hacking

    思路:共有n-1条边连接n个点,即形成一棵树.一开始需要选择一个点hack--将这个点视为根结点,与它相邻的点防御值加1,与它相隔一个在线点的点的防御也加1.当根节点被hack,即这个点被删除,又变成 ...