一开始我觉得这不是一个弱弱的费用流吗?

每个豆豆拆点,入点出点随便连连

由于肯定是DAG图,边权为正的最大费用肯定能增广出来

于是我们只要跑总流量为2的最大费用最大流不就行了吗

但是

这样会TLE,因为会出现稠密图,spfa跑稠密图太慢

既然这样,能不能换一个找增广路的方法呢?

最大路径更快的方法就是DAG图的拓扑排序+dp

但好像无法处理反向弧导致带来的环啊

那既然这样,反正总流量很小,

我们可以用DAG解决最大路径特有的dp找第一条增广路(因为这时候是肯定的DAG),跑出一个较大费用

然后再用spfa增广呢?这样就大大减小了时间复杂度

这就是AC方法

注意这道题空间卡的较紧,能开integer不要开longint

实现起来要小心

 const mo=;
type node=record
       next:longint;
       point,flow,cost:integer;
     end; var edge:array[..] of node;
    v:array[..] of boolean;
    d,rd,pre,pref:array[..] of integer;
    q:array[..] of integer;
    p,x,y,cur:array[..] of longint;
    len:longint;
    max,s,t,i,j,n,m,be:integer; procedure add(x,y,f,c:integer);
  begin
    edge[len].point:=y;
    edge[len].flow:=f;
    edge[len].cost:=c;
    edge[len].next:=p[x];
    p[x]:=len;
  end; procedure preflow;
  var x,y,f,r,i,j:longint;
  begin
    r:=;
    f:=;
    q[]:=s;
    while f<=r do  //拓扑排序
    begin
      x:=q[f];
      i:=p[x];
      while i<>- do
      begin
        y:=edge[i].point;
        dec(rd[y]);
        if (rd[y]=) then
        begin
          inc(r);
          q[r]:=y;
        end;
        i:=edge[i].next;
      end;
      inc(f);
    end;
    fillchar(d,sizeof(d),);
    d[s]:=;
    for j:= to r do //DAG图根据拓扑的顺序dp解决最大路径
    begin
      x:=q[j];
      i:=p[x];
      while i<>- do
      begin
        y:=edge[i].point;
        if d[y]<d[x]+ then
        begin
          d[y]:=d[x]+;
          pre[y]:=x;
          cur[y]:=i;
          if d[y]>d[max] then max:=y;
        end;
        i:=edge[i].next;
      end;
    end;
  end; procedure change;   //建立预处理后的网络继续增广
  var i,j,e,w:longint;
  begin
    for i:= to n do
      pref[i]:=;
    i:=max;
    while i<>s do   //相当于找到第一条增广路然后弄一弄反向弧什么的
    begin
      pref[i]:=;
      j:=cur[i];
      dec(edge[j].flow);
      if pre[i]=s then w:=i;
      i:=pre[i];
    end;
    len:=-;
    for i:= to n do
      for j:= to n do
        if (i<>j) then
          if (x[i]<=x[j]) and (y[i]<=y[j]) then
          begin
            inc(len);
            edge[len].point:=j+n;
            inc(len);
            add(j+n,i,-edge[len-].flow,)  //添加原来没加的反向弧
          end;     p[s]:=-;
    for i:= to n do   //下面很多处理的细节,不赘述,繁杂但不难想到
    begin
      if i=w then e:= else e:=;
      inc(len);
      add(s,i+n,e,);
      inc(len);
      add(i+n,s,-e,);
    end;
// 表示超级源点,s表示起点,t表示超级汇点
    t:=*n+;
    inc(len);
    add(,s,,);
    inc(len);
    add(s,,,);
    for i:= to n do   
    begin
      inc(len);
      add(i+n,i,pref[i],);
      inc(len);
      add(i,i+n,-pref[i],-);
      if max=i then e:=   
      else e:=;
      inc(len);
      add(i,t,e,);
      inc(len);
      add(t,i,-e,);
    end;
  end; function spfa:boolean;
  var f,r,x,y,i,j,head,tail:longint;
  begin
    fillchar(v,sizeof(v),false);
    for i:= to t do
      d[i]:=-mo;
    d[]:=;
    q[]:=;
    f:=;
    r:=;
    head:=;
    tail:=;
    while head<=tail do  //缩空间的写法,好久没这么写了
    begin
      x:=q[f];
      v[x]:=false;
      i:=p[x];
      while i<>- do
      begin
        y:=edge[i].point;
        if edge[i].flow> then
        begin
          if d[x]+edge[i].cost>d[y] then
          begin
            d[y]:=edge[i].cost+d[x];
            pre[y]:=x;
            cur[y]:=i;
            if not v[y] then
            begin
              v[y]:=true;
              r:=(r+) mod mo;
              inc(tail);
              q[r]:=y;
            end;
          end;
        end;
        i:=edge[i].next;
      end;
      f:=(f+) mod mo;
      inc(head);
    end;
    if d[t]=-mo then exit(false) else exit(true);
  end; function mincost:longint;  //最大费用最大流
  var i,j:longint;
  begin
    be:=d[max];
    mincost:=be;   //预处理出来的费用
    while spfa do
    begin
      mincost:=mincost+d[t];
      i:=t;
      while i<> do
      begin
        j:=cur[i];
        dec(edge[j].flow);
        inc(edge[j xor ].flow);
        i:=pre[i];
      end;
    end;
  end; begin
  len:=-;
  fillchar(p,sizeof(p),);
  readln(n);
  for i:= to n do
    readln(x[i],y[i]);
  for i:= to n do
    for j:= to n do
      if (i<>j) then
        if (x[i]<=x[j]) and (y[i]<=y[j]) then
        begin
          len:=len+;
          add(i,j,,);  //个人感觉这样建之后添加边更容易一点
          inc(rd[j]);
        end;   s:=*n+;
  for i:= to n do
    if rd[i]= then
    begin
      len:=len+;
      add(s,i,,);
      inc(rd[i]); 
    end;   preflow;
  change;
  writeln(mincost);
