2023.1.12

A.Odd Divisor

https://codeforces.com/contest/1475/problem/A

Statement

给定一个正整数 \(n \ (2 \leq n \leq 10 ^ {14})\) ,判断其是否至少有一个大于 \(1\) 的奇因子。

Solution

\(n\) 为奇数,那 \(n\) 本身就是一个符合条件的奇因子。

考虑偶数的情况:偶数可以由两个偶数相乘得到,也可以通过一奇一偶相乘得到。能被表示成 \(2\)\(x\) 次幂的 \(n\) 只会属于第一种情况,其他则可以表示为第二种。判断 \(n\) 是否为 \(2\)\(x\) 次幂有两种方法:(1) 看 \(n \& (n - 1)\) 是否为 \(0\)。(2)不断除 \(2\) 直至最后变成 \(1\),途中不出现奇质因子。

B.Advertising Agency

https://codeforces.com/contest/1475/problem/E

Statement

给定 \(n\) 个数,从中选 \(k\) 个,要求 \(\sum _{i = 1} ^ {k} a_i\) 最大,求方案数。

Solution

由于 \(sum\) 最大且固定,那么所有方案本质上都是相同的。(反证法:将 \(a_1, a_2, \cdots a_n\) 按降序排序,若存在 \(a_j (k < j \leq n)\) 使得方案更优,说明 \(a_j > a_k\), 那其必定排在前 \(k\) 大,与已知条件矛盾)。

故答案取决于前 \(k\) 大的数中,最小的那个数 \(x\) 的个数。令 \(cnt_x\) 表示 \(n\) 个数中 \(x\) 出现的次数,\(cnt_y\) 表示前 \(k\) 大的数中严格大于 \(x\) 的个数,易得答案为 \(\binom{cnt_x}{k - cnt_y}\)

Code

#include <bits/stdc++.h>
const int lim = 1000;
const int mod = 1e9 + 7;
using namespace std;

int T, n, k, a[lim + 5], c[lim + 5][lim + 5];

void init() {
    c[0][0] = 1;
    for (int i = 1; i <= lim; i++) {
        c[i][0] = 1;
        for (int j = 1; j <= i; j++) c[i][j] = (c[i - 1][j] + c[i - 1][j - 1]) % mod;
    }
}

int main() {
    ios::sync_with_stdio(false);
	cin.tie(0); cout.tie(0);
	freopen("in","r", stdin);
    cin >> T;
    init();
    while (T--) {
        cin >> n >> k;
        for (int i = 1; i <= n; i++) cin >> a[i];
        sort(a + 1, a + n + 1, greater<int>());
        int cnt1 = 0, cnt2 = 0, minn = a[k];
        for (int i = 1; i <= n; i++) {
            cnt1 += (a[i] == minn);
            cnt2 += (a[i] > minn);
        }
        cout << c[cnt1][k - cnt2] << "\n";
    }
	return 0;
}

C.Perfect Keyboard

https://codeforces.com/contest/1303/problem/C

Statement

给定一个仅包含小写字母且相邻字母不一样的字符串 \(s\),将 abcdefg...xyz 重排,重排后的字符串为 \(t\) ,询问是否能使得 \(s\) 中相邻的字母在 \(t\) 中也全部相邻,若可以则输出方案。

Solution

\(s\) 中相邻的字母进行连边(始终保证由字典序小的连接字典序大的),不难发现若出现度数大于 \(2\) 的节点或有环出现时,则答案不存在。否则可以从度数为 \(1\) 的点出发,对树进行深度优先遍历并记录路径,得到的答案序列则是符合条件的。

Code

#include <bits/stdc++.h>
const int N = 50;
using namespace std;

string s;
int T, deg[N];
bool vis[N], g[N][N];

void dfs(int x, string &ans) {
    vis[x] = 1;
    ans += (char)('a' + x - 1);
    for (int i = 1; i <= 26; i++) {
        if ((!g[x][i]) || (vis[i]))
            continue;
        dfs(i, ans);
    }
}

int main() {
    ios::sync_with_stdio(false);
	cin.tie(0); cout.tie(0);
	freopen("in","r", stdin);
    cin >> T;
    while (T--) {
        cin >> s;
        if ((int)s.length() == 1) {
            cout << "YES" << endl;
            for (char c = 'a'; c <= 'z'; c++) cout << c;
            cout << endl;
            continue;
        }
        for (int i = 0; i < (int)s.length() - 1; i++)
            g[s[i] - 'a' + 1][s[i + 1] - 'a' + 1] = g[s[i + 1] - 'a' + 1][s[i] - 'a' + 1] = 1;
        for (int i = 1; i <= 26; i++) {
            for (int j = i + 1; j <= 26; j++) {
                if (g[i][j]) {
                    deg[i]++;
                    deg[j]++;
                }
            }
        }
        bool flag = 0;
        for (int i = 1; i <= 26; i++)
            flag |= (deg[i] > 2);
        string ans = "";
        bool sign = 0;
        for (int i = 1; i <= 26; i++) {
            if (deg[i] == 1 && (!vis[i])) {
                sign = true;
                dfs(i, ans);
            }
        }
        if (flag || (!sign))
            cout << "NO" << endl;
        else {
            cout << "YES" << endl;
            for (char c = 'a'; c <= 'z'; c++) {
                if (!vis[c - 'a' + 1])
                    ans += c;
            }
            cout << ans << endl;
        }
        memset(g, 0, sizeof(g));
        memset(deg, 0, sizeof(deg));
        memset(vis, 0, sizeof(vis));
    }
	return 0;
}
posted @ 2023-01-14 00:23  BeyondLimits  阅读(35)  评论(0编辑  收藏  举报