(从未见过如此美妙且单纯的贪心题...)标签是贪心还就真只有贪心也是挺难得的hhh
其实题目难度完全达不到黄题的水平,整体思路非常清晰明了,就是尽量将代价大的线率先切割。然后就是需要简单分析一下每一刀之后,对接下来的横切和竖切的次数有何影响即可。
题目最大的恶心之处就是配的样例实在是水到了极点,没有任何借鉴价值qaq,所以这里附上一组其他的样例。
输入: 输出:
8 6 233
3 8 2 7 12 6 5
11 1 4 9 10
接下来呈上代码^_^,有可能有疑问的地方都附上了注释qaq,希望能帮助你更好理解~~~
Code:
/*实际上每一条线都需要切割到才能满足结果,能够贪心之处在于分割后的各个块不能一起切割,所以要在最开始分割时尽量分割代价大的块。
而需要切多少刀呢?横着每切一刀,下一次竖着切就需要多一刀,反之同样。故考虑将横竖整体排序,然后从大到小依次切割即可*/
#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
int n, m;
long long ans;
struct mat
{
long long cnt;
int op;//op=0为横着的,op=1为竖着的
}a[4005];
bool cmp(mat x, mat y)
{
return x.cnt > y.cnt;
}
int main(){
cin >> n >> m;
for (int i = 1; i <= n - 1; i++)
{
cin >> a[i].cnt;
a[i].op = 0;
}
for (int i = n; i < n + m - 1; i++)
{
cin >> a[i].cnt;
a[i].op = 1;
}
sort(a + 1, a + n + m - 1, cmp);//将横竖的代价整体排序
int xcnt = 1, ycnt = 1;//分别记录横着和竖着需要切多少刀
for (int i = 1; i < n + m - 1; i++)
{
if (!a[i].op)//如果当前切的一刀为横着的,以后竖着切就都需要多一刀
{
ycnt++;
ans += xcnt * a[i].cnt;
}
else//如果当前切的一刀为竖着的,以后横着切就都需要多一刀
{
xcnt++;
ans += ycnt * a[i].cnt;
}
}
cout << ans;
return 0;
}