zoukankan      html  css  js  c++  java
  • [APIO2018]铁人两项

    题目

    题意:在一个无向图里面选三个点(s),(c),(f)

    需要能够从(s)出发,经过(c),到达(f)点,中间不能提前经过(f),且需要是一个简单路径

    Solution:

    简单路径当然就是园方树了,想想怎么统计答案
    yy一下可以发现,有一条路径(s)(f),中间能选的点就是路径上的圆点和
    因为在一个点双连通分量里面,一定有一个不重复的路径到出点去。
    想想怎么赋权,方点赋连接的圆点个数,圆点为-1,最后一条路径的权值和刚好就是去掉两个头的圆点个数,
    然后就是个简单的dp计数了

    code:

    #include<bits/stdc++.h>
    #define ll long long
    #define R register
    #define pb push_back
    template<class T>
    void rea(T &x)
    {
    	int f(0);char ch=getchar();x = 0;
    	while(!isdigit(ch)){f|=ch=='-';ch=getchar();}
    	while(isdigit(ch)){x=(x<<3)+(x<<1)+(ch^48);ch=getchar();}
    	x = f?-x:x;
    }
    using namespace std;
    const int N = 500005;
    vector<int> I[N], e[N<<1];
    ll ans;
    int n, m, node, num;
    int dfn[N], low[N], dfc, sta[N], tp;
    int val[N<<1], siz[N<<1];
    bool vis[N<<1];
    void tarjan(int x)
    {
    	dfn[x] = low[x] = ++dfc;
    	sta[++tp] = x;
    	for(R int i = 0; i < I[x].size(); ++i)
    	{
    		if(!dfn[I[x][i]])
    		{
    			tarjan(I[x][i]);
    			low[x] = min(low[x], low[I[x][i]]);
    			if(low[I[x][i]] == dfn[x])
    			{
    				val[++node] = 1;
    				for(R int o = 0; o != I[x][i]; --tp)
    				{
    					o = sta[tp];
    					e[node].pb(o);
    					e[o].pb(node);
    					++val[node];
    				}
    				e[x].pb(node), e[node].pb(x);
    			}
    		}
    		else low[x] = min(low[x], dfn[I[x][i]]);
    	}
    }
    void cal(int x, int fa)
    {
    	siz[x] = (x<=n);
    	//cout<<x<<" "<<val[x]<<endl;
    	for(R int i = 0; i < e[x].size(); ++i) if(e[x][i] != fa)
    	{
    		cal(e[x][i], x);
    		ans += 2ll * siz[x] * siz[e[x][i]] * val[x];
    		siz[x] += siz[e[x][i]];
    	}
    	ans += 2ll * siz[x] * (num-siz[x]) * val[x];
    }
    int main()
    {
    	rea(n), rea(m); node = n;
    	memset(val, -1, sizeof val);
    	int x, y;
    	for(R int i = 1; i <= m; ++i)
    		rea(x), rea(y), I[x].pb(y), I[y].pb(x);//读入用I!! 
    	for(R int i = 1; i <= n; ++i) 
    		if(!dfn[i]) num = dfc, tarjan(i), num = dfc-num, cal(i, 0);
    	printf("%lld
    ", ans);
        return 0;
    }
    
    
  • 相关阅读:
    Office 2010激活 NO KMS products detected问题
    强制换行/不换行 (兼容IE)
    el-radio再次点击取消选中
    几个 JavaScript 实用小技巧
    微信小程序-点击事件传递参数
    微信小程序-存取本地缓存
    微信小程序-路由方式
    element-ui 表格排序失效
    watch监听(数组或者对象)
    vue 引入 base64或者md5对密码进行加密
  • 原文地址:https://www.cnblogs.com/heanda/p/12482102.html
Copyright © 2011-2022 走看看