动态规划:P5569 [SDOI2008] 石子合并 GarsiaWachs算法

P5569 [SDOI2008] 石子合并

 

 

 

 

    这题就是P1775石子合并的数据加强版,我们那题采用的是区间DP,时间复杂度为O(n3) (4*1e4)的三次方=1.6*1e13,显然超时。这里就必须用一个算法,叫做GarsiaWachs算法,可以降低时间复杂度至n2甚至nlogn.

    GarsiaWachs算法:他的做法就是对于一个序列a,先int ans=0,然后从左到右找到a[k-1]<a[k+1],将a[k]和a[k-1]合并成一个新的元素a[temp],从a[k-1]向左寻找,找到第一个比他大的,插入在他后面,ans+=a[temp],注意要把a[-1]和a[终点]定义为无穷大,方便寻找插入。然后再重复这个操作,直到序列中没有a[k-1]<a[k+1].原理我也不会推导,只知道这么个结论。

    这是我的代码:利用stl来操作,但是要开O2优化,否则只有70分,T3个点。

 1 //GarsiaWachs
 2 //开O2
 3 #include<iostream>
 4 #include<cmath>
 5 #include<algorithm>
 6 #include<vector>
 7 #include<climits>//INT_MAX的头文件
 8 using namespace std;
 9 int read()
10 {
11     int f = 1;
12     int ans = 0;
13     char c = getchar();
14     if (c > '9' || c < '0')
15     {
16         f = -1;
17         c = getchar();
18     }
19     while (c >= '0' && c <= '9')
20     {
21         ans = (ans << 1) + (ans << 3) + (c - '0');
22         c = getchar();
23     }
24     return f * ans;
25 }
26 int main()
27 {
28     int n;
29     n = read();
30     vector<int>v;
31     v.push_back(INT_MAX - 1);
32     for (int i = 1; i <= n; ++i)
33     {
34         v.push_back(read());
35     }
36     v.push_back(INT_MAX);
37     int k, j, temp, sum = 0;
38     while (n-- > 1)
39     {
40         for (k = 1; k <= n; ++k)
41         {
42             if (v[k - 1] < v[k + 1])
43                 break;
44         }
45         temp = v[k - 1] + v[k];
46         for (j = k - 1; j >= 0; --j)
47         {
48             if (v[j] > temp)
49                 break;
50         }
51         v.erase(v.begin() + k - 1);//删掉k-1;
52         v.erase(v.begin() + k - 1);//删掉k
53         v.insert(v.begin() + j + 1, temp);//插入和
54         sum += temp;
55     }
56     cout << sum;
57     return 0;
58 
59 }

 

没开O2:

 

 开O2:

 

 

最后补充一下:这个GarsiaWachs算法的标准代码模板,时间复杂度也特别低,高效通过,但是我看了很久也理解不了他的原理。

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 const int maxn=50005;
 4 
 5 int n;
 6 int a[maxn];
 7 int num;
 8 int ans;
 9 
10 void combine(int k){
11 
12     int temp=a[k]+a[k-1];
13     ans+=temp;
14     for (int i=k;i<num-1;i++){
15         a[i]=a[i+1];
16     }
17     num--;
18 
19     int j=0;
20     for (j=k-1;j>0&&a[j-1]<temp;j--){
21         a[j]=a[j-1];
22     }
23     a[j]=temp;
24 
25     while (j>=2&&a[j]>=a[j-2]){
26         int d=num-j;
27         combine(j-1);
28         j=num-d;
29     }
30 }
31 
32 int main()
33 {
34     cin >> n;
35     for (int i=0;i<n;i++){
36         cin >> a[i];
37     }
38     num=1;
39     ans=0;
40     for (int i=1;i<n;i++){
41         a[num++]=a[i];
42         while (num>=3&&a[num-3]<=a[num-1]){
43             combine(num-2);
44         }
45     }
46     while (num>1){
47         combine(num-1);
48     }
49 
50     cout << ans << endl;
51 }

 

 

 

 

    

posted @ 2022-04-17 16:53  朱朱成  阅读(56)  评论(0编辑  收藏  举报