10分算法:对于城市网络为一条单向链的数据,

20分算法:对于n<=20的数据,暴力搜出所有的可能路径。 结合以上可以得到30分。

60分算法:分析题意可得使者会带着去的城市也就是这个城市所在强联通分量的其他城市,这个过程的代价也就是这个强联通分量的城市数-1,且他可以选择任何一个其中的城市离开这个强联通分量。于是我们求出所有强联通分量,记录下每一个包含的城市数,然后缩点。接下来再用dfs,由于数据是构造的,只能得到60分。

100分算法:在缩点之后,这个图变成了一个有向无环图,我们将一条边连向的SCC包含的城市数看做边权(<--其实一条边的边权是一个pair,pair.first为其连向的SCC包含的城市数;pair.second为原本的边权+其连向的SCC包含的城市数-1,这样问题就转化为了求一个有向无环图上的最长路径,利用DP或者Spfa或者其他的方法求解即可,本题没有构造卡Spfa的数据。

比赛的时候写的缩点+暴搜,被卡成60分。

//下划线为官方题解。

 #include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std;
#define M 500001
#define N 100001
struct Node
{
int v,w;
Node(const int &a,const int &b){v=a;w=b;}
Node(){}
};
struct Edge
{
int u,v,w;
Edge(const int &a,const int &b,const int &c){u=a;v=b;w=c;}
Edge(){}
}edges[M];
vector<Node>ljb[M];
vector<int>vs,G[M],rG[M];
typedef vector<int>::iterator ITER;
typedef vector<Node>::iterator Edge_ITER;
int sum,n,m,x,y,z,anscity,cnt[N],cmp[N],anscost=;
bool vis[N];
void dfs(int U)
{
vis[U]=;
for(ITER it=G[U].begin();it!=G[U].end();it++)
if(!vis[*it])
dfs(*it);
vs.push_back(U);
}
void rdfs(int U)
{
vis[U]=;
cmp[U]=sum;
cnt[sum]++;
for(ITER it=rG[U].begin();it!=rG[U].end();it++)
if(!vis[*it])
rdfs(*it);
}
void scc()
{
for(int i=;i<=n;i++) if(!vis[i]) dfs(i);
memset(vis,,sizeof(vis));
ITER it=vs.end(); it--;
for(;;it--)
{
if(!vis[*it])
{
sum++;
rdfs(*it);
}
if(it==vs.begin()) break;
}
}
void dfs3(int U,int city,int cost)
{
if(city>anscity)
{
anscity=city;
anscost=cost;
}
else if(city==anscity && cost<anscost)
anscost=cost;
for(Edge_ITER it=ljb[U].begin();it!=ljb[U].end();it++)
dfs3((*it).v,city+cnt[(*it).v],cost+(*it).w+cnt[(*it).v]-);
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++)
{
scanf("%d%d%d",&x,&y,&z);
G[x].push_back(y);
rG[y].push_back(x);
edges[i]=Edge(x,y,z);
}
scc();
for(int i=;i<=m;i++)
if(cmp[edges[i].u]!=cmp[edges[i].v])
ljb[cmp[edges[i].u]].push_back(Node(cmp[edges[i].v],edges[i].w));
dfs3(cmp[],cnt[cmp[]],cnt[cmp[]]-);
printf("%d %d\n",anscity,anscost);
return ;
}

后来写的缩点+spfa,要注意所谓的spfa最长路是双关键字比较的,优先让pair.first最大化,再让pair.second最小化,重载下运算符就可以了。

 #include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
