终于忙完期末考试了,即将进入愉快的暑假(虽然暑假作业奇多,但好歹终于能有大量时间刷题了)

先把上次新一类最小割留下的一道题目A了复习一下;

题目看起来很复杂,实际上和bzoj2132是同一个类型的

用S集合表示放正能量,T集合表示放负能量(可自行参照上一篇文章)

只是有几个不同点:

  1. 有些点是已经确定为S或T集合中的点

  2. 选择点属于S或者属于T都是没有收益的

对于未知的问题我们要转化为已知的问题来求解

首先我们来处理2,考虑到割是我们所得不到的收益,

于是我们可以把每个点选择属于S或者属于T产生的收益都当做是个1,

也就是对于每个点i,连接s--->i, i--->t 容量为1;

最后ans=n*n*n(抵消我们之前强加的收益)+图中的边数-mincut;

下面就是1没有解决了,考虑到割是我们所得不到的收益,

所以,对于那些已经确定为S集合中的点(T集合同理),我们只要使

s--->i 容量为inf,也就让这条边一定不成为割边,那么就满足了这个点的确定性;

好了,即使是立体的,依然可以黑白染色划分二分图,所以其余实现细节类似bzoj2132,这里不加赘述。

 const inf=;
      dx:array[..] of integer=(-,,,,,);
      dy:array[..] of integer=(,,-,,,);
      dz:array[..] of integer=(,,,,-,);
type node=record
       next,flow,point:longint;
     end; var edge:array[..] of node;
    h,numh,p,cur,d,pre:array[..] of longint;
    num:array[..,..,..] of longint;
    t,len,i,j,n,k,x,y,z,m,ans,sum:longint;
    s:ansistring; function min(a,b:longint):longint;
  begin
    if a>b then exit(b) else exit(a);
  end; procedure add(x,y,f:longint);
  begin
    inc(len);
    edge[len].point:=y;
    edge[len].flow:=f;
    edge[len].next:=p[x];
    p[x]:=len;
  end; function sap:longint;
  var u,i,j,q,neck,tmp:longint;
  begin
    numh[]:=t+;
    for i:= to t do
      cur[i]:=p[i];
    u:=;
    sap:=;
    neck:=inf;
    while h[]<t+ do
    begin
      d[u]:=neck;
      i:=cur[u];
      while i<>- do
      begin
        j:=edge[i].point;
        if (edge[i].flow>) and (h[u]=h[j]+) then
        begin
          cur[u]:=i;
          pre[j]:=u;
          neck:=min(neck,edge[i].flow);
          u:=j;
          if u=t then
          begin
            while u<> do
            begin
              u:=pre[u];
              i:=cur[u];
              dec(edge[i].flow,neck);
              inc(edge[i xor ].flow,neck);
            end;
            sap:=sap+neck;
          end;
          break;
        end;
        i:=edge[i].next;
      end;
      if i=- then
      begin
        dec(numh[h[u]]);
        if numh[h[u]]= then exit;
        tmp:=t;
        i:=p[u];
        q:=-;
        while i<>- do
        begin
          j:=edge[i].point;
          if (edge[i].flow>) and (h[j]<tmp) then
          begin
            q:=i;
            tmp:=h[j];
          end;
          i:=edge[i].next;
        end;
        h[u]:=tmp+;
        inc(numh[h[u]]);
        cur[u]:=q;
        if u<> then
        begin
          u:=pre[u];
          neck:=d[u];
        end;
      end;
    end;
  end; begin
  len:=-;
  fillchar(p,sizeof(p),);
  readln(n);
  t:=n*n*n+;
  for i:= to n do
  begin
    for j:= to n do
    begin
      readln(s);
      for k:= to n do
      begin
        inc(m);
        num[i,j,k]:=m;
        if (s[k]='N') and ((i+j+k) mod =) or (s[k]='P') and ((i+j+k) mod =) then
        begin
          add(,m,inf);
          add(m,,);
          add(m,t,);
          add(t,m,);
        end
        else if (s[k]='N') and ((i+j+k) mod =) or (s[k]='P') and ((i+j+k) mod =) then
        begin
          add(m,t,inf);
          add(t,m,);
          add(,m,);
          add(m,,);
        end
        else if s[k]='?' then
        begin
          add(,m,);
          add(m,,);
          add(m,t,);
          add(t,m,);
        end;
      end;
    end;
    if i<>n then readln;
  end;
  for i:= to n do
    for j:= to n do
      for k:= to n do
        for m:= to do
        begin
          x:=i+dx[m];
          y:=j+dy[m];
          z:=k+dz[m];
          if num[x,y,z]> then
          begin
            add(num[i,j,k],num[x,y,z],);
            add(num[x,y,z],num[i,j,k],);
            inc(sum);
          end;
        end;
  sum:=sum div +n*n*n;
  writeln(sum-sap);
