A - 最大値

maxとかstd::max_elementとか

#!/usr/bin/env python3
_ = int(input())
print(max(list(map(int, input().split()))))

B - 和

sumとかstd::accumulateとか

#!/usr/bin/env python3
_ = input()
print(sum(map(int, input().split())))

C - カンマ区切り

tail -n 1あるいはreadを忘れてWAが生えた

read;tr \  ,

D - ソート

sort

#!/usr/bin/env python3
_ = input()
print(*sorted(map(int, input().split())))

E - 1は何番目?

list.indexstd::find

#!/usr/bin/env python3
_ = input()
print(list(map(int, input().split())).index(1) + 1)

F - 見える数

問題文がすごく難しい。正解は$\{ 1 \le i \le N \mid \forall j. 1 \le j \lt i \to a_j \lt a_i \}$の要素数。

#!/usr/bin/env python3
n = int(input())
a = list(map(int, input().split()))
max_a_j = -1
result = 0
for a_i in a:
    result += (max_a_j < a_i)
    max_a_j = max(max_a_j, a_i)
print(result)

G - あまり

多倍長整数は最高

#!/usr/bin/env python3
_ = int(input())
a = ''.join(input().split())
print(int(a) % (10 ** 9 + 7))

H - LIS

$O(N\ log N)$。 貼る。あるいは検索。 ライブラリになかったので足しておいた。

#include <cstdio>
#include <vector>
#define repeat(i, n) for (int i = 0; (i) < int(n); ++(i))
using namespace std;

template <typename T>
vector<T> longest_increasing_subsequence(vector<T> const & xs) {
    vector<T> l; // l[i] is the last element of the increasing subsequence whose length is i+1
    l.push_back(xs.front());
    for (auto && x : xs) {
        auto it = lower_bound(l.begin(), l.end(), x);
        if (it == l.end()) {
            l.push_back(x);
        } else {
            *it = x;
        }
    }
    return l;
}

int main() {
    int n; scanf("%d", &n);
    vector<int> a(n); repeat (i, n) scanf("%d", &a[i]);
    printf("%d\n", int(longest_increasing_subsequence(a).size()));
    return 0;
}

I - 和がNの区間

しゃくとり法。$O(N)$。

#include <cstdio>
#include <vector>
#define repeat(i, n) for (int i = 0; (i) < int(n); ++(i))
using namespace std;

int main() {
    int n; scanf("%d", &n);
    vector<int> a(n); repeat (i, n) scanf("%d", &a[i]);
    int result = 0;
    for (int l = 0, r = 0, acc = 0; l < n; acc -= a[l], ++ l) {
        while (r < n and acc + a[r] <= n) { acc += a[r]; ++ r; }
        result += acc == n;
    }
    printf("%d\n", result);
    return 0;
}

J - 転倒数

$O(N \log N)$。 ライブラリになかったので足しておいた2。

#include <algorithm>
#include <cstdio>
#include <vector>
#define repeat(i, n) for (int i = 0; (i) < int(n); ++(i))
using ll = long long;
using namespace std;

template <typename Monoid>
struct binary_indexed_tree { // on monoid
    typedef typename Monoid::underlying_type underlying_type;
    vector<underlying_type> data;
    Monoid mon;
    binary_indexed_tree(size_t n, Monoid const & a_mon = Monoid()) : mon(a_mon) {
        data.resize(n, mon.unit());
    }
    void point_append(size_t i, underlying_type z) { // data[i] += z
        for (size_t j = i + 1; j <= data.size(); j += j & -j) data[j - 1] = mon.append(data[j - 1], z);
    }
    underlying_type initial_range_concat(size_t i) { // sum [0, i)
        underlying_type acc = mon.unit();
        for (size_t j = i; 0 < j; j -= j & -j) acc = mon.append(data[j - 1], acc);
        return acc;
    }
};
struct plus_t {
    typedef int underlying_type;
    int unit() const { return 0; }
    int append(int a, int b) const { return a + b; }
};

ll inversion_number(vector<int> const & a) {
    int n = a.size();
    binary_indexed_tree<plus_t> bit(n + 1);
    ll result = 0;
    repeat (i, n) {
        result += i - bit.initial_range_concat(a[i] + 1);
        bit.point_append(a[i], 1);
    }
    return result;
}

int main() {
    int n; scanf("%d", &n);
    vector<int> a(n); repeat (i, n) scanf("%d", &a[i]);
    ll result = inversion_number(a);
    printf("%lld\n", result);
    return 0;
}

K - 辞書順で何番目?

$O(N \log N)$。$n_i = \{ j \mid i \lt j \land a_i \gt a_j \}$として$1 + \sum_i n_i (n - i - 1)!$。

階乗進数ぽさがあったから上手にできないかなと思ったが思い付かず。

#include <algorithm>
#include <cstdio>
#include <vector>
#define repeat(i, n) for (int i = 0; (i) < int(n); ++(i))
#define repeat_from(i, m, n) for (int i = (m); (i) < int(n); ++(i))
using ll = long long;
using namespace std;

template <typename Monoid>
struct binary_indexed_tree { // on monoid
    typedef typename Monoid::underlying_type underlying_type;
    vector<underlying_type> data;
    Monoid mon;
    binary_indexed_tree(size_t n, Monoid const & a_mon = Monoid()) : mon(a_mon) {
        data.resize(n, mon.unit());
    }
    void point_append(size_t i, underlying_type z) { // data[i] += z
        for (size_t j = i + 1; j <= data.size(); j += j & -j) data[j - 1] = mon.append(data[j - 1], z);
    }
    underlying_type initial_range_concat(size_t i) { // sum [0, i)
        underlying_type acc = mon.unit();
        for (size_t j = i; 0 < j; j -= j & -j) acc = mon.append(data[j - 1], acc);
        return acc;
    }
};
struct plus_t {
    typedef int underlying_type;
    int unit() const { return 0; }
    int append(int a, int b) const { return a + b; }
    int invert(int a) const { return - a; }
};
template <int mod>
int fact(int n) {
    static vector<int> memo(1, 1);
    if (memo.size() <= n) {
        int l = memo.size();
        memo.resize(n+1);
        repeat_from (i, l, n+1) memo[i] = memo[i-1] *(ll) i % mod;
    }
    return memo[n];
}

constexpr int mod = 1e9+7;
int main() {
    int n; scanf("%d", &n);
    vector<int> a(n); repeat (i, n) { scanf("%d", &a[i]); -- a[i]; }
    binary_indexed_tree<plus_t> bit(n + 1);
    repeat (i, n) bit.point_append(i, 1);
    ll result = 1;
    repeat (i, n) {
        result += bit.initial_range_concat(a[i]) *(ll) fact<mod>(n - i - 1) % mod;
        bit.point_append(a[i], -1);
    }
    result %= mod;
    printf("%lld\n", result);
    return 0;
}

L - N回スワップ

$i - a_i$に辺を張って無向グラフを作り連結成分数を$k$とする。 $n - k$回のswapでsortできてこれが最小。 置換の偶奇は変えられないので、$n - k \equiv 0 \pmod{2}$かどうかが答え。

#!/usr/bin/env python3
n = int(input())
a = list(map(lambda s: int(s) - 1, input().split()))
used = [ False ] * n
cnt = 0
for i in range(n):
    j = i
    used[j] = True
    while a[j] != j:
        j = a[j]
        if used[j]:
            break
        used[j] = True
        cnt += 1
print(['NO', 'YES'][(n - cnt) % 2 == 0])