5.16周测

A. 01 分界最小翻转

题目描述

给定一个仅由字符 '0' 和 '1' 组成的字符串,你可以对字符串中的任意一个字符执行任意次翻转操作:将 '0' 变为 '1',或将 '1' 变为 '0'。每次翻转计为 1 次变换。

要求通过最少的翻转次数,使得字符串满足所有的 '0' 都严格出现在所有的 '1' 的前面。允许最终字符串中不存在 '0'(全 1),也允许不存在 '1'(全 0)。请计算并输出这个最少翻转次数。

输入格式

输入包含多组测试数据,每组数据占一行,为一个仅由 '0' 和 '1' 组成的非空字符串。

输入数据直到文件结束(EOF)为止。

输出格式

对于每组测试数据,输出一行一个整数,表示满足要求的最少翻转次数。

数据范围

单个字符串长度 n:1≤n≤1051≤n≤10^51≤n≤105

所有测试数据的字符串总长度不超过 10610^6106

样例输入输出

  • 输入样例

0101

1100

10

01

0000

1111

  • 输出样例

1

2

1

0

0

0

  • 样例解释

输入 "0101":最优方案为翻转第 3 位的 '0' 为 '1'(得到 "0111"),或翻转第 2 位的 '1' 为 '0'(得到 "0001"),均只需 1 次翻转。

输入 "1100":最优方案为翻转两个 '0' 为 '1'(全 1)或翻转两个 '1' 为 '0'(全 0),均需 2 次翻转。

输入 "10":必须翻转其中一个字符,最少 1 次。

输入 "01":本身已满足要求,无需翻转。

全 0 或全 1 字符串:天然符合要求,翻转次数为 0。


思路:枚举 + 统计前缀

n为字符串s的长度

定义以边界为i翻转的字符串:
s中第0个字符到第i个字符为0,第i+1个字符到第n个字符为1

特别的:

  • i = 0:以s最左边(第一个字符s[0]前面)为翻转边界,最终翻转为全为1的字符串
  • i = n:以s[n - 1]为翻转边界,最终翻转为全为0的字符串

total_ones表示s中1的总数
cur_ones:表示第1个字符到第i个字符1的总数

定义变量min_flips = n - total_ones:初始化为以i = 0为边界所需的翻转次数

i为边界翻转的所需的次数:
flips = 前i个字符中1的个数 + 后n-i个字符中0的个数
flips = cur_ones + n - total_ones - (i - cur_ones)

  • 第一次遍历,统计total_ones
  • 第二次遍历,枚举所有边界,取最小值

C++

cpp 复制代码
#include <iostream>
#include <string>
#include <algorithm>
using namespace std;

int main()
{
    //优化
    ios::sync_with_stdio(false);
    cin.tie(nullptr);

    string s;
    while (cin >> s)
    {
        int n = s.size();
        //定义分界点i:第0个字符(不存在)到第i个字符为0,第i+1个字符到最后一个字符为1
        //i = 0:全为1的字符串
        //i = n:全为0的字符串

        int total_ones = 0;
        //首次遍历:统计1的个数
        for (auto ch : s)
            if (ch == '1')
                total_ones++;
        
        int min_flips = n - total_ones;//初始化为i=0的情况

        int cur_ones = 0;//第0个字符到第i个字符的1的总数
        //再次遍历:枚举边界,求最小翻转次数
        for (int i = 1; i <= n; ++i)
        {
            //统计第0个字符到第i个字符1的总数
            if (s[i - 1] == '1')
                cur_ones++;
            //套用公式
            int flips = cur_ones + n - total_ones - (i - cur_ones);
            min_flips = min(min_flips, flips);   
        }
        cout << min_flips << '\n';//性能优化:读到EOF再刷新缓冲区
    }
}
相关推荐
凡人叶枫3 小时前
Effective C++ 条款30:透彻了解 inlining 的里里外外
linux·开发语言·c++·嵌入式开发·effective c++
noipp3 小时前
推荐题目:洛谷 P10907 [蓝桥杯 2024 国 B] 蚂蚁开会
c语言·c++·算法·编程·洛谷
学逆向的4 小时前
C++纯虚函数
开发语言·c++·网络安全
凡人叶枫5 小时前
Effective C++ 条款22:将成员变量声明为 private
linux·开发语言·c++
坚果派·白晓明6 小时前
【鸿蒙PC】SDL3 移植:AtomCode Skills 4 步速通多媒体库适配
c++·华为·ai编程·harmonyos·atomcode·c/c++三方库
赴生-7 小时前
C++进阶 C++11(下)
开发语言·c++
有点。7 小时前
C++(贪心算法一)
c++·贪心算法
WBluuue7 小时前
数据结构与算法:有序表(二):跳表
数据结构·c++·算法·skiplist
赴生-8 小时前
C++进阶 异常
开发语言·c++
凡人叶枫9 小时前
Effective C++ 条款28:避免使用 handles 指向对象内部
linux·服务器·开发语言·c++·嵌入式开发