Phục vụ (DHBB CT)

Xem PDF

Điểm: 500 (p) Thời gian: 1.0s Bộ nhớ: 1023M Input: bàn phím Output: màn hình

Một công ty cung cấp dịch vụ cho các đối tác của mình đặt tại n vùng khác nhau được đánh số \(1, 2, 3, …, n\). Công ty có \(3\) nhân viên phục vụ lưu động. Nếu xuất hiện một yêu cầu tại một địa điểm mà hiện đang không có nhân viên đang ở đó, một trong ba nhân viên di chuyển từ vị trí hiện tại của anh ta đến trực tiếp địa điểm xuất hiện yêu cầu mà không qua bất kỳ một địa điểm trung gian nào khác. Tại mọi thời điểm, chỉ có một nhân viên di chuyển. Các nhân viên chỉ di chuyển khi có yêu cầu phục vụ và không có hai nhân viên nào ở cùng một vị trí tại bất kỳ thời điểm. Chi phí để di chuyển từ vị trí \(i\) đến vị trí \(j\)\(C_{ij}\). Chú ý rằng hàm chi phí không nhất thiết phải là đối xứng, tuy nhiên chi phí khi không di chuyển luôn bằng \(0(Cii=0)\). Các yêu cầu phải được thực hiện theo thứ tự xuất hiện (yêu cầu xuất hiện trước phải được phục vụ trước, phục vụ xong một yêu cầu mới phục vụ yêu cầu tiếp theo).

Yêu cầu: Hãy tìm lịch di chuyển các nhân viên phục vụ yêu cầu sao cho tổng chi phí là nhỏ nhất.

Input

  • Dòng đầu tiên chứa hai số nguyên dương \(n,m,\) lần lượt là số vùng khác nhau và số yêu cầu cần phục vụ \((3 \leq n \leq 200,1 \leq m \leq 1000)\).
  • n dòng tiếp theo, mỗi dòng chứa \(n\) số nguyên không âm có giá trị không vượt quá \(2000\). Số thứ \(j\) của dòng thứ \(i\) là giá trị của \(C_{ij}\) - chi phí để đi từ \(i\) đến \(j\).
  • Dòng cuối cùng chứa \(m\) số nguyên là danh sách các yêu cầu phục vụ theo thứ tự đăng ký. Mỗi yêu cầu được mô tả bằng một số nguyên - số hiệu địa điểm, nơi yêu cầu xảy ra. Ban đầu ba nhân viên phục vụ đang ở các địa điểm \(1, 2\)\(3\).

Output

  • Ghi ra một số nguyên \(S\) là tổng chi phí nhỏ nhất tìm được.

Scoring

  • Subtask \(1\) (\(24\%\) số điểm): \(3 \leq n , m \leq 8\)
  • Subtask \(2\) (\(20\%\) số điểm): \(8 < n , m \leq 50\)
  • Subtask \(3\) (\(28\%\) số điểm): \(50 < n \leq 200, 50 < m \leq 1000\)

Example

Test 1

Input
5 9
0 1 1 1 1
1 0 2 3 2
1 1 0 4 1
2 1 5 0 1
4 2 3 4 0
4 2 4 1 5 4 3 2 1 
Output
5
Note

Một phương án tối ưu là \((1,2,1,2,2,1,3,1,3)\). Ở đây số thứ \(i\) là số hiệu của nhân viên phục vụ yêu cầu thứ \(i\).


Bình luận

  • SPyofgame 4:32 p.m. 13 Tháng 6, 2020 chỉnh sửa 2

    Spoiler Alert


    Mình sẽ viết editorial kĩ hơn sau khi mình rảnh ;-;

    Hint 1

    • Quy hoạch động

    Thử các trường hợp \(a\) -> \(serv_i\) | \(b\) -> \(serv_i\) | \(c\) -> \(serv_i\) với \(a \neq b \neq c\)

    Hint 2

    • Giảm chiều quy hoạch động

    Nhận xét: Trong 3 người phục vụ có 1 người ở vị trí \(serv_{i-1}\)

    Hint 3

    • Giảm chiều quy hoạch động

    Nhận thấy mình chỉ quan tâm \(dp_i\)\(dp_{i - 1}\) nên mình có thể giảm từ \(m\) chiều vùng nhớ xuống \(2\) chiều vùng nhớ


    Reference AC code | \(O(n ^ 3)\) time | \(O(2 * n^2)\) auxiliary space | Online Solving, DP_count

    C++
    int main()
    {
        /// Inout
        int n = readInt();
        int m = readInt();
    
        /// cost[i][j] is price to move from (i) -> (j)
        vector<vi> cost(n + 2, vi(n + 2, 0));
        for (int i = 1; i <= n; ++i)
            for (int j = 1; j <= n; ++j)
                getInt(cost[i][j]);
    
        /// 3-dimentional dp vector
        vector<vector<vi> > f(2, vector<vi>(n + 1, vi(n + 1, +INF)));
        f[0][2][3] = f[0][3][2] = 0; /// base case
    
        /// Solving problem
        int serv = 1;
        for (int t = 1; t <= m; ++t)
        {
            bool cur = t & 1;
            bool pre = !cur;
    
            int c = serv;
            getInt(serv); /// Online input
            f[cur].assign(n + 1, vi(n + 1, +INF)); /// Reset for getting min
            for (int a = 1; a <= n; ++a)
            {
                if (a == c) continue; /// statement conditional
                for (int b = 1; b <= n; ++b)
                {
                    if (b == c || b == a) continue; /// statement conditional
    
                    /// Finding min for each case: c -> serv | a -> serv | b -> serv
                    minimize(f[cur][a][b], f[pre][a][b] + cost[c][serv]);
                    minimize(f[cur][b][c], f[pre][a][b] + cost[a][serv]);
                    minimize(f[cur][c][a], f[pre][a][b] + cost[b][serv]);
                }
            }
        }
    
        /// Finding result
        int res = +INF;
        for (int a = 1; a <= n; ++a)
            for (int b = 1; b <= n; ++b)
                minimize(res, f[m & 1][a][b]);
    
        /// Output
        cout << res;
        return 0;
    }