<题目链接>

<转载于  >>> >

题目链接:

n个社**发传单,有a,b,c三个参数,*发的规则是,*发给序号为a,a+c....a+k*c,序号要求是小于等于b 这其中,有一个学生只收到了奇数传单,要求找出这个学生的编号与得到的传单数目 。

解题分析:

用二分来划分区间,如果左区间传单之和为奇数,则那个学生在左区间,否则在右区间,由于每个社*的区间内得到传单的学生为等差数列,所以可以很容易得到枚举的区间内的传单数。

 

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

const int M =20000+5;
typedef long long ll;
ll n,a[M],b[M],c[M];

ll solve(ll mid){     //求出以mid为末尾所发传单的前缀
    ll sum=0;
    for(int i=1;i<=n;i++){
        ll cal=min(mid,b[i]);
        if(cal>=a[i])
            sum+=(cal-a[i])/c[i]+1;
    }
    return sum;
}

int main(){
    while(scanf("%lld",&n)!=EOF){
        for(int i=1;i<=n;i++){
            scanf("%lld%lld%lld",&a[i],&b[i],&c[i]);
        }
        ll l=0,r=1ll<<31; //注意这里的ll不能省
        while(l<r){
            ll mid=(l+r)>>1;    //二分枚举右界
            if(solve(mid)%2)r=mid;    //如果左区间为奇数,则选择左区间
            else l=mid+1;   //若右区间为奇数,则将右界向右移动
        }
        if(l==1ll<<31){
            printf("DC Qiang is unhappy.\n");
        }
        else{
            printf("%lld %lld\n",l,solve(l)-solve(l-1));
        }
    }
    return 0;
}

 

下面是异或的做法,思路很清晰,但是不太明白为什么这个复杂度能过。

#include <cstdio>
#include <cstring>

const int M =20000+5;
typedef long long ll;
ll n,a[M],b[M],c[M];

int main(){
    while(scanf("%lld",&n)!=EOF){
        for(int i=1;i<=n;i++){
            scanf("%lld%lld%lld",&a[i],&b[i],&c[i]);
        }
        ll res=0;
        for(int i=1;i<=n;i++){   //看不懂这个循环的复杂度,数据如果稍微极限一点,肯定会tle吧
            for(ll j=a[i];j<=b[i];j+=c[i]){
                res^=j;    //因为只有一个学生有奇数个传单,根据异或的性质,偶数次异或结果仍然为0,奇数次异或结果仍然为j,所以如果res!=0,那么res就是该学生的下标
            }
        }
        if(!res)printf("DC Qiang is unhappy.\n");
        else{
            ll ans=0;
            for(int i=1;i<=n;i++){
                if(res>=a[i]&&res<=b[i]&&(res-a[i])%c[i]==0) //第i个社*给编号为res的学生贡献了一张传单
                    ans++;
            }
            printf("%lld %lld\n",res,ans);
        }
    }
    return 0;
}

 

 

2018-09-22