CodeForces 122G Lucky Array(一脸懵逼的树状数组)

Petya loves lucky numbers. Everybody knows that lucky numbers are positive integers whose decimal representation contains only the lucky digits 4 and 7. For example, numbers 47, 744, 4 are lucky and 5, 17, 467 are not.

Petya has an array consisting of n numbers. He wants to perform m operations of two types:

add l r d — add an integer d to all elements whose indexes belong to the interval from l to r, inclusive (1 ≤ l ≤ r ≤ n, 1 ≤ d ≤ 104);
count l r — find and print on the screen how many lucky numbers there are among elements with indexes that belong to the interval from l to r inclusive (1 ≤ l ≤ r ≤ n). Each lucky number should be counted as many times as it appears in the interval.
Petya has a list of all operations. The operations are such that after all additions the array won't have numbers that would exceed 104. Help Petya write a program that would perform these operations.

Input
The first line contains two integers n and m (1 ≤ n, m ≤ 105) — the number of numbers in the array and the number of operations correspondingly. The second line contains n positive integers, none of which exceeds 104 — those are the array numbers. Next m lines contain operations, one per line. They correspond to the description given in the statement.

It is guaranteed that after all operations are fulfilled each number in the array will not exceed 104.

Output
For each operation of the second type print the single number on the single line — the number of lucky numbers in the corresponding interval.

Examples
input
3 6
2 3 4
count 1 3
count 1 2
add 1 3 2
count 1 3
add 2 3 3
count 1 3
output
1
0
1
1
input
4 5
4 4 4 4
count 1 4
add 1 4 3
count 1 4
add 2 3 40
count 1 4
output
4
4
4
Note
In the first sample after the first addition the array will look in the following manner:

4 5 6

After the second addition:

4 8 9

The second sample after the first addition:

7 7 7 7

After the second addition:

7 47 47 7

 

题意:

给出一个数列,两种操作

1、区间加

2、区间统计只由4和7组成的数字的个数。

保证每个数都不超过10000

 

题解:

这题没什么可以讲的,我就来neta一下我写这道题的心路历程吧

首先看到题意,emmm,10000看起来很可做啊……

过了一会,woc,什么鬼?!这玩意是可以区间修改的?

半个小时……

woc,我不会啊

一个小时……

抓狂ing

一个半小时

算了算了,破罐子破摔了,我写个树状数组点修改代替区间修改看看能不能卡过去吧……

一边写一边想

naive,DIV1的E是这么容易水过去的?

然后……

 

sorry,树状数组是真的可以为所欲为的

emmm,神题神题,rbq,rbq,不得了,不得了,告辞!

我裤子都脱了你就给我看这个?!

 

算了,发泄完了,代码如下:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;

int a[100010],f[100010],sum[100010],n,m;

void get(int now)
{
    if(now*10+7<=10000)
    {
        f[now*10+7]=1;
        get(now*10+7);
    }
    if(now*10+4<=10000)
    {
        f[now*10+4]=1;
        get(now*10+4);
    }
}

int lowbit(int x)
{
    return (-x)&x;
}

void add(int i,int val)
{
    while(i<=n)
    {
        sum[i]+=val;
        i+=lowbit(i);
    }
}

int count(int i)
{
    int ans=0;
    while(i>0)
    {
        ans+=sum[i];
        i-=lowbit(i);
    }
    return ans;
}

int main()
{
    scanf("%d%d",&n,&m);
    get(0);
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&a[i]);
        if(f[a[i]])
        {
            add(i,1);
        }
    }
    char c[10];
    int l,r,val;
    while(m--)
    {
        scanf("\n%s",c);
        if(c[0]=='c')
        {
            scanf("%d%d",&l,&r);
            printf("%d\n",count(r)-count(l-1));
        }
        else
        {
            scanf("%d%d%d",&l,&r,&val);
            for(int i=l;i<=r;i++)
            {
                if(f[a[i]])
                {
                    add(i,-1);
                }
                a[i]+=val;
                if(f[a[i]])
                {
                    add(i,1);
                }
            }
        }
    }
}

 

posted @ 2018-05-01 20:17  Styx-ferryman  阅读(554)  评论(0编辑  收藏  举报