Hướng dẫn giải của Atcoder Educational DP Contest N - Slimes


Chỉ dùng lời giải này khi không có ý tưởng, và đừng copy-paste code từ lời giải này. Hãy tôn trọng người ra đề và người viết lời giải.
Nộp một lời giải chính thức trước khi tự giải là một hành động có thể bị ban.

Gọi ~dp[i][j]~ là chi phí nhỏ nhất để ghép các cục thạch thứ ~i~ tới cục thạch thứ ~j~.

Để tính ~dp[i][j]~, ta muốn ghép hai cục thạch nhỏ hơn đã được ghép trong đoạn ~[i, j]~.

Ta duyệt qua tất cả ~k~ trong đoạn ~[i,j-1]~ và thử kết hợp hai cục thạch được tạo từ đoạn ~[i,k]~ và ~[k+1,j]~.

Chi phí thực hiện điều trên là ~dp[i][k] + dp[k+1][j] + a[i] + a[i+1] + ... + a[j]~. Để tính ~a[i] + a[i+1] + ... + a[j]~ ta có thể sử dụng mảng cộng dồn.

Độ phức tạp tính toán là ~O(N^3)~.

Bạn có thể tối ưu hơn nữa bằng KNUTH'S OPTIMIZATION (bài viết được viết bằng tiếng Anh).

#include <bits/stdc++.h>
using namespace std;

const int MAXN = 400;
const long long INF = 0x3f3f3f3f3f3f3f3f;

long long query(vector<long long>& p, int l, int r){
    if( l == 0 ){
        return p[r];
    }
    else return p[r] - p[l - 1];
}

int main() {
    ios_base::sync_with_stdio(0);cin.tie(0);cout.tie(0);

    int n;
    cin >> n;
    vector<long long> a(n), p(n);
    for (long long& x : a) cin >> x;
    for (int i = 0; i < n; ++i)
        p[i] = (i == 0 ? 0 : p[i - 1]) + a[i];

    static long long dp[MAXN][MAXN];
    memset(dp, 0x3f, sizeof(dp));

    for (int i = 0; i + 1 < n; ++i) dp[i][i + 1] = a[i] + a[i + 1];

    for (int i = 0; i < n; ++i) dp[i][i] = 0;

    for (int i = n - 1; i >= 0; --i)
        for (int j = i + 2; j < n; ++j) {
            long long best = INF;
            for (int k = i; k < j; ++k)
                best = min(best, dp[i][k] + dp[k + 1][j]);
            dp[i][j] = query(p, i, j) + best;
        }
    cout << dp[0][n - 1] << '\n';

    return 0;
}

Bình luận

Hãy đọc nội quy trước khi bình luận.



  • 10
    Motconmeocon  đã bình luận lúc 8, Tháng 6, 2023, 5:24

    Trực quan : coi ban đầu các viên thạch được để trên các cái bàn, cái bàn thứ (i) chứa viên thạch thứ (i).

    Việc ghép hai viên thạch (i) và (j) sẽ coi như mang viên thạch (i) qua bàn (j) hoặc ngược lại.

    Như vậy , việc ghép thạch ở hai vị trí (i) và (j) sẽ được xem như ghép tất cả các viên thạch trong đoạn [i,j] rồi đưa về cái bàn vị trí (i) hoặc (j). Qui tắc ghép vẫn áp dụng như đề bài ( ghép được hai thạch liền kề ).

    Việc cần làm là tìm đáp án tối ưu, giải bằng quy hoạch động. Và khi đó :

    • dp[i][j] là đáp án ghép tối ưu cho viên thạch từ (i) đến (j).

    • dp[i][i] = 0

    • dp[i][i+1] = ar[i] + ar[i+1] , giá trị mặc định để ghép hai viên thạch kề nhau

    • dp[i][j] = min(dp[i][k] + dp[k+1][j]) + cost , công thức truy hồi phụ thuộc vào bài toán con nhỏ hơn, lí do là (i->k) và (k+1->j) vì đây sẽ giải quyết bài toán cơ sở (2 phần tử).

    • cost = sum(ar[i] -> ar[j]), dồn từ (i) -> (j) thì lần dồn cuối chắc chắn mất sum(ar[i] -> ar[j]).

    • Đáp án là dp[0][N-1].