zoukankan      html  css  js  c++  java
  • 2016.11.14测试 长乐一中2014NOIP复赛模拟题 第一题。

    1.正确答案

    【题目描述】

    小H与小Y刚刚参加完UOIP外卡组的初赛,就迫不及待的跑出考场对答案。

    “吔,我的答案和你都不一样!”,小Y说道,”我们去找神犇们问答案吧”。

    外卡组试卷中共有m道判断题,小H与小Y一共从其他n个神犇那问了答案。之后又从小G那里得知,这n个神犇中有p个考了满分,q个考了零分,其他神犇不为满分或零分。这可让小Y与小H犯了难。你能帮助他们还原出标准答案吗?如有多解则输出字典序最小的那个。无解输出-1。

    【输入格式】

    第一行四个整数n, m, p, q,意义如上描述。

    接下来n行,每一行m个字符’N’或’Y’,表示这题这个神犇的答案。

    【输出格式】

    仅一行,一个长度为m的字符串或是-1。

    【样例输入】

    2 2 2 0

    YY

    YY

    【样例输出】

    YY

    【数据范围】

    30% : n <= 100.

    60% : n <= 5000 , m <= 100.

    100% : 1 <= n <= 30000 , 1 <= m <= 500.  0 <= p , q 且 p + q <= n. 

    题解是这样说的:

    30%:  O(n ^ 2 * m)暴力判断。

    100%: 很显然答案的可能性最多只有n种,所以我们将所有人的答案按字典序排序后枚举     将每个人的答案作为正确答案来进行判断。由于是判断题,若当前人的答案为正确答      案则零分者的答案也就确定了,那么只需统计出这两种答案的人数判断是否满足题意      即可。这一步使用字符串哈希即可解决。

    代码如下 :

    #include <algorithm>
    #include <iostream>
    #include <cstring>
    #include <cstdio>
    #include <cmath>
    using namespace std;
    
    const int N = 3e4 + 2, M = 5e2 + 2, sed = 31, SED = 131, mod = 70177, MOD = 92311;
    int n, m, p, q, ans, hash[N], HASH[N];
    int top, info[mod], nxt[N * 2], fet[N * 2], cnt[N * 2];
    struct node {
        char s[M];
        inline bool operator < (const node &b) const {
            return strcmp(s, b.s) < 0;
        }
    } a[N];
    
    inline void Insert(const int &x, const int &y) {
        for (int k = info[x]; k; k = nxt[k])
            if (fet[k] == y) {
                ++cnt[k]; return ;
            }
        nxt[++top] = info[x]; info[x] = top;
        fet[top] = y; cnt[top] = 1;
        return ;
    }
    
    inline int Query(const int &x, const int &y) {
        for (int k = info[x]; k; k = nxt[k])
            if (fet[k] == y) return cnt[k];
        return 0;
    }
    
    inline void Solve1() {
        int tmp, TMP; ans = -1;
        for (int i = 0; i < n; ++i) {
            tmp = TMP = 0;
            for (int j = 0; j < m; ++j) {
                tmp = (tmp * sed + (a[i].s[j] == 'N')) % mod;
                TMP = (TMP * SED + (a[i].s[j] == 'N')) % MOD;
            }
            hash[i] = tmp, HASH[i] = TMP;
            Insert(tmp, TMP);
        }
        for (int i = 0; i < n; ++i)
            if (Query(hash[i], HASH[i]) == p) {
                tmp = TMP = 0;
                for (int j = 0; j < m; ++j) {
                    tmp = (tmp * sed + (a[i].s[j] == 'Y')) % mod;
                    TMP = (TMP * SED + (a[i].s[j] == 'Y')) % MOD;
                }
                if (Query(tmp, TMP) == q) {
                    ans = i; break;
                }
            }
        if (ans != -1) printf("%s
    ", a[ans].s);
        else     puts("-1");
        return ;
    }
    
    char cur[M];
    inline void Solve2() {
        int tmp, TMP; ans = -1;
        for (int i = 0; i < n; ++i) {
            tmp = TMP = 0;
            for (int j = 0; j < m; ++j) {
                tmp = (tmp * sed + (a[i].s[j] == 'N')) % mod;
                TMP = (TMP * SED + (a[i].s[j] == 'N')) % MOD;
            }
            hash[i] = tmp, HASH[i] = TMP;
            Insert(tmp, TMP);
        }
        for (int i = n - 1; i >= 0; --i)
            if (Query(hash[i], HASH[i]) == q) {
                tmp = TMP = 0;
                for (int j = 0; j < m; ++j) {
                    tmp = (tmp * sed + (a[i].s[j] == 'Y')) % mod;
                    TMP = (TMP * SED + (a[i].s[j] == 'Y')) % MOD;
                }
                if (Query(tmp, TMP) == p) {
                    ans = i; break;
                }
            }
        if (ans != -1) {
            for (int i = 0; i < m; ++i)
                cur[i] = a[ans].s[i] == 'N' ? 'Y' : 'N';
            printf("%s
    ", cur);
        }
        else     puts("-1");
        return ;
    }
    
    void Solve3() {
        int tmp, TMP;
        for (int i = 0; i < n; ++i) {
            tmp = TMP = 0;
            for (int j = 0; j < m; ++j) {
                tmp = (tmp * sed + (a[i].s[j] == 'N')) % mod;
                TMP = (TMP * SED + (a[i].s[j] == 'N')) % MOD;
            }
            Insert(tmp, TMP);
            tmp = TMP = 0;
            for (int j = 0; j < m; ++j) {
                tmp = (tmp * sed + (a[i].s[j] == 'Y')) % mod;
                TMP = (TMP * SED + (a[i].s[j] == 'Y')) % MOD;
            }
            Insert(tmp, TMP);
        }
        bool flag = true;
        for (int i = 0; i < m; ++i) cur[i] = 'N';
        do {
            tmp = TMP = 0;
            for (int j = 0; j < m; ++j) {
                tmp = (tmp * sed + (cur[j] == 'N')) % mod;
                TMP = (TMP * SED + (cur[j] == 'N')) % MOD;
            }
            if (Query(tmp, TMP) == 0) {
                flag = true; break;
            }
            flag = false;
            for (int j = m - 1; j >= 0; --j)
                if (cur[j] == 'Y') cur[j] = 'N';
                else {
                    cur[j] = 'Y'; flag = true; break;
                }
        } while (flag);
        if (flag) printf("%s
    ", cur);
        else     puts("-1");
        return ;
    }
    
    int main() {
        freopen("answer.in", "r", stdin);
        freopen("answer.out", "w", stdout);
        scanf("%d%d%d%d", &n, &m, &p, &q);
        for (int i = 0; i < n; ++i) scanf("%s", a[i].s);
        sort(a, a + n);
        if (p) Solve1();
        else if (q) Solve2();
        else     Solve3();
        fclose(stdin); fclose(stdout);
        return 0;
    }
    View Code
    我们都在命运之湖上荡舟划桨,波浪起伏着而我们无法逃脱孤航。但是假使我们迷失了方向,波浪将指引我们穿越另一天的曙光。
  • 相关阅读:
    FastDFS迁移步骤
    Centos7 单节点安装 FastDFS + FastDHT服务
    CentOS 7.0 上安装和配置 VNC 服务器
    Ubuntu 18.04 安装 Xfce桌面和VNC的方法
    Ubuntu 16.04设置root用户登录图形界面
    Windows批量查找文件
    WIN10打开网络共享文件夹提示0x80004005怎么解决?(转载)
    ack 工具
    win7/win10 未分配磁盘怎样创建扩展分区 也就是逻辑分区(转截)
    [bzoj4842][bzoj1283][Neerc2016]Delight for a Cat/序列_线性规划_费用流
  • 原文地址:https://www.cnblogs.com/ruojisun/p/6063168.html
Copyright © 2011-2022 走看看