码迷,mamicode.com
首页 > 其他好文 > 详细

20160818训练记录

时间:2016-08-18 21:25:29      阅读:193      评论:0      收藏:0      [点我收藏+]

标签:

T1

技术分享

“开心题”

每个点如果不是根节点,子树个数等于度数减一。然后随便算一算

T2

技术分享

数据范围

技术分享

 

首先对于两个长度相等的偶数串。把他两两分组

1 2 3 4
5 6 7 8

考虑一个和8相等的(包括交换内部后)的块

先把他调到头部然后调到尾部。然后删除这一块

然后发现这样的可以对每个串求出字典序最小的等价表达。。其实随便计算一下就解决了

#include<cstdio>
#include<string.h>
#include<iostream>
#include<algorithm>
using namespace std;
int n,sc[55],len[55],ls[55][66],_ls[55];
char mz[55][66];
bool same(pair<char,char>p[],int b){
    pair<char,char >pb[55];int _p=0;
    for(int j=1;j<=len[b]/2;j++){
        if(mz[b][(j<<1)-1]>mz[b][j<<1])
            pb[++_p]=make_pair(mz[b][(j<<1)],mz[b][(j<<1)-1]);
        else pb[++_p]=make_pair(mz[b][(j<<1)-1],mz[b][(j<<1)]);
    }
    sort(pb+1,pb+_p+1);
    for(int i=1;i<=_p;i++){
        if(p[i]!=pb[i]){
            return false;
        }    
    }return true;
}
int main(){
    freopen("list.in","r",stdin);
    freopen("list.out","w",stdout);
    scanf("%d",&n);
    for(int i=1;i<=n;i++){
        scanf("%s",mz[i]+1);len[i]=strlen(mz[i]+1);
    }
    for(int i=1;i<=n;i++){
        if(!sc[i]){
            for(int j=1;j<=len[i]/2;j++){
                if(mz[i][(j<<1)-1]>mz[i][j<<1])
                    p[++_p]=make_pair(mz[i][(j<<1)],mz[i][(j<<1)-1]);
                else p[++_p]=make_pair(mz[i][(j<<1)-1],mz[i][(j<<1)]);
            } 
            sort(p+1,p+_p+1);
            for(int j=1;j<=n;j++){
                if(len[i]!=len[j]||(len[i]&1)!=(len[j]&1))continue;
                if((len[i]&1)&&(len[j]&1)){
                    if(mz[i][len[i]]!=mz[i][len[j]])continue;
                }
                if(same(p,j)){
                    sc[i]=sc[j]=true; ls[i][++_ls[i]]=j;
                }
        
            }
        }
    }
    int ans=0;

    for(int i=1;i<=n;i++){
        ans+=(_ls[i]&1);
    }
    cout<<ans;
    return 0;
}

T3

对于一个图。给定固定点对,求删边一条边后点对的最短路

 

20%暴力

//segment tree beats!
//I can program a segment but tree,not for this problem
#include<queue>
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
typedef long long ll;

ll inf =1000000000ll*1000000000ll;
struct edge{
    int to,next,v;
}e[400005];
int n,m,q,_s,_t,du,dv,last[210000],cnt,inq[200001];
ll dis[200001];
void insert_directed(int a,int b,int c){e[++cnt]=(edge){b,last[a],c};last[a]=cnt;}
void link(int a,int b,int c){insert_directed(a,b,c);insert_directed(b,a,c);}
int gg;
ll spfa(int s,int t){
    queue<int>q;
    for(int i=1;i<=n;i++)dis[i]=1000000000ll*1000000000ll,inq[i]=false;
    q.push(s);dis[s]=0;inq[s]=true;
    while(!q.empty()){
        int c=q.front();q.pop();inq[c]=false;
        for(int i=last[c];i;i=e[i].next){    
            int x=e[i].to;
            if(c==du&&x==dv)continue;
            if(c==dv&&x==du)continue;
                
                if(dis[c]+e[i].v<dis[x]){
                    dis[x]=dis[c]+e[i].v;
                    if(!inq[x]){
                        q.push(x);inq[x]=true;
                    }
                }
            
        }
    }
    return dis[t];
}
namespace luangao{
    int pre[200001];
    ll spfa(int s,int t){
        queue<int>q;
        for(int i=1;i<=n;i++)dis[i]=1000000000ll*1000000000ll,inq[i]=false;
        q.push(s);dis[s]=0;inq[s]=true;
        while(!q.empty()){
            int c=q.front();q.pop();inq[c]=false;
            for(int i=last[c];i;i=e[i].next){    
                int x=e[i].to;
                if(c==du&&x==dv)continue;
                if(c==dv&&x==du)continue;
                    
                    if(dis[c]+e[i].v<dis[x]){
                        pre[x]=c;
                        dis[x]=dis[c]+e[i].v;
                        if(!inq[x]){
                            q.push(x);inq[x]=true;
                        }
                    }
                
            }
        }
        return dis[t];
    }
    ll hs(int a,int b){
        return a*200001+b;
    }
    ll H[200001];int _H=0;
    bool find(ll x){
        int l=1,r=_H,mid;
        while(l<=r){
            //if(x==hs(1,2))cout<<l <<‘ ‘<<r<<endl;
            mid=(l+r)/2;
            if(x==H[mid])return true;
            if(x>H[mid])l=mid+1;
            else r=mid-1;
        }
        return x==H[mid];
    }
    int main(){
        ll sssp=spfa(_s,_t);
        int t=_t;
        while(pre[t]){
            H[++_H]=hs(pre[t],t);
            H[++_H]=hs(t,pre[t]);
            t=pre[t];
        }
        sort(H+1,H+_H+1);
        //for(int i=1;i<=_H;i++)cout<<H[i]<<endl; 
        //cout<<hs(1,2)<<endl;
        for(int i=1;i<=q;i++){
            scanf("%d %d",&du,&dv);
            ll x=hs(du,dv);
            //cout<<find(x)<<‘:‘;
            ll ans;
            if(find(x))ans=spfa(_s,_t);
            else ans=sssp;
            if(ans>=inf)puts("Infinity");
            else printf("%I64d\n",ans);
        }
        return 0;
    }
}
int main(){
    freopen("dream.in","r",stdin);
    freopen("dream.out","w",stdout);
    scanf("%d %d",&n,&m);
    for(int i=1;i<=m;i++){
        int u,v,va;
        scanf("%d %d %d",&u,&v,&va);link(u,v,va);
    }
    scanf("%d %d %d" ,&_s,&_t,&q);
    if(n<=4000&&m<=4000)
        for(int i=1;i<=q;i++){
            scanf("%d %d",&du,&dv);
            ll ans=spfa(_s,_t);
            if(ans>=inf)puts("Infinity");
            else printf("%I64d\n",ans);
        }
    else luangao::main();
}

正解

 

两种做法

一种线段树的。http://tonyfang.is-programmer.com/posts/205232.html

OrzFang%%%

第二种堆的做法==

考虑一个最短路图的情况。最短路图是每一条边都在最短路上的边

那么影响答案的肯定是割边被删除

那么我们考虑s-t中的联通块依序处理。如果是这之间的割边答案会变成这个和后面所有的联通块之间连边最小值与两端到s、t的最短路和

 

扫描一下联通块

这样每条边都会进堆出堆一次 复杂度得到保证

20160818训练记录

标签:

原文地址:http://www.cnblogs.com/chouti/p/5785190.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!