end.

bzoj1976的更多相关文章

  1. 【BZOJ1976】[BeiJing2010组队]能量魔方 Cube 最小割

    [BZOJ1976][BeiJing2010组队]能量魔方 Cube Description 小C 有一个能量魔方,这个魔方可神奇了,只要按照特定方式,放入不同的 能量水晶,就可以产生巨大的能量. 能 ...

  2. 【BZOJ-1976】能量魔方Cube 最小割 + 黑白染色

    1976: [BeiJing2010组队]能量魔方 Cube Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 884  Solved: 307[Submi ...

  3. BZOJ1976: [BeiJing2010组队]能量魔方 Cube

    1976: [BeiJing2010组队]能量魔方 Cube Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 832  Solved: 281[Submi ...

  4. 【BZOJ1976】能量魔方 [最小割]

    能量魔方 Time Limit: 10 Sec  Memory Limit: 64 MB[Submit][Status][Discuss] Description 小C 有一个能量魔方,这个魔方可神奇 ...

  5. 【bzoj1976】[BeiJing2010组队]能量魔方 Cube 网络流最小割

    题目描述 一个n*n*n的立方体,每个位置为0或1.有些位置已经确定,还有一些需要待填入.问最后可以得到的 相邻且填入的数不同的点对 的数目最大. 输入 第一行包含一个数N,表示魔方的大小. 接下来 ...

  6. 二分图&网络流&最小割等问题的总结

    二分图基础: 最大匹配:匈牙利算法 最小点覆盖=最大匹配 最小边覆盖=总节点数-最大匹配 最大独立集=点数-最大匹配 网络流: 技巧: 1.拆点为边,即一个点有限制,可将其转化为边 BZOJ1066, ...

随机推荐

  1. 暑假集训(3)第一弹 -----还是畅通工程(hdu1233)

    题意梗概:N(n<100)个村子想要富起来,自然就要先修路,不过到底还是没富起来,所以陷入了一个怪圈 :资金不足->修不起路->资金不足...... 为了实现走向全民小康社会,全面实 ...

  2. (CodeForces 510C) Fox And Names 拓扑排序

    题目链接:http://codeforces.com/problemset/problem/510/C Fox Ciel is going to publish a paper on FOCS (Fo ...

  3. makefile-0711-168 SEVERE ERROR: Input file:

    ld: 0711-168 SEVERE ERROR: Input file: /cicm/commlib/include        Input files must be regular file ...

  4. mysql的1045解决方法

    mysql的连接方式有两种: UNIX域套接字连接,如: mysql -u root -p mysql -h localhost -u root -p TCP/IP套接字连接,如: mysql -h ...

  5. position:absolute,绝对定位和相对定位,JQ隐藏和显示

    需要在指定位置,用绝对定位. 如果直接写position:absolute,top:0;left:0,那就是以浏览器的左上角为参照了 现在需要在某一个指定位置用绝对定位 解决方法 在需要用绝对定位(p ...

  6. platform_driver_register()--如何match之后调用probe

    int platform_driver_register(struct platform_driver *drv) { drv->driver.bus = &platform_bus_t ...

  7. Hive 自定义函数(转)

    Hive是一种构建在Hadoop上的数据仓库,Hive把SQL查询转换为一系列在Hadoop集群中运行的MapReduce作业,是MapReduce更高层次的抽象,不用编写具体的MapReduce方法 ...

  8. merge into 和 update 的效率对比

    以前只考虑 merge into 只是在特定场合下方便才使用的,今天才发现,merge into 竟然会比 update 在更新数据时有这么大的改进.其实呢,merge into部分的update和u ...

  9. 数据库获取前N条记录SQL Server与SQLite的区别

    在使用sql语句进行前20条记录查询时SQL Server可以这样写: 1: select top 20 * from [table] order by ids desc 2: select top ...

  10. String类中常用的操作

    一.获取: 1.获取字符串的长度(注意是方法,不是跟数组的属性一样的) int length(); 1 public static void getLength(){ 2 String s = &qu ...