https://codeforces.com/gym/104065/problem/G
题目大意:有n个数,每次操作把相邻的数中有比自身大的数删去,问要使剩余数字数量为1,至少需要多少次操作
1<=n<=1e5;1<=ai<=n
思路:因为每个数互不相同,对于相邻的两个数,肯定有一个数更小会被删去,所以每次操作至少删去一半的数,那么总的操作次数也就logn次,那么我们模拟的话,时间复杂度只要O(nlogn),所以我们用两个vector,然后每次遍历将不该删的数都放入另一个容器中即可
#include<iostream>
#include<cstdio>
#include<vector>
#include<list>
using namespace std;
const int N = 1e5 + 5;
vector<int>v1, v2;
int main()
{
int n;
cin >> n;
for (int i = 1; i <= n; i++)
{
int x;
scanf("%d", &x);
v1.push_back(x);
}
if (n == 1)
{//特判1个元素的情况
cout << 0;
return 0;
}
int ans = 0;
while (1)
{
for (int i = 0; i < v1.size(); i++)
{//将v1中不该删的放入v2
if (i == 0)
{//对于第一个元素,只需判断后一个数
if (v1[i] > v1[i + 1])
{
v2.push_back(v1[i]);
}
continue;
}
if (i == v1.size() - 1)
{//对于最后一个数,只需判断前一个数
if (v1[i] > v1[i - 1])
{
v2.push_back(v1[i]);
}
continue;
}
if (v1[i] > v1[i + 1] && v1[i] > v1[i - 1])
{//判断是否大于两边的数
v2.push_back(v1[i]);
}
}
ans++;//操作次数+
if (v2.size() == 1)
break;//如果删完只剩一个元素,退出循环
v1.clear();//将v1情况备用
for (int i = 0; i < v2.size(); i++)
{//将v2中不改删的放入v1
if (i == 0)
{
if (v2[i] > v2[i + 1])
{
v1.push_back(v2[i]);
}
continue;
}
if (i == v2.size() - 1)
{
if (v2[i] > v2[i - 1])
{
v1.push_back(v2[i]);
}
continue;
}
if (v2[i] > v2[i + 1] && v2[i] > v2[i - 1])
{
v1.push_back(v2[i]);
}
}
ans++;
if (v1.size() == 1)
break;
v2.clear();
}
cout << ans;
return 0;
}