Description

  有N个正整数,需要从中选出一些数,使这些数的和最大。
  若两个数a,b同时满足以下条件,则a,b不能同时被选
  1:存在正整数C,使a*a+b*b=c*c
  2:gcd(a,b)=1

Input

  第一行一个正整数n,表示数的个数。
  第二行n个正整数a1,a2,?an。

Output

  最大的和。

Sample Input

5
3 4 5 6 7

Sample Output

22

HINT

  n<=3000。

Source

  网络流

Solution

  所以这道题a的数据范围是什么......用long long可以过,不知道int行不行。

  嗯,把所有有关系的数字连一条边,这道题就变成了选出一些点使这些点两两没有边相连,求最大点权。这样就变成了最大点权独立集问题。

  然后好像这个图一定是二分图,就可以用网络流做了。如果不是二分图就是NP问题了233。

  有一个奇怪的定理:最大点权独立集 = 总权值 - 最小点权覆盖集 = 总权值 - 最小割 = 总权值 - 最大流。

  好像不只一个定理,怪我咯。

  把每个数字拆成两个点,从源点连向一个点,另一个点连向汇点,边权均为数字大小。然后把有关系的点之间连一条边,边权无限大。

  跑一遍最大流,答案就是总权值 - 最大流 / 2。

 #include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll INF = ;
struct edge
{
int v, nxt;
ll w;
}e[];
queue<int> Q;
ll d[];
int fst[], etot = , sss, ttt, level[]; void addedge(int u, int v, ll w)
{
e[++etot] = (edge){v, fst[u], w}, fst[u] = etot;
} bool istri(ll a, ll b)
{
ll c = (ll)sqrt(a * a + b * b + 0.1);
return c * c == a * a + b * b;
} ll gcd(ll a, ll b)
{
return b ? gcd(b, a % b) : a;
} int BFS()
{
memset(level, , sizeof(level));
Q.push(sss), level[sss] = ;
while(!Q.empty())
{
int u = Q.front();
Q.pop();
for(int i = fst[u]; i; i = e[i].nxt)
if(!level[e[i].v] && e[i].w)
Q.push(e[i].v), level[e[i].v] = level[u] + ;
}
return level[ttt];
} ll Dinic(int u, ll lim)
{
ll tmp = lim;
if(u == ttt) return lim;
for(int i = fst[u]; i; i = e[i].nxt)
if(level[e[i].v] == level[u] + && e[i].w)
{
ll flow = Dinic(e[i].v, min(tmp, e[i].w));
e[i].w -= flow, e[i ^ ].w += flow;
if(!(tmp -= flow)) break;
}
if(tmp == lim) level[u] = ;
return lim - tmp;
} int main()
{
int n;
ll ans = ;
cin >> n;
sss = (n << ) + , ttt = (n << ) + ;
for(int i = ; i <= n; i++)
cin >> d[i];
for(int i = ; i <= n; i++)
{
addedge(sss, i, d[i]), addedge(i, sss, );
addedge(i + n, ttt, d[i]), addedge(ttt, i + n, );
}
for(int i = ; i <= n; i++)
for(int j = ; j <= n; j++)
if(istri(d[i], d[j]) && gcd(d[i], d[j]) == )
addedge(i, j + n, INF), addedge(j + n, i, );
while(BFS())
ans += Dinic(sss, INF);
ans = -(ans >> );
for(int i = ; i <= n; i++)
ans += d[i];
cout << ans << endl;
return ;
}

[BZOJ3275] Number (网络流)的更多相关文章

  1. [BZOJ3275]Number解题报告|网络流

    Description 有N个正整数,需要从中选出一些数,使这些数的和最大.若两个数a,b同时满足以下条件,则a,b不能同时被选1:存在正整数C,使a*a+b*b=c*c2:gcd(a,b)=1 这道 ...

  2. bzoj3275: Number

    最小割...然后推一下可知不能的情况必定为一奇一偶,于是s->奇->偶->t.跑最小割即可. #include<cstdio> #include<cstring&g ...

  3. bzoj3275: Number(最小割)

    3275: Number 题目:传送门 题解: 双倍经验@bzoj3158 代码: #include<cstdio> #include<cstring> #include< ...

  4. 【最小割】【Dinic】bzoj3275 Number

    每个点拆点,分别向源/汇连a[i]的边,满足条件的相互连INF的边,答案为sum-maxflow*2. 因为若有几个点不能同时被选,我们要贪心地选择其中和尽量大的部分,这可以由最小割来保证. #inc ...

  5. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  6. 【BZOJ-3275&3158】Number&千钧一发 最小割

    3275: Number Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 748  Solved: 316[Submit][Status][Discus ...

  7. 【BZOJ3275】Number 最小割

    [BZOJ3275]Number Description 有N个正整数,需要从中选出一些数,使这些数的和最大.若两个数a,b同时满足以下条件,则a,b不能同时被选1:存在正整数C,使a*a+b*b=c ...

  8. 【POJ2699】The Maximum Number of Strong Kings(网络流)

    Description A tournament can be represented by a complete graph in which each vertex denotes a playe ...

  9. POJ 2699 The Maximum Number of Strong Kings ——网络流

    一定存在一种最优方案,使得分数前几个人是SK 所以我们可以二分答案或者枚举,然后就是经典的网络流建模. 另:输入很Excited #include <cstdio> #include &l ...

随机推荐

  1. Oracle数据库中SCOTT用户下的默认表

    ①EMP(雇员表): ②DEPT(部门表): ③BONUS(奖金表): ④SALGRADE(工资等级表):

  2. PHP常见面试题总结

    1.include 和 require 都能把另外一个文件包含到当前文件中 他们有什么区别?include 和 include_once 又有什么区别? 二者区别只有一个,那就是对包含文件的需求程度 ...

  3. Android Stdio 中的Rendering Problems Android N requires the IDE to be running with Java 1.8 or later Install a supported JDK解决办法

    出现如下图所示的错误 解决办法为: 然后在里面输入SDK 下载 下载APILevel为23版本的SDK 换成23版本的SDK 完美解决问题

  4. 用Open SSH生成公钥和私钥(Win)

    也可以使用 dsa 加密算法进行加密,命令如下: ssh-keygen -t dsa

  5. hihoCoder 1493 : 歌德巴赫猜想 素数筛法

    题意:哥德巴赫猜想认为"每一个大于2的偶数,都能表示成两个质数之和".给定一个大于2的偶数N,你能找到两个质数P和Q满足P<=Q并且P+Q=N吗?如果有多组解,输出P最小的一 ...

  6. WinSock WSAEventSelect 模型总结

    前言 本文配套代码:https://github.com/TTGuoying/WSAEventSelect-model 由于篇幅原因,本文假设你已经熟悉了利用Socket进行TCP/IP编程的基本原理 ...

  7. 标准库bind函数中使用占位符placeholders

    placeholders ,占位符.表示新的函数对象中参数的位置.当调用新的函数对象时,新函数对象会调用被调用函数,并且其参数会传递到被调用函数参数列表中持有与新函数对象中位置对应的占位符. 举个例子 ...

  8. mysql学习笔记02 CRUD操作

    添加数据insert into 表名(字段列表) values(对应字段的列表值) 查询数据 select *from 表名 where 条件select *from 表名 where 1条件 1表示 ...

  9. 安装STS报错(三)

    安装STS报错 1.具体报错如下 Failure to transfer org.codehaus.plexus:plexus-archiver:jar:1.2 from http://repo.ma ...

  10. ORA-00900: invalid SQL statement

    1.错误描述 SQL> startup;        startup        ORA-00900: invalid SQL statement 2.错误原因 3.解决办法