• AtCoder Grand Contest 032D


    AtCoder Grand Contest 032D - Rotation Sort

    题目大意

    • 给出一个 N N N的排列,每次可以花费 A A A的代价把任意区间向左循环转一格,或花费 B B B的代价向右循环转一格,求排序的最小代价。
    • N ≤ 5000 N≤5000 N5000

    题解

    • 首先可以把操作简化一下,实质上就是花费 B B B的代价把某个数向左插入到任意位置,或花费 A A A的代价把某个数向右插入到任意位置。如果能想到这里,剩下的就更加自然了。
    • 接着会发现,每个数最多被操作一次,且不操作的数必须构成单调上升子序列。
    • 那么可以设 f i f_i fi为执行到第 i i i个数且不操作的最小代价,枚举 j j j转移,需满足 a j < a i a_j<a_i aj<ai ∀ x ∈ ( j , i ) a x ∉ [ a j , a i ] forall xin(j,i) a_x otin[a_j,a_i] x(j,i)ax/[aj,ai],则再枚举中间每个数计算转移的代价。
    • 为了方便,可以在排列最后加上一位正无穷。
    • 这样是 O ( N 3 ) O(N^3) O(N3)的,但最后一维枚举可以省去,复杂度降为 O ( N 2 ) O(N^2) O(N2)。(其实 O ( N 3 ) O(N^3) O(N3)也可以过)

    代码

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    #define N 5010
    #define ll long long
    ll f[N];
    int a[N];
    int main() {
    	int n, L, R, i, j;
    	scanf("%d%d%d", &n, &R, &L);
    	for(i = 1; i <= n; i++) scanf("%d", &a[i]);
    	a[n + 1] = 2e9;
    	f[0] = 0;
    	for(i = 1; i <= n + 1; i++) {
    		f[i] = 1e16;
    		ll s = 0; int mx = -1;
    		for(j = i - 1; j >= 0; j--) {
    			if(a[j] < a[i] && a[j] > mx) f[i] = min(f[i], f[j] + s);
    			if(a[j] > a[i]) s += R; else s += L, mx = max(a[j], mx);
    		}
    	}
    	printf("%lld
    ", f[n + 1]);
    	return 0;
    }
    
    哈哈哈哈哈哈哈哈哈哈
  • 相关阅读:
    linux系统cpu和内存占用率
    虚拟机网卡设置
    C语言中打印返回值
    MQTT_DEMO
    MQTT-C-UDP_PUB
    MQTT-C-PUB
    结构体指针用法
    linux系统如何操作隐藏文件
    mqtt学习笔记
    XML文件的读取----cElementTree
  • 原文地址:https://www.cnblogs.com/LZA119/p/13910014.html
Copyright © 2020-2023  润新知