常州大學新生寒假訓練會試 題解
【題目鏈接】
A - 添加逗號
註意是從後往前三個三個加逗號,最前面不允許有逗號
#include <bits/stdc++.h> using namespace std; const int maxn = 1e5 + 10; char s[maxn]; char ans[maxn]; int sz; int main() { scanf("%s", s); int len = strlen(s); sz = 0; int t = 0; for(int i = len - 1; i >= 0; i --) { ans[sz ++] = s[i]; ans[sz] = 0; t ++; if(t % 3 == 0 && i != 0) ans[sz ++] = ‘,‘, ans[sz] = 0; } len = strlen(ans); for(int i = len - 1; i >= 0; i --) { printf("%c", ans[i]); } printf("\n"); return 0; }
B - 對稱
可以遞歸求解。
#include <bits/stdc++.h> using namespace std; long long n, m; long long work(long long r, long long c) { if(r % 2 == 0 || c % 2 == 0) return 0LL; return 4 * work(r / 2, c / 2) + 1LL; } int main() { scanf("%lld%lld", &n, &m); printf("%lld\n", work(n, m)); return 0; }
C - 競賽技巧
排序。
#include <bits/stdc++.h> using namespace std; const int maxn = 1e5 + 10; struct X{ int h, m, s; void out() { printf("%d %d %d\n", h, m, s); } }s[maxn]; int n; bool cmp(const X&a, const X&b) { if(a.h != b.h) return a.h < b.h; if(a.m != b.m) return a.m < b.m; return a.s < b.s; } int main() { scanf("%d", &n); for(int i = 1; i <= n; i ++) { scanf("%d%d%d", &s[i].h, &s[i].m, &s[i].s); } sort(s + 1, s + 1 + n, cmp); for(int i = 1; i <= n; i ++) { s[i].out(); } return 0; }
D - 訓練技巧
設$dp[0][i]$表示以$i$為結尾的最大價值,$dp[1][i]$表示$j(j < i)$為結尾的最大價值。可見,該$dp$為$O(n^2)$效率的。
可以拿一個單調隊列優化,明天更新。
#include <bits/stdc++.h> using namespace std; const int maxn = 1e5 + 10; int n, k; long long a[maxn]; long long dp[2][maxn]; long long sum[maxn]; int q[5 * maxn]; int first, last; bool check2(int a, int b) { if(dp[1][b] - dp[1][a] > sum[b] - sum[a]) return 1; return 0; } int main() { scanf("%d%d", &n, &k); for(int i = 1; i <= n; i ++) { scanf("%lld", &a[i]); sum[i] = sum[i - 1] + a[i]; } /* dp[0][i] // 以i結尾 dp[1][i] // 不以i結尾 */ first = 0, last = -1; for(int i = 1; i <= k; i ++) { dp[0][i] = sum[i]; dp[1][i + 1] = max(dp[0][i], dp[1][i]); while(1) { if(last - first + 1 == 0) break; if(check2(q[last], i)) last --; else break; } last ++; q[last] = i; } for(int i = k + 1; i <= n; i ++) { while(1) { if(last < first) break; if(i - q[first] > k) first ++; else break; } /* printf("debug %d: ", i); for(int i = first; i <= last; i ++) { printf("%d ", q[i]); } printf("\n"); */ dp[0][i] = dp[1][q[first]] + sum[i] - sum[q[first]]; dp[1][i + 1] = max(dp[0][i], dp[1][i]); while(1) { if(last - first + 1 == 0) break; if(check2(q[last], i)) last --; else break; } last ++; q[last] = i; /* printf("debug %d: ", i); for(int i = first; i <= last; i ++) { printf("%d ", q[i]); } printf("\n"); */ } for(int i = 1; i <= n; i ++) { // printf("%d %lld %lld\n", i, dp[0][i], dp[1][i]); } long long ans = 0; for(int i = 1; i <= n; i ++) { ans = max(ans, dp[0][i]); ans = max(ans, dp[1][i]); } printf("%lld\n", ans); return 0; }
E - 這是一個數學題
化簡之後可以發現$A_i = A_0*\frac{n-i}{n} + A_n*\frac{i}{n}$,這樣就很容易求解了。
import java.math.BigDecimal; import java.math.BigInteger; import java.util.*; public class Main { static Scanner cin = new Scanner(System.in); public static void main(String[] args) { int n = cin.nextInt(); long a0 = cin.nextLong(); long an = cin.nextLong(); int Q = cin.nextInt(); while(Q -- > 0) { long L = cin.nextLong(); long R = cin.nextLong(); long len = R - L + 1; BigInteger ans = BigInteger.ZERO; BigInteger A = BigInteger.ZERO;; BigInteger B = BigInteger.ZERO;; long x = len * n; long y = (L + R) * len / 2; A = BigInteger.valueOf(x).subtract(BigInteger.valueOf(y)); A = A.multiply(BigInteger.valueOf(a0)); A = A.divide(BigInteger.valueOf(n)); B = BigInteger.valueOf(an).multiply(BigInteger.valueOf(y)).divide(BigInteger.valueOf(n)); ans = A.add(B); System.out.println(ans); } } }
F - 大佬的生日大禮包
如果$x$個人能滿足,那麽$[0,x]$個人都能滿足,依據這個性質,我們可以對答案進行二分。
接下來就是驗證$x$個人能否滿足:
首先觀察到,無論是哪一種大禮包,都會用掉一個U盤和一個鼠標,除此之外,每種禮包再外加一個物品。
因此U盤或者鼠標數量不足$x$個,則無解。
將U盤和鼠標數量都減去$x$個後,問題就變成了:三種物品分別有$p_0$,$p_1$,$p_2$個,問是否存在排列方案使得相鄰兩個物品種類不同。
只要驗證$\sum_{i=0}^2min(p_i, \frac{x+1}{2})$和$x$的大小關系即可。
#include <bits/stdc++.h> using namespace std; int T, a, b ,c; int check(int x) { int p[3]; p[0] = a - x; p[1] = b - x; p[2] = c; if(p[0] < 0 || p[1] < 0) return 0; for(int i = 0; i < 3; i ++) { p[i] = min(p[i], (x + 1) / 2); } if(p[0] + p[1] + p[2] >= x) return 1; return 0; } int main() { scanf("%d", &T); while(T --) { scanf("%d%d%d", &a, &b, &c); int L = 0, R = a + b + c, ans = 0; while(L <= R) { int mid = (L + R) / 2; if(check(mid)) ans = mid, L = mid + 1; else R = mid - 1; } printf("%d\n", ans); } return 0; }
G - 零下e度
公式1:$\frac{1}{e}= \sum_{i = 0}^{\infty} (-1)^i\frac{1}{i!}$
公式2:$e= \sum_{i = 0}^{\infty} \frac{1}{i!}$
知道上面這個公式就會做了。這題卡常數,$mod$如果不是const定義的話容易超時。
#include <bits/stdc++.h> using namespace std; const int mod = 998244353LL; long long ans; int n; int main() { scanf("%d", &n); long long u = 1; int p = (n % 2) ? -1 : 1; for(int i = n; i >= 1; i --) { ans = (ans + p * u + mod) % mod; u = (u * i) % mod; p = p * -1; } ans = (ans + u) % mod; printf("%lld\n", ans); return 0; }
H - 酸堿滴定
模擬題。還沒寫。
I - 合成反應
暴力,每次判斷那些還不能執行的方程式,直到不能增加元素為止。
#include <bits/stdc++.h> using namespace std; const int maxn = 1e5 + 10; int f[maxn]; int a[maxn], b[maxn], c[maxn]; int q, n, k, m; set<int> st; int main() { scanf("%d%d%d%d", &k, &n, &m, &q); for(int i = 1; i <= n; i ++) { scanf("%d%d%d", &a[i], &b[i], &c[i]); st.insert(i); } for(int i = 1; i <= m; i ++) { int x; scanf("%d", &x); f[x] = 1; } vector<int> vec; while(!st.empty()) { set<int>::iterator it; vec.clear(); for(it = st.begin(); it != st.end(); it ++) { int id = *it; if((f[a[id]] && f[b[id]]) || f[c[id]]) { f[a[id]] = 1; f[b[id]] = 1; f[c[id]] = 1; vec.push_back(id); } } if(vec.size() == 0) break; for(int i = 0; i < vec.size(); i ++) { st.erase(vec[i]); } } while(q --) { int x; scanf("%d", &x); printf("%s\n", f[x] ? "Yes" : "No"); } return 0; }
J - 同分異構體
手算 or OEIS
#include <bits/stdc++.h> using namespace std; int ans[] = {1, 1, 1, 1, 2, 3, 5, 9, 18, 35, 75, 159}; int main() { int n; scanf("%d", &n); printf("%d\n", ans[n]); return 0; }
常州大學新生寒假訓練會試 題解