end.

bzoj1930的更多相关文章

  1. 【BZOJ1930】[Shoi2003]pacman 吃豆豆 最大费用最大流

    [BZOJ1930][Shoi2003]pacman 吃豆豆 Description 两个PACMAN吃豆豆.一开始的时候,PACMAN都在坐标原点的左下方,豆豆都在右上方.PACMAN走到豆豆处就会 ...

  2. BZOJ1930 [Shoi2003]pacman 吃豆豆

     dp,首先建出图,f[i][j]表示a吃到了i点,b吃到了j点的最大值,转移的时候转移拓扑序小的那一维,如果i拓扑序小于j,那么转移到f[k][j],否则转移到f[i][k],建出的图边数也要优化, ...

  3. 【BZOJ1930】【SHOI2003】吃豆豆

    初见杀…… 原题: 两个PACMAN吃豆豆.一开始的时候,PACMAN都在坐标原点的左下方,豆豆都在右上方.PACMAN走到豆豆处就会吃掉它.PACMAN行走的路线很奇怪,只能向右走或者向上走,他们行 ...

  4. [转载]hzwer的bzoj题单

    counter: 664BZOJ1601 BZOJ1003 BZOJ1002 BZOJ1192 BZOJ1303 BZOJ1270 BZOJ3039 BZOJ1191 BZOJ1059 BZOJ120 ...

  5. BZOJ刷题列表【转载于hzwer】

    沿着黄学长的步伐~~ 红色为已刷,黑色为未刷,看我多久能搞完吧... Update on 7.26 :之前咕了好久...(足见博主的flag是多么emmm......)这几天开始会抽时间刷的,每天几道 ...

随机推荐

  1. 2016/01/19 javascript学习笔记-name属性

    1. name属性只在少数html元素中有效:包括表单.表单元素.<iframe>和<img>元素. 基于name属性的值选取html元素,可以使用document对象的get ...

  2. C语言中的%0nd,%nd,%-nd

    C语言中的%0nd printf --> formatted print/格式化输出 一.十进制 d -> decimal/十(shí)进制 int a=1; int b=1234; do ...

  3. C# 调用load事件

    在一个函数或者事件中调用另外的事件,例如调用Load事件 private void EventForm_Load(object sender, EventArgs e) { //相关内容 } priv ...

  4. WTL 中的常见问题汇总

    1.CRect,CPoint,CSize的使用 WTL提供了CString,CRect,CPoint和CSize,可能后来版本的ATL也提供了,WTL作者推荐使用ATL的实现,所以:#include ...

  5. Android_Json实例

    概要: 最近由于自己的兴趣,想在Android开发一个自己的App,需要使用服务器,所以交换数据是逃不掉了的,但是学生党没有固定的服务器,因此使用的新浪的SAE,在学习的前期下可以尝试一下,挺不错的一 ...

  6. C#关于编码、解码相关问题

    编码.解码技术是我们在程序中开发中经常使用到的,对一些敏感信息的存储,比如密码之类的,我们一般是不会直接以明文直接存储到数据库的,而是会通过各种算法,可以是现成的MD5(一种散列算法).或者是Hash ...

  7. 如何使用ERStudio 生成comment

    在ER使用中,在生成sql过程中,如何批量生成字段描述,如何批量添加Owner,请看下文: 1.ER生成字段描述 2.ER生成描述添加Owner 使用的ER版本是8.0,英文版本,在操作过程中,有些配 ...

  8. js简单实现删除记录时的提示效果

    删除记录时的提示效果,挺人性化的,实现的方法有很多,在本文为大家介绍下使用js是如何实现的 样式 复制代码代码如下: <style type="text/css">  ...

  9. JS获得QQ号码的昵称,头像,生日

    这篇文章主要介绍了JS获得QQ号码的昵称,头像,生日的简单实例,有需要的朋友可以参考一下 http://r.qzone.qq.com/cgi-bin/user/cgi_personal_card?ui ...

  10. win7环境下配置Java环境

    ==下载Java SE Development Kit 8u45== http://www.oracle.com/technetwork/java/javase/downloads/jdk8-down ...