【LGR-067】洛谷 1 月月赛 II & CSGRound 3 Div.2 题解
Visits: 374
压岁钱
模拟。
封印的钱拿个数组记录就行了。
const int N = 1e6 + 7;
int n, o, x, y, ans;
ll c[N], now;
int main() {
rd(n);
for (int i = 1; i <= n; i++) {
now += c[i], rd(o), rd(x);
switch (o) {
case 1 : now += x; break;
case 2 :
if (now < x) ++ans;
else now -= x;
break;
case 3 :
now -= x, rd(y), c[y] += x;
}
}
print(ans);
return 0;
}
斗牛
选 $n-2$ 张牌相当于选两张牌,同时如果有解,答案一定是 $((\Sigma – 1) \bmod 10) + 1$。
开个大小为 $10$ 的桶即可。
const int N = 1e6 + 7;
int n, a[N], s, c[10];
int main() {
rd(n);
for (int i = 1; i <= n; i++) rd(a[i]), a[i] %= 10, s += a[i];
for (int i = 1; i <= n; i++) {
int x = (s % 10 - a[i] + 10) % 10;
if (c[x]) return print((s - 1) % 10 + 1), 0;
c[a[i]%10] = 1;
}
print(0);
return 0;
}
游戏
枚举先手取牌的张数,贪心出此时合法的 $X$ 的范围,差分即可。
贪心时二分可以做到 $\mathcal O(n \log n)$,双指针可以做到 $\mathcal O(n)$。
const int N = 1e6 + 7;
int n, k, c[N];
ll a[N];
int main() {
rd(n);
for (int i = 1; i <= n; i++) rd(a[i]), a[i] += a[i-1];
a[++n] = 1e18, rd(k);
for (int i = 1, p = 2; i < n; i++) {
while (a[p] < 2 * a[i]) ++p;
if (a[i] <= k) ++c[a[i]];
if (a[p] - a[i] <= k) --c[a[p]-a[i]];
}
vi ans;
for (int i = 1; i <= k; i++)
if ((c[i] += c[i-1])) ans.pb(i);
print(ans.size());
for (auto x : ans) print(x, ' ');
return 0;
}
出游
套路题。
给定一张 $n$ 个点的有向图,一开始选择一些点,这些点每次走一步,问 $T$ 步后这些点可能在哪些位置。
类似这样的问题都可以用 bitset 优化矩阵乘法做到 $\mathcal O(\frac{n^3\log T}w)$,比如 CF576D Flights for Regular Customers。
我这题代码就是从那题的代码改的。
最终你会得到一个矩阵 $a$,$a_{i,j}$ 表示一开始如果 $i$ 被选择了,那么最终 $j$ 会被选。
那么 $j$ 最后不参加的概率就是所有 $a_{i,j} = 1$ 的 $i$ 一开始不参加的概率。
人数的期望就是所有人参加的概率之和。
const int N = 500;
int n, m;
modint p[N], ans;
#define bt bitset<N>
struct mt {
bt a[N];
inline friend mt operator * (mt x, mt y) {
mt z;
for (int i = 0; i < n; i++)
for (int j = 0; j < n; j++)
if (x.a[i][j]) z.a[i] |= y.a[j];
return z;
}
} a, b;
inline void ksm(mt x, int y, mt &z) {
while (y) {
if (y & 1) z = z * x;
x = x * x, y >>= 1;
}
}
int main() {
rd(n), rd(m);
for (int i = 0, k; i < n; i++) {
rd(p[i]), rd(k);
for (int j = 0, x; j < k; j++)
rd(x), a.a[--x][i] = 1;
b.a[i][i] = 1;
}
ksm(a, m, b);
for (int i = 0; i < n; i++) {
modint o = 1;
for (int j = 0; j < n; j++)
if (b.a[j][i]) o *= 1 - p[j];
ans += 1 - o;
}
print(ans);
return 0;
}