CF-Round#626-div2-C题

CF-Round#626-div2-C题

C. Unusual Competitions

传送门

这道题是div2的C题也是div1的A题哈。

先开始我以为是个简单的括号序列匹配。。后来仔细看题目发现我理解错了题意。是说我怎么样例输出都对不上了(盲目自信)
题目是问你对于没有完全匹配的括号序列,怎样花费最少的代价获得一个匹配的序列。
很明显如果当前左括号的个数不等于右括号的个数的话是无法成立的。直接输出-1就可以啦。
然后我的代码部分首先还特判了一下,就是n为odd奇数的话,那肯定输出-1
这算一个贪心把。
模拟栈。如果左括号可以与栈顶是右括号的匹配,直接将这对括号消掉。
如果无法匹配。那就开始记录当前需要交换的下标index。然后等到左括号的数目等于右括号的数目的时候,就可以进行重排序了。排序成配对的括号序列。花费的代价为r- l + 1。然后维护ans。就行。
每次重排序完毕就将记录左右括号个数的值置0,然后记录当前需要排序的开始位置的值置-1.(代表本次排序已结束,获得了符合要求的括号序列)。再开始新一轮的排序。
因为题目要求是最小的代价。所以隔得越近代价就越小嘛。简单的贪心策略。

我的代码可能if else判断的有点多。。。不好看。。
代码部分:

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

int n;
stack<char> ss;
string s;
int ans;

int main()
{
	cin >> n;
	cin >> s;
	int left = 0;
	int right = 0;
	if (n & 1)
	{
		cout << "-1\n";
	}
	else
	{
		for (int i = 0; i < n; i++)
		{
			if (s[i] == '(')
			{
				left++;
			}
			else
			{
				right++;
			}
		}
		if (left != right)
		{
			cout << "-1\n";
		}
		else
		{
			left = right = 0;
			int mark = -1;
			for (int i = 0; i < n; i++)
			{
				if (s[i] == '(')
				{
					left++;
				}
				else
				{
					if (left)
					{
						left--;
					}
					else
					{
						right++;
						if (mark == -1)
						{
							mark = i;
						}
					}
				}
				if (left == right && left)
				{
					ans += (i - mark + 1);
					left = right = 0;
					mark = -1;
				}
			}
			cout << ans << endl;
		}
	}
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

娃娃酱斯密酱

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值