bzoj1001(bj2006)

Description
现在小朋友们最喜欢的”喜羊羊与灰太狼”,话说灰太狼抓羊不到,但抓兔子还是比较在行的,
而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形:

左上角点为(1,1),右下角点为(N,M)(上图中N=4,M=5).有以下三种类型的道路
1:(x,y)<==>(x+1,y)
2:(x,y)<==>(x,y+1)
3:(x,y)<==>(x+1,y+1)
道路上的权值表示这条路上最多能够通过的兔子数,道路是无向的. 左上角和右下角为兔子的两个窝,
开始时所有的兔子都聚集在左上角(1,1)的窝里,现在它们要跑到右下解(N,M)的窝中去,狼王开始伏击
这些兔子.当然为了保险起见,如果一条道路上最多通过的兔子数为K,狼王需要安排同样数量的K只狼,
才能完全封锁这条道路,你需要帮助狼王安排一个伏击方案,使得在将兔子一网打尽的前提下,参与的
狼的数量要最小。因为狼还要去找喜羊羊麻烦.
Input
第一行为N,M.表示网格的大小,N,M均小于等于1000.
接下来分三部分
第一部分共N行,每行M-1个数,表示横向道路的权值.
第二部分共N-1行,每行M个数,表示纵向道路的权值.
第三部分共N-1行,每行M-1个数,表示斜向道路的权值.
输入文件保证不超过10M
Output
输出一个整数,表示参与伏击的狼的最小数量.

Sample Input
3 4

5 6 4

4 3 1

7 5 3

5 6 7 8

8 7 6 5

5 5 5

6 6 6
Sample Output
14
HINT
2015.4.16新加数据一组,可能会卡掉从前可以过的程序。


看上去很最小鸽
点数过多直接最小鸽会T怎么办?
图形很有规则,最小鸽转最短路
平面图转一波对偶图跑最短路就好了

sb的我spfa写错查了一年

#include<bits/stdc++.h>
using namespace std;
const int maxn = 2001000;
int linkk[maxn]  , t , N , M , n;
int in[1010][1010];
int S , T ;
struct node{
    int n , y , v;
}e[maxn * 5];
int read(){
    int sum = 0;char c = getchar();
    while( c < '0' || c > '9' ) c = getchar();
    while( c >= '0' && c <= '9' ) sum = sum * 10 + c - 48 , c = getchar();
    return sum;
} 
void insert(int x,int y,int z){
    //printf("#%d %d %d\n",x,y,z);
    e[++t].y = y;e[t].n = linkk[x];e[t].v = z;linkk[x] = t;
    e[++t].y = x;e[t].n = linkk[y];e[t].v = z;linkk[y] = t;
    return;
}
void get_in(){
    for(int i = 1;i <= N - 1;++i)
     for(int j = 1;j <= M;++j)
      in[i][j] = read();
}
void init(){
    N = read();M = read();
    n = ( N - 1 ) * ( M - 1 ) * 2 ; 
    S = n + 1;T = n + 2;
    for(int i = 1;i <= M - 1;++i)//第一排横边 
     insert( S , i , read() );
    for(int i = 1;i <= N - 2;++i)//中间排横边 
     for(int j = 1;j <= M - 1;++j)
      insert( (M - 1) * (2 * i - 1) + j, (M - 1) * (2 * i) + j, read());
    if( N > 2 )
     for(int i = 1;i <= M - 1;++i)     //最后一排横匾 
      insert( (M - 1) * (2 *(N - 1) - 1) + i , T , read());

    get_in();
    for(int i = 1;i <= N - 1;++i) //第一排竖边 
     insert( 1 + (M - 1) * (2 * i - 1) , T , in[i][1]);
    for(int i = 1;i <= M - 2;++i) //中间排竖边 
     for(int j = 1;j <= N - 1;++j)
      insert( i + (M - 1) * (2 * (j - 1)) , i + 1 + (M - 1)*(2 * (j - 1) + 1) , in[j][i + 1]);
    if(M > 2)
     for(int i = 1;i <= N - 1;++i)//最后一排竖边
      insert( (M - 1) * ( 2 * i - 1)  , S , in[i][M]);

    for(int i = 1;i <= N - 1;++i)
     for(int j = 1;j <= M - 1;++j)
      insert( (M - 1) * (2 * (i - 1)) + j , (M - 1) * (2 * (i - 1) + 1) + j , read()); 

    return;
}
int dis[maxn];
bool flag[maxn];
void spfa(){
    queue<int>q;flag[S] = true;q.push(S);
    memset( dis , 10 ,sizeof(dis));
    dis[S] = 0;
    while(q.size()){
        int v = q.front();

        for(int i = linkk[v];i;i = e[i].n)

         if(dis[v] + e[i].v <= dis[e[i].y]){
            int y = e[i].y;
            dis[y] = dis[v] + e[i].v;
            if(!flag[y])
             q.push(y) , flag[y] = true;
         }
        flag[v] = false;
        q.pop();
    }
    printf("%d",dis[T]);
    return;
}
int main(){
    init();
    spfa();
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值