汇总牛客线上赛题目,为了统一我会将牛客寒假暑假训练题更新在这里.✌️
状态: 牛客周赛111,史上最有意思的结论场
牛客周赛Round-93
F
一个线性状态 DP
这个题目的思路比较简单,但是其实实现是有一点考验码量的,那个奇偶性的判断。就是一个线性DP
这个位置与只与上一个位置有关系,所以自然可以想到动态规划,但是这个题目的码量有一点要求,整体思维难度不大。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 #include <bits/stdc++.h> using namespace std;#define int long long const int mod = 1e9 + 7 ;signed main () { int n; cin >> n; vector<vector<int >> a (n + 1 , vector <int >(n + 1 , -1 )); for (int i = 1 ; i <= n; i++){ for (int j = 1 ; j <= i; j++){ cin >> a[i][j]; } } vector<vector<vector<int >>> dp (2 , vector<vector<int >>(n + 3 , vector <int >(n + 3 , 0 ))); if (n % 2 == 0 ){ int t1 = n / 2 , t2 = n / 2 + 1 ; for (int i = 1 ; i <= t1; i++){ if (a[t1][i] == a[t2][i]) dp[0 ][i][i]++; if (a[t1][i] == a[t2][i + 1 ]) dp[0 ][i][i + 1 ]++; } int temp = 1 ; for (int d = 1 ; d < n / 2 ; d++){ for (int i = 1 ; i <= n; i ++){ for (int j = 1 ; j <= n; j++){ dp[d & 1 ][i][j] = 0 ; } } temp &= 1 ; for (int i = 1 ; i <= n; i++){ for (int j = 1 ; j <= n; j++){ if (a[t1 - d][i] == a[t2 + d][j]){ dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i][j]+dp[d & 1 ][i][j]) % mod; dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i + 1 ][j] + dp[d & 1 ][i][j])% mod; dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i][j - 1 ] + dp[d & 1 ][i][j])% mod; dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i + 1 ][j - 1 ] + dp[d & 1 ][i][j])% mod; } } } } int ans = 0 ; for (int i = 1 ; i <= n; i++){ ans = (ans + dp[(n + 1 ) / 2 - 1 & 1 ][1 ][i]) % mod; } cout << ans << endl; }else { int t1 = (n / 2 ) + 1 ; for (int i = 1 ; i <= t1; i++){ dp[0 ][i][i] = 1 ; } int temp = 1 ; for (int d = 1 ; d <= n / 2 ; d++){ for (int i = 1 ; i <= n; i++){ for (int j = 1 ; j <= n; j++){ dp[d & 1 ][i][j] = 0 ; } } temp &= 1 ; for (int i = 1 ; i <= n; i++){ for (int j = 1 ; j <= n; j++){ if (a[t1-d][i] == a[t1 + d][j]){ dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i][j]+dp[d & 1 ][i][j]) % mod; dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i + 1 ][j] + dp[d & 1 ][i][j])% mod; dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i][j - 1 ] + dp[d & 1 ][i][j])% mod; dp[d & 1 ][i][j] = (dp[(d - 1 ) & 1 ][i + 1 ][j - 1 ] + dp[d & 1 ][i][j])% mod; } } } } int ans = 0 ; for (int i = 1 ; i <= n; i++){ ans = (ans + dp[(n + 1 )/2 - 1 & 1 ][1 ][i]) % mod; } cout << ans << endl; } }
牛客周赛Round-94
这一场我感觉偏推理场吧,主要被前面的题目唬到了,然后后面做起来有点害怕,看到了那个类似 成都赛场上面没做出来的一个签到题目,感觉心里很慌。
这个好像不是一个拓扑排序的题目,是一个构造题!
最讨厌看到的显然成立
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 #include <bits/stdc++.h> using namespace std;#define int long long #define fr first #define sc second #define endl '\n' using PII = pair<int ,int >;void slove () { string s; cin >> s; int n = s.size (); bool ex = true ; for (int i = 0 ; i < n; i++){ if (s[i] == '-' ){ ex = false ; break ; } } if (ex){ cout << "No" << endl; return ; } int d[4 ] = {0 , n - 1 , n - 2 , n - 3 }; for (int i = 0 ; i < 4 ; i++){ if (s[d[i]] != '>' ){ cout << "No" << endl; return ; } } vector<pair<int , int >> ans; int i; for ( i = n - 3 ; i > 1 ; i--){ if (s[i] == '>' ){ ans.push_back ({0 , i + 3 }); }else break ; } for (int j = 1 ; j < i; j++){ if (s[j] == '>' ){ ans.push_back ({j, i + 3 - j + 1 }); } } cout << "Yes" << " " << ans.size () << endl; for (auto [x, y] : ans){ cout << x + 1 << " " << y << endl; } } signed main () { int t; cin >> t; while (t--) slove (); return 0 ; }
知识点:对于一个数字的二进制形式 , 2 相当于在二进制后面新增0, 2 相当于删除二进制的最后一位
有一个坑 :如果当 n 为 1的时候,是不要进行特判里面的,因为如果为 1 的时候,他的2 的倍数一定是早就出现过的
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 #include <bits/stdc++.h> using namespace std;#define int long long #define fr first #define sc second #define endl '\n' using PII = pair<int ,int >;signed main () { int t; cin >> t; while (t--){ int n, k; cin >> n >> k; int ans = k + 1 ; for (int i = 1 ; i <= k; i++){ if (n % 2 != 0 && n != 1 ){ ans += (k - i); } ans ++; n /= 2 ; if (n == 0 ){ break ; } } cout << ans << endl; } return 0 ; }
有一个结论
结论
定义小球集合 的函数 ,表示将小球集合 分为若干组,满足以下所有条件 的最少分组个数:
思路: 这个题目最重要的一个点就是上面的结论,上面的结论是针对于一个固定的数组的时候的结果,显然可以见到就是 一个序列的分组个数就跟序列的绝对众数和数列的大小有关,所以有一个很新奇的思路:首先我们将所有情况的答案都算作 (sum/2) 的情况处理,然后我们再去枚举 存在众数的情况,这样就可以减少复杂的思考,然后我们只需要知道什么情况才是 的情况,根据 结论的式子可以得到,只有当当前数字作为这个序列的绝对众数( 的时候才是这个结果,所以我们可以枚举每一个数字作为绝对众数
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 #include <bits/stdc++.h> using namespace std;#define int long long const int mod = 998244353 ;const int N = 5e3 + 20 ;int power (int x, int n) { int ans = 1 ; while (n){ if (n & 1 ){ ans = ans * x % mod; } x = x * x % mod; n >>= 1 ; } return ans; } vector<int > fac (N + 1 ) ; vector<int > inv (N+ 1 ) ; void init () { fac[0 ] = 1 ; for (int i = 1 ; i <= N; i ++){ fac[i] = fac[i - 1 ] * i % mod; } inv[N] = power (fac[N], mod - 2 ); for (int i = N - 1 ; i >= 0 ; i --){ inv[i] = inv[i + 1 ] * (i + 1 ) % mod; } } int C (int n, int m) { if (m > n){ return 0 ; } return fac[n] * inv[m] % mod * inv[n - m] % mod; } void slove () { int n; cin >> n; map<int ,int > mp1; for (int i = 0 , t; i < n; i++){ cin >> t; mp1[t]++; } int n1 = mp1. size (); int ans = 0 ; for (int i = 1 ; i <= n; i++){ ans = (ans + ((i + 1 ) / 2 ) * C (n, i) % mod) % mod; } vector<int > a (1 ) ; for (auto [x, y] : mp1){ a.push_back (y); } for (int i = 1 ; i <= n1; i++){ for (int j = 1 ; j <= a[i]; j++){ for (int k = 0 ; k < j; k++){ int t1 = C (a[i], j), t2 = C (n - a[i], k), len = (j + k); int temp1 = (len + 1 ) / 2 * t1 % mod * t2 % mod; int add = (j * t1 % mod * t2 % mod); ans = (ans - temp1 + mod) % mod; ans = (ans + add) % mod; } } } cout << ans << endl; } signed main () { init (); int t; cin >> t; while (t--) slove (); }
牛客周赛Round-103
绝世好题,这个题目运用了很多算法,字典树,哈希,也提供的一个很好的思路去快速解决 前缀匹配的问题
收获: 已知N个字符串 ,查询给定一个字符串 ,我现在需要查询 与这N个字符串的公共前缀和,
O(log(Q.size()))的解决办法思路: 先预处理S的字典树(前缀树),然后用hash计数每一个前缀的出现个数(map<string,int> 的时间复杂度远远高于 map<int,int>),然后去二分这个 Q_s字符串的前缀,去找最长的匹配长度即可
易错点: 针对于 字典树 来说,字典树递归应该是 (u -> tree[u][k])
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 #include <bits/stdc++.h> using namespace std;#define int long long const int base = 39127 ;const int N = 1e5 + 10 ;vector<int > power (N,1 ) ;void slove () { int n, m; cin >> n >> m; string s; cin >> s; vector<int > hash1 (n, 0 ) ; hash1[0 ] = (s[0 ] - 'a' + 1 ); for (int i = 1 ; i < n; i++){ hash1[i] = hash1[i - 1 ] * base + (s[i] - 'a' + 1 ); } auto get = [&](int l, int r) -> int { if (l > 0 ){ l = hash1[l - 1 ] * power[r - l + 1 ]; } r = hash1[r]; return r - l; }; vector<string> t (m) ; int mx = 0 , sum = 0 ; for (int i = 0 ; i < m; i++){ cin >> t[i]; int len = t[i].size () + 1 ; sum += len; } vector tree (sum + 1 , vector<int >(27 , 0 )) ; vector<int > pa (sum + 1 , 0 ) ; int cnt = 1 ; auto insert = [&](string s1) -> void { int cur = 1 ; pa[cur]++; for (int i = 0 ; i < s1. size (); i++){ int path = s1[i] - 'a' ; if (tree[cur][path] == 0 ){ tree[cur][path] = ++cnt; } cur = tree[cur][path]; pa[cur]++; } }; for (int i = 0 ; i < m; i++) insert (t[i]); map<int , int > mp1; auto dfs = [&](auto ff, int layer, int h, int now) -> void { for (int k = 0 ; k < 27 ; k++){ if (tree[layer][k] == 0 ) continue ; int h1 = h * base + (k + 1 ); int cost = pa[tree[layer][k]] + now; mp1[h1] = cost; ff (ff, tree[layer][k], h1, cost); } }; dfs (dfs, 1 , 0 , 0 ); int ans = 0 ; for (int i = 0 ; i < n; i++){ int l = i, r = n - 1 ; while (l <= r){ int mid = (l + r + 1 ) >> 1 ; int t1 = get (i, mid); if (mp1. count (t1)){ l = mid + 1 ; ans = max (ans, mp1[t1]); }else r = mid - 1 ; } } cout << ans << endl; } signed main () { ios::sync_with_stdio (false ); cin.tie (0 ); for (int i = 1 ; i < power.size (); i++) power[i] = power[i - 1 ] * base; int t; cin >> t; while (t--) slove (); }
牛客周赛Round-104
[小红的树不动点]
针对于这个问题,首先我们应该观察贡献是怎么来的,比如 i 的贡献是要基于 i-1 的贡献之上的,只有包括 i-1 的子树再包括 i,i才会有贡献,我们可以应用 dfn 序来判断当前节点是不是在 K 这个节点的子树下面
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 #include <bits/stdc++.h> using namespace std;#define int long long signed main () { int n; cin >> n; vector dj (n + 1 , vector<int >()) ; for (int i = 0 ; i < n - 1 ; i++){ int u, v; cin >> u >> v; dj[u].push_back (v); dj[v].push_back (u); } vector<int > dfn (n + 1 , 0 ) ; vector<int > size (n + 1 , 1 ) ; vector<int > deep (n + 1 , 0 ) ; vector<int > fa (n + 1 , 0 ) ; int idx = 1 ; auto dfs = [&](auto ff, int u, int f) -> void { dfn[u] = idx++; deep[u] = deep[f] + 1 ; fa[u] = f; for (auto x : dj[u]){ if (x == f) continue ; ff (ff, x, u); size[u] += size[x]; } }; dfs (dfs, n, 0 ); int ans = 0 ; int lca = 1 ; for (int i = 1 ; i <= n; i++){ while (lca > 0 && !(dfn[lca] <= dfn[i] && dfn[lca] + size[lca] - 1 >= dfn[i])){ lca = fa[lca]; } ans += deep[lca]; } cout << ans << endl; }
牛客周赛Round-105
反思:这道题目经验不足,思考方向不对,怎么会不往二进制的角度去想问题呢?按理来说,题目中遇见或,异或,与 等问题,我们都应该往二进制的角度去想问题,我一直从如何三个节点的路径上想问题是完全错误的
三个节点的异或值可以从节点的每一位去观察,因为你三个节点在某一位上的情况很少,只有 001,011,010…
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 #include <bits/stdc++.h> using namespace std;#define int long long #define fr first #define sc second #define endl '\n' using PII = pair<int ,int >;const int mod = 998244353 ;signed main () { ios::sync_with_stdio (false ); cin.tie (0 ); int n, m; cin >> n >> m; vector<int > v (n + 1 , 0 ) ; for (int i = 1 ; i <= n; i++) cin >> v[i]; vector<vector<int >> a (n + 1 ); for (int i = 0 ; i < m; i++){ int u, v; cin >> u >> v; a[u].push_back (v); a[v].push_back (u); } int ans = 0 ; for (int i = 0 ; i <= 32 ; i++){ for (int u = 1 ; u <= n; u++){ int cnt0 = 0 , cnt1 = 0 , falg = (v[u] >> i & 1 ); for (auto x : a[u]){ int tt = (v[x] >> i & 1 ); cnt0 += (tt == 0 ); cnt1 += (tt == 1 ); } int ans1 = (1 << i); if (falg == 0 ){ ans1 = ans1 * (cnt1 * cnt0 % mod) % mod; }else { ans1 = ans1 * ((cnt0 * (cnt0 - 1 ) / 2 ) % mod + (cnt1 * (cnt1 - 1 ) / 2 ) % mod) % mod; } ans = (ans + ans1) % mod; } } cout << ans << endl; return 0 ; }
2025牛客暑期多校训练营3
E-Equal
针对于这个题目,其实最重要的是一个知识点,就是因数分解的知识点
常见的因数分解一个数字是:O(n)
1 2 3 4 5 6 for (int i = 2 ; i <= n; i++) { while (n % i == 0 ) { factors.push_back (i); n /= i; } }
但是针对于一些时间复杂度严苛的题目,这个算法便是不可行的,于是就有了预处理的解决办法
解决思路: 一个数的因式分解 == 这个数字的最小质因数 * (m),这样可以将 m 不断分解成最小质因数,这样的乘机也是因式分子
一个数字的时间复杂度为 O(log(n))
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 const int N = 5e6 + 10 ;vector<int > prime (N + 1 , 0 ) ;void init () { for (int i = 2 ; i <= N; i++){ if (prime[i] == 0 ){ prime[i] = i; } for (int j = 2 ; j * i <= N; j++){ if (prime[j * i] == 0 ){ prime[j * i] = prime[i]; } } } } signed main () { init (); map<int , int > mp; for (int i = 1 ; i <= n; i++){ while (a[i] > 1 ){ int t = prime[a[i]]; mp[t]++; a[i] /= t; } } }
牛客周赛Round-107
F
这个题目是一个离线处理的题目,也是一个很好又很经典的离线题目
题意:
我一开始是想通过DP来找到这个位置的最早被覆盖的位置,其实和题目的想法差不多,但是我这么写的话就是没有这么简单…
思路: 一个很明显的递推关系,就是如果这个点被X覆盖了,那么比X大的覆盖图的大小只会比X的覆盖图更大(这个也是我想使用DP的原因,因为有一个很明显的递推关系),但是DP不行,因为没有一个明显的递推公式,所以通过离线处理,通过对X进行排序,在前面的基础上进行扩展
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 #include <bits/stdc++.h> using namespace std;#define int long long struct node { int val, x, y; bool operator < (const node &n) const { return val > n.val; } }; int dx[4 ] = {0 , 1 , -1 , 0 };int dy[4 ] = {1 , 0 , 0 , -1 };signed main () { int n, m, q; cin >> n >> m >> q; vector dj (n + 1 , vector<int >(m + 1 , 0 )) ; for (int i = 1 ; i <= n; i++){ for (int j = 1 ; j <= m; j++){ cin >> dj[i][j]; } } vector<array<int ,2>> t1 (q); for (int i = 0 ; i < q; i++){ cin >> t1[i][0 ]; t1[i][1 ] = i + 1 ; } sort (t1. begin (), t1. end (), [&](array<int ,2 > p1, array<int ,2 > p2){ return p1[0 ] < p2[0 ]; }); vector<int > ans (q + 1 , 0 ) ; vector vis (n + 1 , vector<bool > (m + 1 , 0 )) ; int cnt = 0 ; priority_queue<node> h; for (int i = 1 ; i <= m; i++){ h.push ({dj[1 ][i], 1 , i}); } for (int i = 0 ; i < q; i++){ auto temp = t1[i]; while (!h.empty () && temp[0 ] >= h.top ().val){ auto X = h.top (); h.pop (); if (vis[X.x][X.y]) continue ; vis[X.x][X.y] = 1 ; cnt++; for (int i = 0 ; i < 4 ; i++){ int tx = X.x + dx[i], ty = X.y + dy[i]; if (tx <= n && ty <= m && tx >= 1 && ty >= 1 ){ if (!vis[tx][ty]) h.push ({dj[tx][ty], tx, ty}); } } } ans[temp[1 ]] = cnt; } for (int i = 1 ; i <= q; i++) cout << ans[i] << " " ; }
目前赛时题目补完了,9月份之前补完所有题目 + 寒假多校 + HDU春季联赛的一些题目.
牛客多校第一场
I题
这个题目是一个区间DP,写了一会的递归形式想把这个题目的一些基础思想理解,但是写完以后发现题目看错了😭
整体思路:
记录每一个区间的每一种可能的{价值, b}
然后使用二分查找 子区间小于 b 的最小价值,只要求最接近的b的位置的信息(信息用前缀求最小来处理)
条件一:
条件二:
AND
总体思路:
就是一个大区间的价值是子区间中满足条件二的最小价值,所以针对于 满足条件二,我们就可以使用 前缀和 的操作,将小于条件二的最小价值存储在 大的 b 的元素(优化关键)
感觉:
区间DP像一种感觉,你仔细去思考其中的流程反而容易将自己绕晕,就是要深刻理解DP的灵魂,就是我目前的状态只跟我前面一个状态有关其实这个题目也好理解,因为 b要求递增,但是b其实就跟目前的切割点有关,所以我只要看b和子区间的b的关系就行了,b并没有顺推关系
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 void slove () { int n; cin >> n; vector<int > a (n + 1 ) ,sum (n + 1 , 0 ) ; for (int i = 1 ; i <= n; i++) cin >> a[i],sum[i] = sum[i - 1 ] + a[i]; vector dp (n + 1 , vector(n + 1 , vector<PII>())) ; for (int i = 1 ; i <= n; i++) dp[i][i].push_back ({0ll ,0ll }); auto get = [&](int x, vector<PII>& t) -> int { int l = 0 , r = t.size () - 1 , mid, ans = -1 ; while (l <= r){ mid = (l + r) >> 1 ; if (t[mid].sc <= x){ l = mid + 1 ; ans = t[mid].fr; }else r = mid - 1 ; } return ans; }; for (int len = 2 ; len <= n; len++){ for (int l = 1 , r = l + len - 1 ; r <= n; l++,r++){ for (int k = l; k < r; k++){ int l1 = sum[k] - sum[l - 1 ], l2 = sum[r] - sum[k]; int b = abs (l1 - l2), cost = (min (l1, l2) * ceil (log2 (l1 + l2))); int c_l1 = get (b, dp[l][k]), c_l2 = get (b, dp[k + 1 ][r]); if (len == n){ if (c_l1 == -1 || c_l2 == -1 ){ cout << -1 << " " ; }else cout << c_l1 + c_l2 + cost << " " ; continue ; } if (c_l1 == -1 || c_l2 == -1 ) continue ; dp[l][r].push_back ({c_l1 + cost + c_l2, b}); } sort (dp[l][r].begin (), dp[l][r].end (), [&](PII p1, PII p2){ if (p1. sc == p2. sc) return p1.f r < p2.f r; else return p1. sc < p2. sc; }); for (int i = 1 ; i < dp[l][r].size (); i++){ dp[l][r][i].fr = min (dp[l][r][i].fr, dp[l][r][i - 1 ].fr); } } } cout << "\n" ; }
牛客多校第二场
A题
这个题目相对比较基础,但是我比赛场上没有写出来,针对于 DP线形动态规划的问题还是不太敏感(其实比赛的时候想过 DP,但是没有把转移方程写出来,比较遗憾[主要是包子丸写的太快了,导致我方程都没写明白,😈])
题解思路:
针对于这个题目,自然而然的会想到 递归,然后又是针对于一个数据的线性访问,所以我们可以不由自主的想到 DP(动态规划)
状态转移方程的问题: 我一开始是想将 0,1都进行合并然后再去观察这个题目
G[i][j] 就表示 前i 个数字 中 以 j 结尾的 1的段的数量
f[i - 1][0]其实就是为了统计 k 的影响的,可以理解为k构造了多少个新的子序列
以下是代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 #include <bits/stdc++.h> using namespace std;#define int long long #define fr first #define sc second #define endl '\n' using PII = pair<int ,int >;const int mod = 998244353 ;void slove () { int n; cin >> n; int last = -1 ; vector<int > a; int t; for (int i = 0 ; i < n; i++){ cin >> t; if (t != last || t == -1 ){ a.push_back (t); last = t; } } n = a.size (); vector<array<int ,2>> f (n + 1 ,{0 ,0 }); vector<array<int ,2>> g (n + 1 ,{0 ,0 }); f[0 ][0 ] = 1 ; for (int i = 1 ; i <= n; i++){ if (a[i - 1 ] == 1 ){ g[i][1 ] = ((g[i - 1 ][0 ] + f[i - 1 ][0 ]) % mod + g[i - 1 ][1 ]) % mod; f[i][1 ] = (f[i - 1 ][0 ] + f[i - 1 ][1 ]) % mod; }else if (a[i - 1 ] == 0 ){ g[i][0 ] = (g[i - 1 ][0 ] + g[i - 1 ][1 ]) % mod; f[i][0 ] = (f[i - 1 ][0 ] + f[i - 1 ][1 ]) % mod; }else { g[i][1 ] = ((g[i - 1 ][0 ] + f[i - 1 ][0 ]) % mod + g[i - 1 ][1 ]) % mod; f[i][1 ] = (f[i - 1 ][0 ] + f[i - 1 ][1 ]) % mod; g[i][0 ] = (g[i - 1 ][0 ] + g[i - 1 ][1 ]) % mod; f[i][0 ] = (f[i - 1 ][0 ] + f[i - 1 ][1 ]) % mod; } } cout << (g[n][0 ] + g[n][1 ]) % mod << endl; } signed main () { ios::sync_with_stdio (false ); cin.tie (0 ); int t; cin >> t; while (t--){ slove (); } return 0 ; }
牛客多校第三场
B
题目介绍: 有四种操作,通过不超过 64次操作,使a, b, c相等
思路: 其实操作都是固定的,我可以先将 a 变成 c,然后在这个过程中 b先变成0,然后 b^a == c
但是这个思路还是很巧妙的.从中也有一个特殊性质: if(a != b) 则 a ^ 1 == b ,这个性质也是解答这个题目的关键
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 #include <bits/stdc++.h> using namespace std;#define int long long #define fr first #define sc second #define endl '\n' using PII = pair<int ,int >;void slove () { int a, b, c; cin >> a >> b >> c; if (a == b && b == c && c == a){ cout << "0" << "\n" ; cout << '\n' ; return ; } if (a == 0 && b == 0 && c != 0 ){ cout << "-1" << '\n' ; return ; } int ans = 0 ; vector<int > falg; int ha, hb, hc; auto get = [&](int x) -> int { int i; for (i = 31 ; i >= 0 ; i--){ if ((x >> i) & 1 == 1 ){ break ; } } return i; }; ha = get (a), hb = get (b), hc = get (c); if (ha > hb){ falg.push_back (4 ); b = b ^ a; }else if (ha < hb){ falg.push_back (3 ); a = a ^ b; } ha = max (ha, hb); if (ha < hc){ for (int i = 0 ; i <= ha; i++){ int t1 = (a >> (ha - i)) & 1 ; int t2 = (c >> (hc - i)) & 1 ; if (t1 != t2){ falg.push_back (3 ); a ^= b; } if (i < ha){falg.push_back (2 ); b >>= 1 ;} } for (int i = hc - ha - 1 ; i >= 0 ; i--){ a <<= 1 ; falg.push_back (1 ); if (((c >> i) & 1 ) == 1 ) { falg.push_back (3 ); a ^= b; } } b >>= 1 ; b ^= a; falg.push_back (2 ); falg.push_back (4 ); }else { for (int i = ha; i >= 0 ; i--){ int t1 = (a >> i) & 1 ; int t2 = (c >> i) & 1 ; if (t1 != t2){ falg.push_back (3 ); a ^= b; } b >>= 1 ; falg.push_back (2 ); } b ^= a; falg.push_back (4 ); } cout << falg.size ()<< endl; for (auto x : falg) cout << x << " " ; cout << endl; } signed main () { ios::sync_with_stdio (false ); cin.tie (0 ); int t; cin >> t; while (t--){ slove (); } return 0 ; }
牛客多校第九场
F
题目介绍: 给定一个长度为1的棒子,通过固定一个点选择90度,最少旋转多少次,能旋转到目标棒子的状态的最少次数
思路: 这道题目首先会想到就是暴力模拟,但是其实因为你是固定一个点去移动的话,两个点的状态很难考虑,我们可以尝试去思考一下,两个都是变化的状态能不能转变成就一个变化状态呢…我的想法就是因为你移动的时候,其实可以理解成棒子的中间坐标也是在移动的,而且这个移动是不需要看棒子固定哪一个点进行移动,目标的状态也只有一个,因为中间坐标其实就决定了棒子的状态 ,所以我们可以用中间坐标去替代棒子目前的状态
收获: 这道题给我的感觉就是其实思路是比较简单的,但是越是简单的思路就要考察你解决问题的速度,但是其实简单的思路中可能就隐藏了一些快速的解决方案…
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 #include <bits/stdc++.h> using namespace std;#define int long long #define fr first #define sc second #define endl '\n' using PII = pair<int ,int >;signed main () { ios::sync_with_stdio (false ); cin.tie (0 ); int t; cin >> t; while (t--){ array<int ,2> l, r, tl, tr; cin >> l[0 ] >> l[1 ] >> r[0 ] >> r[1 ] >> tl[0 ] >> tl[1 ] >> tr[0 ] >> tr[1 ]; array<double ,2> d, dt; d[0 ] = (double )(l[0 ] + r[0 ]) / 2 ; d[1 ] = (double )(l[1 ] + r[1 ]) / 2 ; dt[0 ] = (double )(tl[0 ] + tr[0 ]) / 2 ; dt[1 ] = (double )(tl[1 ] + tr[1 ]) / 2 ; int ans = max (abs (d[0 ] - dt[0 ]), abs (d[1 ] - dt[1 ])) * 2 ; cout << ans << endl; } return 0 ; }
牛客周赛110
F
相邻元素合并 == 区间划分
牛客周赛111
史上最有意思的结论场