HDU - 4747 Mex

HDU-4747

题意:对于每一段连续区间找到mex的值(mex 即为没出现过的数的最小非负数),然后对所有的改值进行求和, 输出ans。

题解:首先我门可以暴力的跑出 区间 [1,2], [1,3],……,[1,i]的mex值, 由于mex的定义, 我们可以得知, 左边的区间mex一定是不大于右边的区间的mex值, 然后可以通过线段树求和, 来访问这些值的和。 然后查询完一次数据之后, 我们将删除第一个数, 并且判断在[l+1,r-1](l为被删数出现的位置, r为下一个该数出现的位置) 的区间内是否有mex值大于该删除的值, 如果有就更新这一段区间的值, 用线段树加速更新操作,最后删完所有的点所统计的和就是答案了。

代码:

 

  1 #include<bits/stdc++.h>
  2 using namespace std;
  3 #define Fopen freopen("_in.txt","r",stdin); freopen("_out.txt","w",stdout);
  4 #define LL long long
  5 #define ULL unsigned LL
  6 #define fi first
  7 #define se second
  8 #define pb push_back
  9 #define lson l,m,rt<<1
 10 #define rson m+1,r,rt<<1|1
 11 #define max3(a,b,c) max(a,max(b,c))
 12 #define min3(a,b,c) min(a,min(b,c))
 13 typedef pair<int,int> pll;
 14 const int INF = 0x3f3f3f3f;
 15 const LL mod = 1e9+7;
 16 const int N = 2e5+10;
 17 LL Max[N<<2], sum[N<<2], lazy[N<<2];
 18 int a[N], mex[N];
 19 void PushUp(int rt){
 20     sum[rt] = sum[rt<<1] + sum[rt<<1|1];
 21     Max[rt] = max(Max[rt<<1|1], Max[rt<<1]);
 22 }
 23 void PushDown(int rt, int llen, int rlen){
 24     if(lazy[rt] != -1){
 25         lazy[rt<<1] = lazy[rt<<1|1] = lazy[rt];
 26         sum[rt<<1] = lazy[rt]*llen;
 27         sum[rt<<1|1] = lazy[rt]*rlen;
 28         Max[rt<<1] = Max[rt<<1|1] = lazy[rt];
 29         lazy[rt] = -1;
 30     }
 31 }
 32 void Build(int l, int r, int rt){
 33     lazy[rt] = -1;
 34     if(l == r) {
 35         Max[rt] = sum[rt] = mex[l];
 36         return ;
 37     }
 38     int m = l+r >> 1;
 39     Build(lson);
 40     Build(rson);
 41     PushUp(rt);
 42 }
 43 void Update(int l, int r, int rt, int L, int R, int c){
 44     if(L <= l && r <= R){
 45         Max[rt] = lazy[rt] = c;
 46         sum[rt] = 1ll * (r-l+1) * c;
 47         return ;
 48     }
 49     int m = l+r >> 1;
 50     PushDown(rt, m-l+1, r-m);
 51     if(L <= m) Update(lson, L, R, c);
 52     if(m < R) Update(rson, L, R, c);
 53     PushUp(rt);
 54 }
 55 LL Query(int l, int r, int rt, int L, int R){
 56     if(L <= l && r <= R)  return sum[rt];
 57     int m = l+r >> 1;
 58     LL ret = 0;
 59     PushDown(rt, m-l+1, r-m);
 60     if(L <= m) ret += Query(lson, L, R);
 61     if(m < R)  ret += Query(rson, L, R);
 62     return ret;
 63 }
 64 int id(int l, int r, int rt, int v){
 65     if(l == r) return l;
 66     int m = l+r >> 1;
 67     PushDown(rt, m-l+1, r-m);
 68     if(Max[rt<<1] > v) return id(lson, v);
 69     return id(rson, v);
 70 }
 71 map<int, int> mp;
 72 int Next[N];
 73 int main(){
 74     ///Fopen;
 75     int n;
 76     while(~scanf("%d", &n), n){
 77         for(int i = 1; i <= n; i++) scanf("%d", &a[i]);
 78         mp.clear();
 79         int tmp = 0;
 80         for(int i = 1; i <= n; i++){
 81             mp[a[i]] = 1;
 82             while(mp.count(tmp)) tmp++;
 83             mex[i] = tmp;
 84         }
 85 
 86         mp.clear();
 87         for(int i = n; i >= 1; i--){
 88             if(!mp.count(a[i])) Next[i] = n + 1;
 89             else Next[i] = mp[a[i]];
 90             mp[a[i]] = i;
 91         }
 92         Build(1,n,1);
 93         LL ans = 0;
 94         for(int i = 1; i <= n; i++){
 95             ans += sum[1];
 96             if(Max[1] > a[i]){
 97                 int l = id(1,n,1,a[i]);//找到最左边的大于被删除数的位置,由mex的定义可知,后面的值一定也不小于这个节点
 98                 int r = Next[i];
 99                 if(l < r) Update(1, n, 1, l, r-1, a[i]);
100             }
101             Update(1,n,1,i,i,0);
102         }
103         printf("%I64d\n", ans);
104     }
105     return 0;
106 }
HDU 4747

 

posted @ 2018-05-28 00:10  Schenker  阅读(250)  评论(0编辑  收藏  举报