#include<queue>
using namespace std;
#define INF 2147483647
#define M 500001
#define N 100001
queue<int>q;
struct Node
{
int v,w;
Node(const int &a,const int &b){v=a;w=b;}
Node(){}
}dis[N];
bool operator < (const Node &a,const Node &b)
{return a.v!=b.v ? a.v<b.v : a.w>b.w;}
Node operator + (const Node &a,const Node &b)
{return Node(a.v+b.v,a.w+b.w);}
struct Edge
{
int u,v,w;
Edge(const int &a,const int &b,const int &c){u=a;v=b;w=c;}
Edge(){}
}edges[M];
vector<Edge>ljb[M];
vector<int>vs,G[M],rG[M];
typedef vector<int>::iterator ITER;
typedef vector<Edge>::iterator Edge_ITER;
int sum,n,m,x,y,z,anscity,cnt[N],cmp[N],anscost=;
Node ans;
bool vis[N],inq[N];
void dfs(int U)
{
vis[U]=;
for(ITER it=G[U].begin();it!=G[U].end();it++)
if(!vis[*it]) dfs(*it);
vs.push_back(U);
}
void rdfs(int U)
{
vis[U]=; cmp[U]=sum; cnt[sum]++;
for(ITER it=rG[U].begin();it!=rG[U].end();it++)
if(!vis[*it]) rdfs(*it);
}
void scc()
{
for(int i=;i<=n;i++) if(!vis[i]) dfs(i);
memset(vis,,sizeof(vis));
ITER it=vs.end(); it--;
for(;;it--)
{
if(!vis[*it])
{
sum++;
rdfs(*it);
}
if(it==vs.begin()) break;
}
}
void spfa(const int &s)//Node.v权最大化的城市数,Node.w权为最小化的cost
{
for(int i=;i<=sum;i++) dis[i]=Node(-INF,INF);
dis[]=Node(cnt[s],cnt[s]-);
inq[]=; ans=Node(-,);
q.push();
while(!q.empty())
{
int cur=q.front();
for(Edge_ITER it=ljb[cur].begin();it!=ljb[cur].end();it++)
if(dis[(*it).u]<dis[cur]+Node((*it).v,(*it).w))
{
dis[(*it).u]=dis[cur]+Node((*it).v,(*it).w);
if(!inq[(*it).u])
{
q.push((*it).u);
inq[(*it).u]=;
}
}
q.pop(); inq[cur]=;
}
for(int i=;i<=n;i++) ans=max(ans,dis[i]);
printf("%d %d\n",ans.v,ans.w);
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++)
{
scanf("%d%d%d",&x,&y,&z);
G[x].push_back(y);
rG[y].push_back(x);
edges[i]=Edge(x,y,z);
}
scc();
for(int i=;i<=m;i++)
if(cmp[edges[i].u]!=cmp[edges[i].v])
ljb[cmp[edges[i].u]].push_back(Edge(cmp[edges[i].v],cnt[cmp[edges[i].v]],edges[i].w+cnt[cmp[edges[i].v]]-));
spfa();
return ;
}

