对于这种题很容易看出是费用流吧……

但这道题不容易建模;

首先是怎么表示目标状态和其实状态,看起来有黑有白很复杂

但实际上,不难发现,白色格子没什么用,起决定作用的是黑格子

也就是我们可以把问题简化:我们怎么把开始的黑格子移到目标位置

但这个移动不是一般的移动;

在一条路径中,不难发现,起始两点是只要交换一次,其他路径上个点都是要交换2次

由于题目给出的限制是点的交换次数限制c,所以不难想到要拆点

但是平常的拆两点好像无法表示这个特征,

于是我们就拆成3个点……

p1--->p0--->p2

p1表示交换进来,p2表示交换出去;

所以不难得出:

对于每个点,如果它是原图中的黑点,连边<p1,p0,c/2,0>,<p0,p2,(c+1)/2,0>,<s,p0,1,0>;

如果它是新图中的黑点,连边<p1,p0,(c+1)/2>,<p0,p2,c/2,0>,<p0,t,1,0>;

注意存在有的点在新图原图都是黑点的情况(在这里WA了一次)

如果它在两个图中都是白点,那么连边<p1,p0,c/2,0>,<p0,p2,c/2,0>

最后对于原图中任意可达两点,连边<pi2,pj1,inf,1>

注意这道题可以交换对角线,还要判断是否可行,细节挺多

 const dx:array[..] of integer=(-,,,,,-,-,);
      dy:array[..] of integer=(,,,-,,-,,-);
      inf=;
type node=record
       next,point,cost,flow:longint;
     end; var edge:array[..] of node;
    p,pre,cur,d:array[..] of longint;
    v:array[..] of boolean;
    a,b,c:array[..,..] of integer;
    q:array[..] of longint;
    ch,t,n,m,i,j,k,x,y,po,tot,sum,ans,len:longint;
    s:string; procedure add(x,y,f,w:longint);
  begin
    inc(len);
    edge[len].point:=y;
    edge[len].flow:=f;
    edge[len].cost:=w;
    edge[len].next:=p[x];
    p[x]:=len;
  end; function spfa:boolean;
  var i,x,y,f,r:longint;
  begin
    fillchar(v,sizeof(v),false);
    v[]:=true;
    for i:= to t do
      d[i]:=inf;
    d[]:=;
    f:=;
    r:=;
    q[f]:=;
    while f<=r do
    begin
      x:=q[f];
      v[x]:=false;
      i:=p[x];
      while i<>- do
      begin
        y:=edge[i].point;
        if edge[i].flow> then
          if d[y]>d[x]+edge[i].cost then
          begin
            d[y]:=d[x]+edge[i].cost;
            pre[y]:=x;
            cur[y]:=i;
            if not v[y] then
            begin
              inc(r);
              q[r]:=y;
              v[y]:=true;
            end;
          end;
        i:=edge[i].next;
      end;
      inc(f);
    end;
    if d[t]=inf then exit(false) else exit(true);
  end; procedure mincost;
  var i,j,neck:longint;
  begin
    while spfa do
    begin
      i:=t;
      neck:=inf;
      while i<> do
      begin
        j:=cur[i];
        if neck>edge[j].flow then neck:=edge[j].flow;
        i:=pre[i];
      end;
      i:=t;
      while i<> do
      begin
        j:=cur[i];
        dec(edge[j].flow,neck);
        inc(edge[j xor ].flow,neck);
        i:=pre[i];
      end;
      ans:=ans+d[t]*neck;
      dec(ch,neck);
      if ch= then break;
    end;
  end; begin
  readln(n,m);
  len:=-;
  fillchar(p,sizeof(p),);
  t:=*n*m+;
  for i:= to n do
  begin
    readln(s);
    for j:= to m do
    begin
      a[i,j]:=ord(s[j])-;
      tot:=tot+a[i,j];
    end;
  end;
  for i:= to n do
  begin
    readln(s);
    for j:= to m do
    begin
      b[i,j]:=ord(s[j])-;
      sum:=sum+b[i,j];
    end;
  end;
  for i:= to n do
  begin
    readln(s);
    for j:= to m do
      c[i,j]:=ord(s[j])-;
  end;
  if sum<>tot then
  begin
    writeln(-);
    halt;
  end
  else ch:=sum;
  sum:=n*m;
  for i:= to n do
  begin
    for j:= to m do
    begin
      x:=(i-)*m+j;
      if a[i,j]= then
      begin
        add(x+sum,x,c[i,j] shr ,);
        add(x,x+sum,,);
        add(x,x+*sum,(c[i,j]+) shr ,);
        add(x+*sum,x,,);
        add(,x,,);
        add(x,,,);
        if b[i,j]= then    //细节
        begin
          add(x,t,,);
          add(t,x,,);
        end;
      end
      else if b[i,j]= then
      begin
        add(x+sum,x,(c[i,j]+) shr ,);
        add(x,x+sum,,);
        add(x,x+*sum,c[i,j] shr ,);
        add(x+*sum,x,,);
        add(x,t,,);
        add(t,x,,);
      end
      else if (a[i,j]+b[i,j]=) then
      begin
        add(x+sum,x,c[i,j] shr ,);
        add(x,x+sum,,);
        add(x,x+*sum,c[i,j] shr ,);
        add(x+*sum,x,,);
      end;
    end;
  end;   for i:= to n do
    for j:= to m do
    begin
      po:=(i-)*m+j;
      for k:= to do
      begin
        x:=i+dx[k];
        y:=j+dy[k];
        if (x<=n) and (y<=m) and (x>) and (y>) then
        begin
          tot:=(x-)*m+y;
          add(po+*sum,tot+sum,inf,);
          add(tot+sum,po+*sum,,-);
        end;
      end;
    end;
  mincost;
  if ch<> then writeln(-) else writeln(ans);
