zoukankan      html  css  js  c++  java
  • luogu_P3937 Changing

    https://www.luogu.org/problem/P3937

    题目描述

    nn盏灯环形排列,顺时针依次标号为1cdots n1n。初始时刻为00,初始时刻第ii盏灯的亮灭a_iai给定,00表示灭,11表示亮。下一时刻每盏灯的亮灭取决于当前时刻这盏灯与顺时针方向下一盏灯的亮灭。若两盏灯状态相同,则下一时刻该灯灭,否则该灯亮。

    试求时刻tt第kk盏灯的状态。

    输入格式

    第一行,三个整数,分别为n, t, kn,t,k。

    第二行,共nn个整数,分别为00或11,代表a_iai

    输出格式

    共一行,一个数,00或11,代表时刻tt第kk盏灯的状态。

    输入输出样例

    输入 #1
    4 2 1
    1 0 1 0
    输出 #1
    0

    说明/提示

    对于25\%25%的数据,有1leq t, kleq nleq 10001t,kn1000。

    对于60\%60%的数据,有1leq t, kleq nleq 10^51t,kn105。

    对于100\%100%的数据,有1leq t, kleq nleq 3*10^61t,kn3106。


    找了下规律,发现 N(log(N)/log(2)) 作法

    先假设 k=1 ,我们就看答案要的这一位

    0 时刻,ans = a1

    时刻,ans = a1^a2

    时刻,ans = a1^a2^a2^a3 = a1^a3

    时刻,ans = a1^a2^a3^a4

    时刻,ans = a1^a2^a2^a3^a3^a4^a4^a5 = a1^a5

    5 ......................................................................................

    ............................................................................................

    ...................................................................................................

    8 时刻,ans = ......................................................................................... = a1^a9


    这样 log(N)/log(2) 就出来了,事实上并不会T

    与倍增法求LCA有相似处

    #include<iostream>
    #include<cstdio>
    
    #define ri register int
    #define u int
    
    namespace opt {
    
        inline u in() {
            u x(0),f(1);
            char s(getchar());
            while(s<'0'||s>'9') {
                if(s=='-') f=-1;
                s=getchar();
            }
            while(s>='0'&&s<='9') {
                x=(x<<1)+(x<<3)+s-'0';
                s=getchar();
            }
            return x*f;
        }
    
    }
    
    using opt::in;
    
    #include<cmath>
    #include<algorithm>
    
    #define NN 3000005
    
    namespace mainstay {
    
        u N,T,K,a[NN<<1];
    
        inline void solve() {
            while(~scanf("%d%d%d",&N,&T,&K)) {
                for(ri i(1); i<=N; ++i) a[i]=a[i+N]=in();
                u ans(a[K]);
                while(T) {
                    u k(std::log(T)/std::log(2));
                    u p(std::pow(2,k));
                    ans^=a[K+p];
                    for(ri i(1); i<=N; ++i) a[i]^=a[i+p];
                    for(ri i(1); i<=N; ++i) a[i+N]=a[i];
                    T-=std::pow(2,k);
                }
                printf("%d
    ",ans);
            }
        }
    
    }
    
    int main() {
    
        //freopen("TLE.in","r",stdin);
        //freopen("TLE.out","w",stdout);
        std::ios::sync_with_stdio(false);
        mainstay::solve();
    
    }
  • 相关阅读:
    Linux系统管理之进程管理
    Linux交换空间swap讲解
    Linux物理存储结构以及磁盘划分
    Linux 文件系统挂载mount命令
    Linux文件系统df、du、fsck命令讲解
    Linux yum源配置以及yum命令讲解
    Linux rpm查询命令以及RPM包验证
    02Python常用模块_01math
    00_python常见用法笔记
    02_python闯关练习_02Max&Min
  • 原文地址:https://www.cnblogs.com/ling-zhi/p/11774108.html
Copyright © 2011-2022 走看看