zoukankan      html  css  js  c++  java
  • Lucky Array Codeforces

    http://codeforces.com/contest/121/problem/E

    话说这题貌似暴力可A啊。。。

    正解是想出来了,结果重构代码,调了不知道多久才A

    错误记录:

    1.线段树搞混num(节点编号)和l(区间端点)

    2.之前的dfs没有分离,写的非常混乱,迫不得已重构代码

      1 #include<cstdio>
      2 #include<algorithm>
      3 #include<cstring>
      4 #include<vector>
      5 using namespace std;
      6 #define fi first
      7 #define se second
      8 #define mp make_pair
      9 #define pb push_back
     10 typedef long long ll;
     11 typedef unsigned long long ull;
     12 typedef pair<int,int> pii;
     13 #define N 100000
     14 int data[80]={4,7,44,47,74,77,444,447,474,477,744,747,774,777,
     15 4444,4447,4474,4477,4744,4747,4774,4777,7444,7447,7474,7477,
     16 7744,7747,7774,7777};
     17 bool ok[10010];
     18 int sd=30;
     19 int n,m;
     20 int d[N+100];
     21 vector<pii> qq;
     22 //d[i]表示满足data[j]>(i位置的实际值)的最小j
     23 //线段树中查询出的位置的值表示data[d[i]]-(i位置的实际值)
     24 namespace S
     25 {
     26 #define mid (l+((r-l)>>1))
     27 #define lc (num<<1)
     28 #define rc (num<<1|1)
     29 int minn[4*N],sum[4*N],delv[4*N];
     30 void pd(int num)
     31 {
     32     minn[lc]-=delv[num];minn[rc]-=delv[num];
     33     delv[lc]+=delv[num];delv[rc]+=delv[num];
     34     delv[num]=0;
     35 }
     36 void delx(int L,int R,int x,int l,int r,int num)
     37 {
     38     if(L<=l&&r<=R)
     39     {
     40         minn[num]-=x;delv[num]+=x;
     41         return;
     42     }
     43     pd(num);
     44     if(L<=mid)    delx(L,R,x,l,mid,lc);
     45     if(mid<R)    delx(L,R,x,mid+1,r,rc);
     46     minn[num]=min(minn[lc],minn[rc]);
     47 }
     48 void setx(int L,int x,int l,int r,int num)
     49 {
     50     if(l==r)    {minn[num]=x;return;}
     51     pd(num);
     52     if(L<=mid)    setx(L,x,l,mid,lc);
     53     else    setx(L,x,mid+1,r,rc);
     54     minn[num]=min(minn[lc],minn[rc]);
     55 }
     56 void setsum(int L,int x,int l,int r,int num)
     57 {
     58     if(l==r)    {sum[num]=x;return;}
     59     if(L<=mid)    setsum(L,x,l,mid,lc);
     60     else    setsum(L,x,mid+1,r,rc);
     61     sum[num]=sum[lc]+sum[rc];
     62 }
     63 void update(int l,int r,int num)
     64 {
     65     if(minn[num]>0)    return;
     66     if(l==r)    {qq.pb(mp(l,data[d[l]]-minn[num]));return;}//minn[l]->minn[num]
     67     pd(num);
     68     update(l,mid,lc);update(mid+1,r,rc);
     69     minn[num]=min(minn[lc],minn[rc]);
     70 }
     71 int gsum(int L,int R,int l,int r,int num)
     72 {
     73     if(L<=l&&r<=R)    return sum[num];
     74     int ans=0;
     75     if(L<=mid)    ans+=gsum(L,R,l,mid,lc);
     76     if(mid<R)    ans+=gsum(L,R,mid+1,r,rc);
     77     return ans;
     78 }
     79 void work()
     80 {
     81     for(auto pp:qq)
     82     {
     83         int &l=pp.fi,&d=pp.se;
     84         int p=upper_bound(data,data+sd,d)-data;
     85         setx(l,data[p]-d,1,n,1);
     86         setsum(l,ok[d],1,n,1);
     87         ::d[l]=p;
     88     }
     89     qq.clear();
     90 }
     91 }
     92 char tmp[10];
     93 int main()
     94 {
     95     int i,t,L,R,x;
     96     for(i=0;i<sd;i++)    ok[data[i]]=1;
     97     for(i=0;i<sd;i++)    data[i+sd]=data[i]+1;
     98     sd*=2;sort(data,data+sd);
     99     data[sd++]=12000;
    100     scanf("%d%d",&n,&m);
    101     for(i=1;i<=n;i++)    scanf("%d",&t),qq.pb(mp(i,t));
    102     S::work();
    103     while(m--)
    104     {
    105         scanf("%s",tmp);
    106         if(tmp[0]=='c')
    107         {
    108             scanf("%d%d",&L,&R);S::update(1,n,1);S::work();
    109             printf("%d
    ",S::gsum(L,R,1,n,1));
    110         }
    111         else
    112         {
    113             scanf("%d%d%d",&L,&R,&x);
    114             S::delx(L,R,x,1,n,1);
    115         }
    116     }
    117     return 0;
    118 }

    https://codeforces.com/problemset/problem/679/E

    跟上面那道类似,可以发现涉及到42的幂很少(显然出题人不是想让我们写高精)

    因此开一棵线段树维护每个位置到下一个“关键点”(是否是42的幂的属性变化的点)的距离即可

    对于区间修改,就用set维护一下相同的区间,一段相同的放在一起处理

    大概是这样吧,可能还有一些细节。。

  • 相关阅读:
    获取非行间样式
    获取非行间样式
    prompt 方法显示输入对话框
    comfirm 方法显示对话框
    移动端页面常见问题及解决方案
    原生js怎样获取后台端口数据
    canvas描绘渐变的矩形
    cookie 的增加,销毁,读取
    canvas 绘制图形
    数组的排序,去重复
  • 原文地址:https://www.cnblogs.com/hehe54321/p/9321479.html
Copyright © 2011-2022 走看看