http://codeforces.com/contest/652/problem/C
给n个数的数组, n=3e5
给m个敌对pair,
让你选区间,一个合法区间不能包括任何一对敌对pair
首先我们把a[i]和i映射起来先。。
初始化一个end【】数组为0,记录的是第i个数为右端点的最大的敌对左端点(如果end[i]=0表示不存在一个敌对左端点)
然后遍历m对pair,对于每一对pair,找到在原始数组的位置,称为x,y
那么我们就更新 end[y]=x,如果end[y]已存在,则选一个大的 (x大可以包含小较小的情况)
预处理完之后。。
遍历i=1到n,设立flag=1;
如果【end[y]>=flag,则flag更新为end[i]+1,表示当前可以选的最远的左端点,因为如果选到flag左边的点,那么就会选到之前的敌对pair】
对每个位置,如果end[i]为0,则表示这个点没有敌对左端点,
则以i为右端点的合法区间,可以选 i-flag+1个点作为左端点,ans+=i-flag+1
#include <cstdio>
#include <cmath>
#include <cstring>
#include <string>
#include <algorithm>
#include <queue>
#include <map>
#include <set>
#include <vector>
#include <iostream>
using namespace std;
const double pi=acos(-1.0);
double eps=0.000001;
int END[3*100005];
int tm[3*100005];
int max(int a,int b)
{return a>b?a:b;}
int main()
{
int x,y,i;
int m;
int n;cin>>n>>m;
for (i=1;i<=n;i++)
{
scanf("%d",&x);
tm[x]=i;
}
__int64 ans=0;
for (i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
x=tm[x];
y=tm[y];
if (y<x) swap(x,y);
if (!END[y]) END[y]=x;
else END[y]=max(END[y],x);
}
int st=1;
for (i=1;i<=n;i++)
{
if (END[i]&&END[i]>=st)
st=END[i]+1;
ans+=i-st+1;
}
printf("%I64d\n",ans);
printf("\n");
return 0;
}