BZOJ.4819.[SDOI2017]新生舞会(01分数规划 费用流SPFA)
裸01分数规划。二分之后就是裸最大费用最大流了。
写的朴素SPFA费用流,洛谷跑的非常快啊,为什么有人还T成那样。。
当然用二分也很慢,用什么什么迭代会很快。
[Update] 19.2.15
下午写的zkw费用流在BZOJ上T了= =
然而在洛谷上和以前写的跑的差不多快
当然还可以写整数二分或者KM...
输出的时候最好加个eps
,不然可以被卡比如BZOJ discuss里的数据。
第一次写的代码:
//3624kb 4016ms
#include <queue>
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
#define eps 1e-7
#define INF 1e14
const int N=205,M=90000;
int n,src,des,Enum,H[N],to[M],fr[M],nxt[M],cap[M],pre[N];
bool inq[N];
double Ans,dis[N],A[N][N],B[N][N],cost[M];
std::queue<int> q;
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline void AddEdge(int u,int v,int w,double c)
{
to[++Enum]=v, nxt[Enum]=H[u], fr[Enum]=u, H[u]=Enum, cap[Enum]=w, cost[Enum]=c;
to[++Enum]=u, nxt[Enum]=H[v], fr[Enum]=v, H[v]=Enum, cap[Enum]=0, cost[Enum]=-c;
}
bool SPFA()
{
for(int i=1; i<=des; ++i) dis[i]=-INF;
q.push(src), dis[src]=0;
while(!q.empty())
{
int x=q.front();q.pop();
inq[x]=0;
for(int v,i=H[x]; i; i=nxt[i])
if(dis[to[i]]<dis[x]+cost[i] && cap[i])
{
dis[v=to[i]]=dis[x]+cost[i], pre[v]=i;
if(!inq[v]) inq[v]=1,q.push(v);
}
}
return dis[des]>-INF;
}
void MCMF(){
for(int i=des; i!=src; i=fr[pre[i]])
--cap[pre[i]], ++cap[pre[i]^1], Ans+=cost[pre[i]];
}
bool pre_Check(double C)
{
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j)
/*if(i!=j) j是女生,不是自己。。*/
AddEdge(i,j+n,1,A[i][j]-C*B[i][j]);
for(int i=1; i<=n; ++i) AddEdge(src,i,1,0),AddEdge(i+n,des,1,0);
Ans=0;
while(SPFA()){
MCMF();
if(Ans<0) break;
}
return Ans>=0;
}
bool Check(double C)
{
int cnt=1;
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j)
cap[++cnt]=1, cost[cnt]=A[i][j]-C*B[i][j], cap[++cnt]=0, cost[cnt]=-A[i][j]+C*B[i][j];
for(int i=1; i<=n; ++i)
cap[++cnt]=1, cost[cnt]=0, cap[++cnt]=0, cost[cnt]=0,
cap[++cnt]=1, cost[cnt]=0, cap[++cnt]=0, cost[cnt]=0;
Ans=0;
while(SPFA()){
MCMF();
if(Ans<0) break;
}
return Ans>=0;
}
int main()
{
n=read(),Enum=1,src=0,des=n<<1|1;
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j) A[i][j]=read();
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j) B[i][j]=read();
double l=0,r=10000.0,mid;
if(pre_Check(mid=(l+r)*0.5)) l=mid;
else r=mid;
while(r>l+eps)
if(Check(mid=(l+r)*0.5)) l=mid;
else r=mid;
printf("%.6lf",l+eps);
return 0;
}
第二次写的代码:(BZOJ上T掉的zkw= =)
#include <queue>
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define eps 1e-7
#define gc() getchar()
typedef long long LL;
const int N=207,M=(N*N+N)*2;
const double INF=1ll<<61;
int T,Enum,cur[N],H[N],to[M],nxt[M],cap[M],A[103][103],B[103][103];
bool vis[N];
double Cost,cost[M],dis[N];
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-48,c=gc());
return now;
}
inline void AE(int u,int v)
{
to[++Enum]=v, nxt[Enum]=H[u], H[u]=Enum;
to[++Enum]=u, nxt[Enum]=H[v], H[v]=Enum;
}
void PreBuild(int n)
{
Enum=1;
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j) AE(i,j+n);
for(int i=1; i<=n; ++i) AE(0,i), AE(i+n,T);
}
bool BFS()
{
static bool inq[N];
static std::queue<int> q;
for(int i=1; i<=T; ++i) dis[i]=-INF;
q.push(0), dis[0]=0;
while(!q.empty())
{
int x=q.front(); q.pop();
inq[x]=0;
for(int i=H[x],v; i; i=nxt[i])
if(dis[v=to[i]]<dis[x]+cost[i] && cap[i])
dis[v]=dis[x]+cost[i], !inq[v]&&(q.push(v),inq[v]=1);
}
return dis[T]>-INF;
}
bool DFS(int x)
{
if(x==T) return 1;
vis[x]=1;//!!
for(int &i=cur[x]; i; i=nxt[i])
if(!vis[to[i]] && cap[i] && dis[to[i]]==dis[x]+cost[i] && DFS(to[i]))
return --cap[i], ++cap[i^1], Cost+=cost[i], 1;
return 0;
}
bool MCMF()
{
Cost=0;
while(BFS())
{
memset(vis,0,T+1), memcpy(cur,H,T+1<<2);
while(DFS(0) && Cost>=0);
if(Cost<0) break;
}
return Cost>=0;
}
bool Check(int n,double x)
{
int now=1;
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j)
cap[++now]=1, cost[now]=A[i][j]-x*B[i][j], cap[++now]=0, cost[now]=-cost[now-1];
for(int i=1; i<=n; ++i) cap[++now]=1, cap[++now]=0, cap[++now]=1, cap[++now]=0;
return MCMF();
}
int main()
{
freopen("ball.in","r",stdin);
freopen("ball.out","w",stdout);
int n=read(); T=n+n+1;
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j) A[i][j]=read();
for(int i=1; i<=n; ++i)
for(int j=1; j<=n; ++j) B[i][j]=read();
PreBuild(n);
double l=0,r=1e4+1e-4,mid;
while(l+eps<r)
if(Check(n,mid=(l+r)*0.5)) l=mid;
else r=mid;
printf("%.6lf\n",l+eps);
return 0;
}
BZOJ.4819.[SDOI2017]新生舞会(01分数规划 费用流SPFA)的更多相关文章
- 【BZOJ4819】[Sdoi2017]新生舞会 01分数规划+费用流
[BZOJ4819][Sdoi2017]新生舞会 Description 学校组织了一次新生舞会,Cathy作为经验丰富的老学姐,负责为同学们安排舞伴.有n个男生和n个女生参加舞会 买一个男生和一个女 ...
- P3705 [SDOI2017]新生舞会 01分数规划+费用流
$ \color{#0066ff}{ 题目描述 }$ 学校组织了一次新生舞会,Cathy作为经验丰富的老学姐,负责为同学们安排舞伴. 有\(n\)个男生和\(n\)个女生参加舞会买一个男生和一个女生一 ...
- BZOJ_4819_[Sdoi2017]新生舞会_01分数规划+费用流
BZOJ_4819_[Sdoi2017]新生舞会_01分数规划+费用流 Description 学校组织了一次新生舞会,Cathy作为经验丰富的老学姐,负责为同学们安排舞伴.有n个男生和n个女生参加舞 ...
- BZOJ-4819: 新生舞会(01分数规划+费用流)
Description 学校组织了一次新生舞会,Cathy作为经验丰富的老学姐,负责为同学们安排舞伴.有n个男生和n个女生参加舞会 买一个男生和一个女生一起跳舞,互为舞伴.Cathy收集了这些同学之间 ...
- [Sdoi2017]新生舞会(分数规划+费用流)
题解:二分答案mid,然后将每个位置看成a-b*mid,然后由于是n个男生和n个女生匹配,每个人搭配一个cp,于是有点类似于https://www.lydsy.com/JudgeOnline/prob ...
- 4819: [Sdoi2017]新生舞会(分数规划)
4819: [Sdoi2017]新生舞会 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1031 Solved: 530[Submit][Statu ...
- [Sdoi2017]新生舞会 [01分数规划 二分图最大权匹配]
[Sdoi2017]新生舞会 题意:沙茶01分数规划 貌似\(*10^7\)变成整数更科学 #include <iostream> #include <cstdio> #inc ...
- BZOJ4819: [Sdoi2017]新生舞会(01分数规划)
Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1029 Solved: 528[Submit][Status][Discuss] Descripti ...
- bzoj 4819: [Sdoi2017]新生舞会【二分+最小费用最大流】
如果\( b[i]==0 \)那么就是裸的费用流/KM,当然KM快一些但是为什么不写KM呢因为我不会打板子了 考虑二分答案,那么问题变成了判定问题. \[ ans=\frac {a_1+a_2+... ...
随机推荐
- C语言复习---二维数组和二级指针的关系:没关系,别瞎想(重点)
前提:一维数组和一维指针为什么可以替换使用? ] = { , , }; int *p = a; ; i < ; i++) printf("%d ", *(p + i)); 上 ...
- *p++、(*p)++、*++p、++*p 的区别
int a[5]={1,2,3,4,5};int *p = a; *p++ 先取指针p指向的值(数组第一个元素1),再将指针p自增1: cout << *p++; // 结果为 1 cou ...
- 单字节编码&双字节编码
单字节编码(WINDOWS-1252.ISO-8859-1.UTF-8) 双字节编码(UTF-16) Windows 记事本默认会将文件保存为单字节的 ANSI(ASCII).如果您选择 " ...
- 在C#中使用.NET SDK创建控制
下载示例工程 - 8 Kb 介绍 在这篇教程中,我将使用.NET架构创建一个简单的时钟控制示例,这个控制是一个显示当前时间的时钟,我将指导读者实现秒针并显示钟点数.文章加亮处是创建这个控制的关键点,读 ...
- iOS常用小功能
CHENYILONG Blog 常用小功能 技术博客http://www.cnblogs.com/ChenYilong/ 新浪微博http://weibo.com/luohanchenyilong ...
- Basic(消息)的一些属性及方法
AMQP协议:是一个金融级的消息队列,确保消息万无一失 1.消息发布端的确认 手动确认消息是否已经发送 场景:发布消息到RabbitMQ中,我们需要知道这个消息是否发布成功了. *发布确认影响性能 c ...
- sync 解释
sync命令用于强制被改变的内容立刻写入磁盘,更新超块信息. 在Linux/Unix系统中,在文件或数据处理过程中一般先放到内存缓冲区中,等到适当的时候再写入磁盘,以提高系统的运行效率.sync命令则 ...
- Owin中间件动手做
摘要:本文目的是了解Owin基本原理.讲述如何从控制台创建一个自宿主的OwinHost,然后再编写一两个中间件 准备工作 首先通过VisualStudio创建一个控制台应用 然后添加Owin的Nuge ...
- Docker手动搭建sentry错误日志系统
Sentry介绍 在开发过程中,我们通过debug来排查bug,并且使用logging来记录系统的错误.但是logging有很多不足: 必须登陆到服务器查看日志文件 需要主动去查询 输出日志方式无法把 ...
- thymeleaf:访问静态方法
<p class="left tel" th:if="${#strings.startsWith(T(net.common.util.tool.common.Req ...