end.

bzoj2668的更多相关文章

  1. 【BZOJ2668】[cqoi2012]交换棋子 费用流

    [BZOJ2668][cqoi2012]交换棋子 Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列 ...

  2. BZOJ2668 [cqoi2012]交换棋子 【费用流】

    题目链接 BZOJ2668 题解 容易想到由\(S\)向初始的黑点连边,由终态的黑点向\(T\)连边,然后相邻的点间连边 但是这样满足不了交换次数的限制,也无法计算答案 考虑如何满足一个点的交换次数限 ...

  3. 【BZOJ-2668】交换棋子 最小费用最大流

    2668: [cqoi2012]交换棋子 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 1055  Solved: 388[Submit][Status ...

  4. BZOJ2668: [cqoi2012]交换棋子

    题解: 可以戳这里:http://www.cnblogs.com/zig-zag/archive/2013/04/21/3033485.html 其实自己yy一下就知道这样建图的正确性了. 感觉太神奇 ...

  5. BZOJ2668:[CQOI2012]交换棋子——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=2668 https://www.luogu.org/problemnew/show/P3159#sub ...

  6. BZOJ2668:[CQOI2012]交换棋子(费用流)

    题目描述 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列的格子只能参与mi,j次交换. 输入输出格式 输入格式: 第一行 ...

  7. [CQOI2012][bzoj2668] 交换棋子 [费用流]

    题面 传送门 思路 抖机灵 一开始看到这题我以为是棋盘模型-_-|| 然而现实是骨感的 后来我尝试使用插头dp来交换,然后又惨死 最后我不得不把目光转向那个总能化腐朽为神奇的算法:网络流 思维 我们要 ...

  8. BZOJ2668: [cqoi2012]交换棋子(费用流)

    Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列的格子只能参与mi,j次交换. Input 第一行 ...

  9. [bzoj2668] [洛谷P3159] [cqoi2012] 交换棋子

    Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列的格子只能参与mi,j次交换. Input 第一行 ...

随机推荐

  1. linux centos 安装

    本着学习的目的,在自己的电脑上进行 centos 7 安装,记录下这步骤以备忘. 一.Centos 下载 centos 官方(https://www.centos.org/)下载ISO镜像(这是我的下 ...

  2. RTC搭建android下三层应用程序访问服务器MsSql-服务器端

    前几天通过Ro搭建webservice,然后在android下调用,虽然已近成功,但是返回的数据库里的中文有乱码一直未得到解决!rtc6.23版本,已经支持xe5,也支持fmx的android下开发, ...

  3. webstorm ftp发布问题

    通过webstorm发布遇到问题 Invalid descendent file name "/". 解决方案为 点击[Advanced options]勾选[always use ...

  4. FPGA初学心得

    有三种方法在模块中产生逻辑:1.使用连续赋值语句“assign”:2.用实例元件 3.用“always”块.所以在always块中赋值不能使用assign,而是直接给变量赋值就行. reg与wire的 ...

  5. 深入浅出话XAML-学习笔记

    第一章 XAML是什么? 1.1XAML之前 *设计师的设计更不上程序逻辑的变化 *程序员未能完全实现设计师提供的效果图 *效果图与程序功能不能完全匹配 *从效果图到软件UI的转化耗费很多时间 1.2 ...

  6. Webx框架自带的petstore

    Webx框架:http://openwebx.org/ petstore:webx3/webx-sample/petstore/tags/3.0/petstore 编译之后:mvn jetty:run ...

  7. (转)GDT与LDT

    网址:http://blog.csdn.net/billpig/article/details/5833980 保护模式下的段寄存器 由 16位的选择器 与 64位的段描述符寄存器 构成段描述符寄存器 ...

  8. spring dataSourceRouter自动切换数据源

    spring多数据源的切换,主要用到的是AbstractRoutingDataSource这个路由类,当我们的自定义的一个路由分发类继承AbstractRoutingDataSource类后,重写de ...

  9. GPS导航仪常见术语解释

    摘自百度百科: 坐标(coordinate) 有2维.3维两种坐标表示,当GPS能够收到4颗及以上卫星的信号时,它能计算出本地的3维坐标:经度.纬度.高度,若只能收到3颗卫星的信号,它只能计算出2维坐 ...

  10. CQRS学习——Cqrs补丁,async实验以及实现[其二]

    实验——async什么时候提高吞吐 async是一个语法糖,用来简化异步编程,主要是让异步编程在书写上接近于同步编程.总的来收,在await的时候,相当于附加上了一个.ContinueWith(). ...