1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

ABC405

Rated !

A - Is it rated?

Xが1のときと2のときで場合分けして条件分け
とてもシンプル

ABC405_A
int R, X; cin >> R >> X;
if (X == 1) {
    if (1600 <= R && R <= 2999) 
        cout << "Yes" << endl;
    else cout << "No" << endl;
}
else {
    if (1200 <= R && R <= 2399) 
        cout << "Yes" << endl;
    else cout << "No" << endl;
}

B - Not All

C問以降と比べ自由度が高過ぎて悩んでしまった
forループでもとのリストの長さを短くしていき、
1-Mの数字で0個のものを更に2重ループで探した
流石に3重ループは避けられるなら避けたい

解説見たうえで考えるなら頭からその数字が過去に出たかboolリストを作って
全部trueになる=trueにする行為をN回行う(それを数えれば良さそう
↓は時間内に提出したものなので3重ループ

ABC405_B
int N, M; cin >> N >> M;
vector<int> A(N); for (auto &x : A) cin >> x;
for (int i = N; ; i--) {
    for (int j = 1; j <= M; j++) {
        int c = 0;
        for (int k = 0; k < i; k++) {
            if (A[k] == j) c++;
        }
        if (c == 0) {
            cout << N - i << endl;
            return 0;
        }
    }
}

C - Sum of Product

解説のように$S_{i}=\sum_{k=1}^{i}A_{i}$としたときに$\sum_{i=2}^{N}A_{i}S_{i-1}$となるのはわかったうえでちょっと違う手を取った
値の合計を2条したとき個々で掛け算したものとして表にする
例えば$(5+3+7+4)^2$を考える

5 3 7 4
5 25 15 35 20
3 15 9 21 12
7 35 21 49 28
4 20 12 28 16

すると今回必要なのは以下の数字部分だとわかる

5 3 7 4
5 O X X X
3 15 O X X
7 35 21 O X
4 20 12 28 O

不要であるOの部分は要素の2乗の合計で求められ
Xの部分は必要な部分と同じである

そのため合計の2条から要素を2条したものの合計を引き2で割れば出すことができる

ABC405_C
int N; cin >> N;
vector<int> A(N); for (auto &x : A) cin >> x;
int64_t Sum1 = 0;
for (auto x : A) Sum1 += x;
Sum1 *= Sum1;
int64_t Sum2 = 0;
for (auto x : A) Sum2 += x * x;
Sum1 -= Sum2;
Sum1 /= 2;
cout << Sum1 << endl;

D - Escape Route

わかりやすくBFS
Queueに座標とどの方向から来た保存すれば大丈夫
複数queueに入る可能性があるのでqueueにいれるときだけでなくqueueの処理が始まる前に処理済みでないかをチェック
↑なぜか忘れていて1度WRを食らった
最近思っていた上下左右に1列ずつ挿入して範囲外エラー処理分岐を減らすあんがかなりうまく行った

ABC405_D
vector<int> dh = {0, 1, 0, -1}, dw = {1, 0, -1, 0};

int H, W; cin >> H >> W;
vector<vector<char>> S(H + 2, vector<char>(W + 2, 'x'));
for (int h = 1; h < H + 1; h++) {
    for (int w = 1; w < W + 1; w++) {
        cin >> S[h][w];
    }
}

queue<vector<int>> Q;
for (int h = 1; h < H + 1; h++) {
    for (int w = 1; w < W + 1; w++) {
        if (S[h][w] == 'E') {
            Q.push({h, w, 0});
        }
    }
}
vector<char> DL = {'<', '^', '>', 'v'};
while (Q.size()) {
    int h = (Q.front())[0];
    int w = (Q.front())[1];
    int d = (Q.front())[2];
    Q.pop();
    if (!(S[h][w] == '.' || S[h][w] == 'E')) continue;
    if (S[h][w] != 'E') {
        S[h][w] = DL[d];
    }
    for (int i = 0; i <= 3; i++) {
        int th = h + dh[i];
        int tw = w + dw[i];
        if (S[th][tw] != '.') continue;
        Q.push({th, tw, i});
    }
}

for (int h = 1; h < H + 1; h++) {
    for (int w = 1; w < W + 1; w++) cout << S[h][w];
    cout << endl;
}

E - Fruit Lineup

苦手な 何通り・あまり問題
しかし!おそらく解説通りの方法で実装に成功

条件は
リンゴ<バナナ
リンゴ<ブドウ
オレンジ<ブドウ
Aリンゴ Bオレンジ Cバナナ Dブドウ

自分はまずリンゴとブドウの関係に注目して、
この2つだけを見ると必ず1パターンになる
(種類と個数を表す

リンゴ バナナ
A C

ここにバナナをいれるのだが、オレンジをいれるときに一番左のブドウが重要になるのでそれだけ区別して考える

リンゴ バナナ 左ブドウ (ブドウ)と(バナナ)の混合
A i 1 (D-1) (C-i)

バナナとブドウを混ぜたときにいくつかのバナナは一番左のブドウより左にあるだろう それをiとして表した形だ
ここにオレンジを追加するとブドウより左にあるものと混ざる

(オレンジ)と(リンゴとバナナ)の混合 左ブドウ (バナナ)と(ブドウ)の混合
(B) (A+i) 1 (D-1) (C-i)

左のリンゴとバナナというのは必ず左が全部リンゴであり右がバナナであるので1種類として考えると
2種類の順列として考えると$\frac{\left(B+A+i\right)!}{\left(B\right)!\cdot\left(A+i\right)!}$になる
同じように右のバナナとブドウの混合は$\frac{\left(D-1+C-i\right)!}{\left(D-1\right)!\cdot\left(C-i\right)!}$と表せる
つまり$\sum_{i=0}^{C}\frac{\left(B+A+i\right)!}{\left(B\right)!\cdot\left(A+i\right)!}\cdot\frac{\left(D-1+C-i\right)!}{\left(D-1\right)!\cdot\left(C-i\right)!}$これを求めればいい
値は100万未満なのでiでループかけそれぞれで計算を行い合計を出せば良い
998244353で割ったあまりであることと階乗をいちいち計算すると間に合わないので事前に全部計算することだけ気をつければ問題なし

モッドの割り算はとても難しいので事前に作ったのを引っ張り出してきた
(整理できてない範囲だったので危うく間に合わないところだった、、

あまりの計算
int64_t mod (int64_t a) {
    return (a % MOD + MOD) % MOD;
}
int64_t modinv (int64_t a, int64_t b) {
    if (a % b == 1) return 1;
    return (1 - b * modinv(b , a % b)) / (a % b);
}
int64_t mdiv (int64_t a, int64_t b) {
    return mod(a * modinv(b, MOD));
}
ABC405_E
int64_t A, B, C, D; cin >> A >> B >> C >> D;
int64_t Sum = 0;

vector<int64_t> fact(10000000, 1);
int64_t f_tmp = 1;
for (int64_t i = 1; i < 10000000; i++) {
    f_tmp *= i;
    f_tmp %= MOD;
    fact[i] = f_tmp;
}

for (int64_t i = 0; i <= C; i++) {
    int64_t a = mdiv(mdiv(fact[C - i + D - 1], fact[C - i]), fact[D - 1]);
    int64_t b = mdiv(mdiv(fact[B + A + i], fact[B]), fact[A + i]);;
    int64_t c = (a * b) % MOD;
    Sum = (Sum + c) % MOD;
}

cout << Sum << endl;

まとめ

5冠達成できてよかった!
が、タイムが102分とだいぶ遅かった(1ペナ)
一応スコア1234と水パフォ
出てレート上がったがもう少し頑張れたのでは?と思うが3ヶ月空きの復帰戦だから仕方ないか、
image.png
5月10日

1
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?