• Codeforces 467D Fedor and Essay bfs


    题目链接:

    题意:

    给定n个单词。

    以下有m个替换方式。左边的单词能变成右边的单词。

    替换随意次后使得最后字母r个数最少,在r最少的情况下单词总长度最短

    输出字母r的个数和单词长度。

    思路:

    我们觉得一个单词有2个參数。则m个替换规则能够当成m个点的有向图。

    则某些单词的替换终点会确定,所以反向建图bfs一下。

    为了防止某些点被重复更新,所以把每一个点的权值都放到栈里排个序然后bfs。



    #include<stdio.h>
    #include<string.h>
    #include<iostream>
    #include<algorithm>
    #include<queue>
    #include<vector>
    #include<map>
    #include<string>
    using namespace std;
    
    typedef int ll;
    const int inf = 1000000;
    #define N 300005
    int n, m;
    void go(string &x){
        int len = x.length();
        for(int i = 0; i < len; i++)
            if('A' <= x[i] && x[i] <= 'Z')
                x[i] = x[i]-'A'+'a';
    }
    string s[N], a[N], b[N];
    int S[N], A[N], B[N], LEN[N], NUM[N];
    map<string, int> mp;
    struct Edge{
        int to, nex;
    }edge[N*10];
    int head[N], edgenum;
    void init(){memset(head, -1, sizeof head); edgenum = 0;}
    void add(int u, int v){
        Edge E = {v, head[u]};
        edge[edgenum] = E;
        head[u] = edgenum++;
    }
    struct node{
        int x, y, point;
        node (int X = 0, int Y = 0, int P = 0):x(X), y(Y), point(P){}
        bool operator<(const node&D){
            if(D.x!=x)return D.x > x;
            return D.y>y;
        }
    }st[N], dis[N];
    int top;
    bool cmp(node X, node Y){
        if(X.x != Y.x) return X.x < Y.x;
        if(X.y != Y.y) return X.y < Y.y;
        return X.point < Y.point;
    }
    int tot;
    void BFS(int x){
        queue<int>q;
        q.push(x);
        while(!q.empty()){
            int u = q.front(); q.pop();
            for(int i = head[u]; ~i; i = edge[i].nex){
                int v = edge[i].to;
                if(dis[u] < dis[v])
                {
                    dis[v] = dis[u];
                    q.push(v);
                }
            }
        }
    }
    void input(){
        mp.clear();
        init();
        top = tot = 0;
        for(int i = 1; i <= n; i++)
        {
            cin>>s[i]; go(s[i]);
            if(mp.count(s[i])==0)
                {
                    S[i] = mp[s[i]] = ++tot;
                int len = s[i].length(), num = 0;
                for(int j = 0; j < len; j++) num += s[i][j]=='r';
                st[top++] = node(num, len, tot);
                LEN[tot] = len; NUM[tot] = num;
            }
            else S[i] = mp[s[i]];
        }
        scanf("%d", &m);
        for(int i = 1; i <= m; i++)
        {
            cin>>a[i]>>b[i]; go(a[i]); go(b[i]);
            if(mp.count(a[i])==0)
            {
                A[i] = mp[a[i]] = ++tot;
                int len = a[i].length(), num = 0;
                for(int j = 0; j < len; j++) num += a[i][j]=='r';
                st[top++] = node(num, len, tot);
                LEN[tot] = len; NUM[tot] = num;
            }
            else A[i] = mp[a[i]];
            if(mp.count(b[i])==0)
            {
                B[i] = mp[b[i]] = ++tot;
                int len = b[i].length(), num = 0;
                for(int j = 0; j < len; j++) num += b[i][j]=='r';
                st[top++] = node(num, len, tot);
                LEN[tot] = len; NUM[tot] = num;
            }
            else B[i] = mp[b[i]];
            add(B[i], A[i]);
        }
    }
    int main(){
        while(~scanf("%d", &n)){
            input();
            for(int i = 1; i <= tot; i++)dis[i] = node(NUM[i], LEN[i],0);
            sort(st, st+top, cmp);
            for(int i = 0; i < top; i++)
                BFS(st[i].point);
            long long ans1 = 0, ans2 = 0;
            for(int i = 1; i <= n; i++)     
                ans1 += dis[S[i]].x, ans2 += dis[S[i]].y;
            
            printf("%I64d %I64d
    ", ans1, ans2);
        }
        return 0;
    }


  • 相关阅读:
    HDU3584 Cube
    SPOJ1029 Matrix Summation
    SPOJ227 Ordering the Soldiers
    ScriptX,smsx打印控件安装及无法打印的问题
    如何做站
    右下角浮动
    简单的显示关闭
    总结
    ubuntu 上网设置
    固定高度后能自动伸缩
  • 原文地址:https://www.cnblogs.com/liguangsunls/p/7261456.html
Copyright © 2020-2023  润新知