2024.1.16做题纪要

硬币

多少有些人类智慧了。。。。。

题解写的还行。

具体就是每次把当前这一位代表的质数 \(i\) 向后每隔 \(i\) 个数除上 \(i\)

这一位肯定是一个质数,因为若是合数则前面一定会被除上质数。

Kaiserredux
#include <bits/stdc++.h>

long long num[1100000];
long long answer[1100000];

int main() {
    std::ios::sync_with_stdio(false);
    std::cin.tie(0);
    std::cout.tie(0);

    freopen("coins.in", "r", stdin);
    freopen("coins.out", "w", stdout);

    for (long long i = 1; i <= 1e6; ++ i) {
        answer[i] = num[i] = 1ll * i * i + 1ll;

    }
    for (long long i = 1; i <= 1000000; ++ i) {
        if (num[i] == 1)
            continue;
        long long x = num[i];
        for (long long j = i; j <= 1000000; j += x) {
            answer[j] = std::min(answer[j], x);
            while (num[j] % x == 0)
                num[j] /= x;
        }
    }
    long long Q, N;
    std::cin >> Q;
    while (Q --) {
        std::cin >> N;
        if (answer[N] == N * N + 1ll) 
            std::cout << -1 << '\n';
        else
            std::cout << answer[N] << ' ' << (N * N + 1) / answer[N] << '\n';
    }
    return 0;
}

猜数

逆天单调队列优化dp + 大眼观察法。

根本不会证QAQ。

56 Road
// ubsan: undefined
// accoders
#include <bits/stdc++.h>

int N;
long long sum[52100], dp[4000][4000];

class Monotonic_Queue {
public:
    int l, r;
    std::pair<int, long long> value[52100];

    Monotonic_Queue() {
        l = 1, r = 0;
    }

    void pop_from_range(int rRange) {
        while (l <= r && value[l].first > rRange)
            l ++;
    }
    
    void emplace(long long val, int pos) {
        while (l <= r && value[r].second > val)
            r --;
        value[++ r] = std::make_pair(pos, val); 
    }

    long long front() {
        return value[l].second;
    }

    int front_first() {
        return value[l].first;
    }

    void clear() {
        l = 1, r = 0;
    }
}queue;

int to[51000];

int Len(int l, int r) {
    return r - l + 1;
}

int main() {
    std::ios::sync_with_stdio(false);
    std::cin.tie(0);
    std::cout.tie(0);

    freopen("guess.in", "r", stdin);
    freopen("guess.out", "w", stdout);

    int S = 3750;
    std::cin >> N;
    for (int r = S + 110; r >= 1; -- r) 
        for (int l = 2; l <= S; ++ l)
            dp[r][l] = 1e18;
    for (int i = 1; i <= N; ++ i)
        to[i] = i % 3800, sum[i] = 1e18;
    sum[1] = 0;
    long long answer = 0;
    for (int r = 2; r <= N; ++ r) {
        int mid = r;
        queue.clear();
        for (int l = r - 1; r - l + 1 <= std::min(r, S); -- l) {
            queue.emplace(dp[to[r]][Len(l + 1, r)] + l, l);
            while (mid > l && dp[to[mid - 1]][Len(l, mid - 1)] >= dp[to[r]][Len(mid + 1, r)]) {
                mid --;
                queue.pop_from_range(mid);
            }
            dp[to[r]][Len(l, r)] = std::min(queue.front(), dp[to[mid]][Len(l, mid)] + mid + 1);
            sum[r] = std::min(sum[r], std::max(sum[l - 1], dp[to[r]][Len(l + 1, r)]) + l);
        }
        answer += sum[r];
    }
    std::cout << answer << '\n';
    return 0;
}

P3403 跳楼机

同余最短路板子题。

我们先设 \(f_i\) 表示我们只用操作 \(2,3\) 使得 \(f_i \% x = i\),并且让 \(f_i\) 最小。每次将 \(i\) 连向 \((i+y)\%x\)\((i+z)\%x\),边权分别为 \(y\)\(z\)。然后跑最短路就行了。

燃烧的世界
#include <bits/stdc++.h>

typedef long long ll;

class Point {
public:
    ll val, poi;

    bool operator <(const Point &b) const {
        return val > b.val;
    }

    Point(ll _val, ll _poi) {
        val = _val;
        poi = _poi;
    }
};

int cnt, head[110000], next[210000], to[210000];
ll value[210000];

void AddEdge(int u, int v, ll val) {
    ++ cnt;
    next[cnt] = head[u];
    head[u] = cnt;
    to[cnt] = v;
    value[cnt] = val;
}

ll f[110000];
bool visit[110000];
ll H, X, Y, Z;

void Dijkstra(int begin) {
    memset(visit, 0, sizeof(visit));
    for (int i = 0; i < X; ++ i)
        f[i] = 2e18;
    f[1] = 1;
    std::priority_queue<Point> queue;

    queue.emplace(f[1], 1);

    while (queue.size()) {
        int now = queue.top().poi;
        queue.pop();

        if (visit[now])
            continue;

        visit[now] = true;
        for (int i = head[now]; i; i = next[i]) {
            if (f[to[i]] > f[now] + value[i]) {
                f[to[i]] = f[now] + value[i];
                queue.emplace(f[to[i]], to[i]);
            }
        }
    }
}

int main() {
    std::ios::sync_with_stdio(false);
    std::cin.tie(0);
    std::cout.tie(0);

    std::cin >> H >> X >> Y >> Z;
    if (X == 1 || Y == 1 || Z == 1) {
        std::cout << H << '\n';
        return 0;
    }
    if (X > Y)
        std::swap(X, Y);
    if (X > Z)
        std::swap(X, Z);
    for (int i = 0; i < X; ++ i) {
        AddEdge(i, (i + Y) % X, Y);
        AddEdge(i, (i + Z) % X, Z);
    }
    Dijkstra(1);
    ll answer = 0;
    for (int i = 0; i < X; ++ i) {
        if (f[i] > H)
            continue;
        answer += (H - f[i]) / X + 1;
    }
    std::cout << answer << '\n';
    return 0;
}
posted @ 2024-01-16 21:29  觉清风  阅读(29)  评论(4编辑  收藏  举报