(牛客周赛 9)C.小美的01串翻转

题目:

 样例:

输入
10001

输出
8

 思路:

        这里是连续的找子串,权值的意思是  我们取反操作了多少次,

我们有假设长度是 5 ,字符串是 10001

那么相邻不一样的字符串有两种情况   01010  或者  10101,那么它的权值分别是 4 和 1

又因为,我们是找到该字符串的所有子串之和是多少,所以在这里,我们应该知道子串是连续的,

并且我们是一步一步进行取反01 和 10 的情况,记录这样操作的次数,取最少操作数累加即可。

代码详解如此下:

#include <iostream>
#include <unordered_map>
#define endl '\n'
#define YES puts("YES")
#define NO puts("NO")
#define umap unordered_map
#pragma GCC optimize(3,"Ofast","inline")
#define ___G std::ios::sync_with_stdio(false),cin.tie(0), cout.tie(0)
using namespace std;
const int N = 2e6 + 10;

inline void solve()
{
	string s;
	getline(cin, s);
	int ans = 0;
	int n = s.size();
	for (int i = 0; i < n; ++i)
	{
		int res[2] = {0, 0};

		//	 从这里 j == i 开始就是也计算上了子串
		for (int j = i; j < n; ++j)
		{
			// 如果符合 11 或者 00 说明需要操作 一次,
			// 或者 01 的时候我们记录的是强制变成 10 情况,
			// 或者又将 10 强制变成 01的情况
			if ((j % 2 == 0 && s[j] == '1') || (j % 2 == 1 && s[j] == '0'))
			{
				res[0]++;
			}
			else
				res[1]++;

			// 累加我们所有子串的权值
			ans += min(res[0], res[1]);
		}
	}

	cout << ans << endl;
}


int main()
{
//	freopen("a.txt", "r", stdin);
	___G;
	int _t = 1;
//	cin >> _t;
	while (_t--)
	{
		solve();
	}

	return 0;
}

最后提交:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值