【强联通分量缩点】【最长路】【spfa】CH Round #59 - OrzCC杯NOIP模拟赛day1 队爷的讲学计划的更多相关文章

  1. 【强联通分量缩点】【Tarjan】bzoj1051 [HAOI2006]受欢迎的牛

    就是看是否有一些点,从其他任何点出发都可到达 定理:有向无环图中唯一出度为0的点,一定可以由任何点出发均可达. 所以缩点,若出度为零的点(强联通分量)唯一,则答案为该强联通分量中点的度数. 若不唯一, ...

  2. Tarjan求强联通分量+缩点

    提到Tarjan算法就不得不提一提Tarjan这位老人家 Robert Tarjan,计算机科学家,以LCA.强连通分量等算法闻名.他拥有丰富的商业工作经验,1985年开始任教于普林斯顿大学.Tarj ...

  3. 【最小割】【Dinic】【强联通分量缩点】bzoj1797 [Ahoi2009]Mincut 最小割

    结论: 满足条件一:当一条边的起点和终点不在 残量网络的 一个强联通分量中.且满流. 满足条件二:当一条边的起点和终点分别在 S 和 T 的强联通分量中.且满流.. 网上题解很多的. #include ...

  4. 【强联通分量缩点】【最短路】【spfa】bzoj1179 [Apio2009]Atm

    缩点后转化成 DAG图上的单源最长路问题.spfa/dp随便. #include<cstdio> #include<queue> #include<algorithm&g ...

  5. 【强联通分量缩点】【搜索】bzoj2208 [Jsoi2010]连通数

    两次dfs缩点,然后n次dfs暴搜. #include<cstdio> #include<vector> #include<cstring> using names ...

  6. 【POJ 1236 Network of Schools】强联通分量问题 Tarjan算法,缩点

    题目链接:http://poj.org/problem?id=1236 题意:给定一个表示n所学校网络连通关系的有向图.现要通过网络分发软件,规则是:若顶点u,v存在通路,发给u,则v可以通过网络从u ...

  7. [BZOJ1051] [HAOI2006] 受欢迎的牛 (强联通分量)

    Description 每一头牛的愿望就是变成一头最受欢迎的牛.现在有N头牛,给你M对整数(A,B),表示牛A认为牛B受欢迎. 这 种关系是具有传递性的,如果A认为B受欢迎,B认为C受欢迎,那么牛A也 ...

  8. poj 1236 强联通分量

    大致题意给你有一个点数为n<=100的有向图. 求解两个子任务: 1:最少给多少个点信息,这些点的信息可以顺着有向边传遍全图. 2:最少要加多少条边,使得整个图强联通. 求强联通分量再缩点后得到 ...

  9. POJ 2186 强联通分量

    点击打开链接 题意:牛A喜欢牛B,若牛B喜欢牛C,则牛A喜欢牛C,问最后多少牛被其它全部牛喜欢 思路:用强联通分量进行缩点,最后形成的图是有向无环图DAG.而拓扑序的值为DAG的长度,则加一,可是最后 ...

随机推荐

  1. [hdu 3949]线性基+高斯消元

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3949 一开始给做出来的线性基wa了很久,最后加了一步高斯消元就过了. 之所以可以这样做,证明如下. 首 ...

  2. Educational Codeforces Round 55 (Rated for Div. 2):C. Multi-Subject Competition

    C. Multi-Subject Competition 题目链接:https://codeforces.com/contest/1082/problem/C 题意: 给出n个信息,每个信息包含专业编 ...

  3. codeforces 1060 A

    https://codeforces.com/contest/1060/problem/A 题意:电话号码是以8开头的11位数,给你n 个数问最多可以有多少个电话号码 题解:min(8的个数,n/11 ...

  4. oralce的客户端sqlplus

    安装完oracle后,默认的客户端是sqlplus,还有一个公司常用的是PLSQLdeveloper 客户端软件,另外Navicat primie这个可以连接mysql.sqlserver.oracl ...

  5. java获取mysql数据库表、字段、字段类型、字段注释

    最近想要写一个根据数据库表结构生成实体.mapper接口.mapping映射文件.service类的简单代码生成工具,所以查阅了一些资料,怎样获取数据库的表.表中字段.字段类型.字段注释等信息. 最后 ...

  6. WebView使用--文章集锦

    对于android WebView加载不出Html5网页的解决方法 在android4.4中webview的使用相对于之前版本的一些区别 理解WebKit和Chromium: Android 4.4 ...

  7. 「6月雅礼集训 2017 Day2」A

    [题目大意] 给出一棵树,求有多少对点(u,v)满足其路径上不存在两个点a,b满足(a,b)=1 n<=10^5 [题解] 考虑找出所有不符合的点对,共有n*ln(n)对,他们要么是祖先-> ...

  8. codeforces739C - Skills &&金中市队儿童节常数赛

    http://codeforces.com/problemset/problem/739/C 先上链接 这道题 对于蒟蒻的我来说还是很有难度的 调了很久 对于我的代码 mx2是答案 mx1代表单调 m ...

  9. 【Luogu P3834】可持久化数组(可持久化线段树)

    题目链接 可持久化线段树模板题. 这里总结一下可持久化线段树. 可持久化数据结构就是能恢复历史状态的数据结构,比如可持久化\(Trie\),并查集,平衡树. 可持久化数组是最基础的,这里通过可持久化线 ...

  10. magento目录了解

    对magento目录的了解: