一、题目链接

  http://codeforces.com/problemset/problem/591/C

二、题意

  给定一个只含数字0和1的数组,通过如下方式,变成不再变化的01组合,最少需要操作几次。并输出最后得到的“稳定”01串。

  操作方式:数组开头和结尾两个数不变,对于不是开头和结尾的数字a[i],a[i] = (a[i - 1], a[i], a[i + 1])三个数的中位数。

  比如:1 0 1 0 1 0 1

  一次:1 1 0 1 0 1 1

  两次:1 1 1 1 1 1 1

  输出:2

     1 1 1 1 1 1 1

三、思路

  1、这题纯属找规律题,对于连续一排(两个及以上)的0或1,无论怎么变,都和原来一样。

  2、对于101010......这种组合:

    (1)如果长度为奇数,则通过若干次操作后可使这一段变为和开头元素一样。即开头和结尾是0,则通过若干次操作后这段序列变成000……000,如果开头和结尾是1,则通过若干次操作后这段序列变成111……111。

    (2)如果长度为偶数,如果头是0、尾是1,则通过若干次操作后这段变成前一半是0,后一半是1;如果头是1、尾是0,则通过若干次操作后这段变成前一半是1,后一半是0。

  3、对于操作次数的问题,可以打表找到规律如下。

    

  4、可以发现,当长度为2n + 1时(头尾是0和头尾是1是一样的),需要操作的次数为n。同理,当长度为2n时(头尾为0……1和头尾为1……0是一样的),需要操作的次数为n - 1。

  5、找到规律后,遍历一次数字串,记录满足01交叉的子串的起始和末尾下标。然后遍历所有满足01交叉的子串,使用上述规律,记录操作次数的最大值即为结果的第一部分。同时,对该子串使用上述规律做替换,然后输出整个序列即可。

四、正确性证明

  如果某段a[i], a[i + 1], ……,a[j]子串满足01交叉,len = j - i + 1:

  1、如果len为奇数,不妨假设为1010……101,则a[i - 1]和a[j + 1]必为1(不然还可以更长),因为串头和串尾是不变的,而且每操作一次,最靠近开头和结尾的0就会变成1,所以,这段子串必然要经过上述的规律次数才能变成“稳定”串(因为没人可以帮它,它也不能帮别人)。所以上述找规律的操作对于每一段满足01交叉的子串都是独立的,不会相互影响。同理,0101……010这样的子串道理也是一样的。

  2、如果len为偶数,不妨假设为1010……10,则a[i - 1]必为1,a[j + 1]必为0(不然还可以更长),然后证明过程和上述一样了。

五、源代码

  

#include<bits/stdc++.h>
using namespace std;
int n;
];

typedef pair<int, int> PII;
vector<PII> vec;
int main() {
#ifndef ONLINE_JUDGE
    freopen("input.txt", "r", stdin);
#endif // ONLINE_JUDGE
    int s, t;
    while(~scanf("%d", &n)) {
        vec.clear();
        ; i <= n; ++i)scanf("%d", a + i);
        ; i <= n; ++i) {
            ]) {
                ; i <= n && a[i] ^ a[i - ]; ++i);
                t = i;
                vec.push_back(make_pair(s, t));
            }
        }
        ;
        , sz = vec.size(); i < sz; ++i) {
            PII& p = vec[i];
            int len = p.second - p.first;
             == ) {
                ans = max(ans, (len - ) / );//len为奇数,len / 2 == (len - 1) / 2。
                for(int k = p.first; k < p.second; ++k)a[k] = a[p.first];
            } else {
                ans = max(ans, (len - ) / );
                ; ++k)a[k] = a[p.first];
                ; k < p.second; ++k)a[k] = a[p.second - ];
            }
        }
        printf("%d\n", ans);
        ;i <= n;++i)printf("%d%c", a[i], i < n ? ' ' : '\n');
    }
    ;
}

六、附:找规律程序源代码(注意体会二进制的思想)

#include<bits/stdc++.h>
using namespace std;

int bit(long long& a, int x) {
    ;
}

int main() {
    ; i <= ; i += ) {
        , b = ;
        ; j < i; j += )a |= 1LL << j;
        ;
         && a != (1LL << i) - ) {
            b = a;
            ; j < i - ; ++j) {
                ) == bit(a, j + ) && bit(a, j - ) == )b |= (1LL << j);
                ) == bit(a, j + ) && bit(a, j - ) == )b &= ~(1LL << j);
            }
            a = b;
            ++cnt;
        }
        printf("len = %d, cnt = %d\n", i, cnt);
    }
    ;
}

Codeforces-591C题解的更多相关文章

  1. codeforces#536题解

    CodeForces#536 A. Lunar New Year and Cross Counting Description: Lunar New Year is approaching, and ...

  2. 【22.70%】【codeforces 591C】 Median Smoothing

    time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  3. codeforces 1093 题解

    12.18 update:补充了 $ F $ 题的题解 A 题: 题目保证一定有解,就可以考虑用 $ 2 $ 和 $ 3 $ 来凑出这个数 $ n $ 如果 $ n $ 是偶数,我们用 $ n / 2 ...

  4. Codeforces Numbers 题解

    这题只需要会10转P进制就行了. PS:答案需要约分,可以直接用c++自带函数__gcd(x,y). 洛谷网址 Codeforces网址 Code(C++): #include<bits/std ...

  5. Codeforces 691E题解 DP+矩阵快速幂

    题面 传送门:http://codeforces.com/problemset/problem/691/E E. Xor-sequences time limit per test3 seconds ...

  6. Codeforces 833B 题解(DP+线段树)

    题面 传送门:http://codeforces.com/problemset/problem/833/B B. The Bakery time limit per test2.5 seconds m ...

  7. Codeforces 840C 题解(DP+组合数学)

    题面 传送门:http://codeforces.com/problemset/problem/840/C C. On the Bench time limit per test2 seconds m ...

  8. Codeforces 515C 题解(贪心+数论)(思维题)

    题面 传送门:http://codeforces.com/problemset/problem/515/C Drazil is playing a math game with Varda. Let’ ...

  9. Codeforces 475D 题解(二分查找+ST表)

    题面: 传送门:http://codeforces.com/problemset/problem/475/D Given a sequence of integers a1, -, an and q ...

  10. CodeForces CF875C题解

    题解 非常有意思的\(2-SAT\)的题. 听学长讲完之后感觉确实容易想到\(2-SAT\),顺理成章. 显然,对于两个串,对咱们来说有意义的显然是两个串中第一个不同的数字.那么,我们假设两个串分别是 ...

随机推荐

  1. 2017-2018-2 20165202 实验四《Android程序设计》实验报告

    一.实验报告封面 二.实验内容 1.基于Android Studio开发简单的Android应用并部署测试; 2.了解Android.组件.布局管理器的使用: 3.掌握Android中事件处理机制. ...

  2. Express 开发与部署最佳实践 -- 待续

    链接 nginx 代理缓存  压缩 等 全部采用异步 使用try catch  处理同步异常  promise 处理异步 异常,  而不是使用 domains  或者 uncaughtExceptio ...

  3. JSP和JS的区别

    从本科毕业设计开始就一直困扰我,jsp和js这两者的区别,一直处于迷糊状态,也没有搞清楚.今天就简单的介绍下两者的区别. 1.JSP全称是java server page    JS全称是javaSc ...

  4. FairyGUI编辑器制作Unity3D UI值得借鉴

    笔者介绍:姜雪伟,IT公司技术合伙人,IT高级讲师,CSDN社区专家,特邀编辑,畅销书作者,已出版书籍:<手把手教你架构3D游戏引擎>电子工业出版社和<Unity3D实战核心技术详解 ...

  5. iOS开发之如何应对苹果app的ipv6时代?

    WWDC2015苹果宣布在ios9支持纯IPv6的网络服务,并且要求2016年提交到app store的应用必须兼容纯IPv6的网络,要求适配的系统版本是ios9以上(包括ios9). 一 背景介绍 ...

  6. [Python] re正则表达式指南以及常用操作

    一.语法 1. 使用正则表达式进行匹配的流程 2. Python支持的正则表达式元字符和语法 参考: AstralWind的Python正则表达式指南 官方文档:7.2. re — Regular e ...

  7. socket创建UDP服务端和客户端

    UDP服务端代码示例: from socket import * #1.创建数据报套接字 sockfd = socket(AF_INET, SOCK_DGRAM) #2.绑定服务端地 sockfd.b ...

  8. 文件的copy

    def mycopy(src_filename, dst_filename): try: fr = open(src_filename, "rb") try: try: fw = ...

  9. .NET/C# 使窗口永不获得焦点

    有些窗口天生就是为了辅助其它程序而使用的,典型的如“输入法窗口”.这些窗口不希望抢夺其它窗口的焦点. 有 Win32 方法来解决这样的问题,WS_EX_NOACTIVATE 便是关键. 具体来说,是给 ...

  10. For input string: "null"

    java.lang.NumberFormatException: For input string: "null" 在开发中你是否遇到过这样的问题,不管请求到的值是什么都能进入不为 ...