【LGR-067】洛谷 1 月月赛 II & CSGRound 3 Div.2 题解

作者: xht37 分类: 题解 发布时间: 2020-01-29 14:09

点击数:114

【LGR-067】洛谷 1 月月赛 II & CSGRound 3 Div.2

压岁钱

模拟。

封印的钱拿个数组记录就行了。

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;
}

发表评论

电子邮件地址不会被公开。 必填项已用*标注