Codeforces Round #547 (Div. 3)D:Colored Boots(模拟)

博客给出一个链接,介绍了一个字符串配对问题。题目给出n和两个长度为n、由?和小写字母组成的字符串,相同小写字母可配对,?能和任意字母配对,需求最多配对组数并输出方案。解题思路为先字母配对,再用?配对,代码用vector加入位置信息方便删减,最后给出AC代码。

链接:http://codeforces.com/problemset/problem/1141/D

题意:给出一个n,接着给出两个长度为n的字符串,由?和小写字母组成,两个相同的小写字母可以

配成一副,?和任意的字母都可以配对,问这两个串互相配对,最多多少组,输出任意的配对方案;

因为?是任意都可以的,所以可以先枚举26个小写字母,将能通过字母配对的进行字母配对,最后再

消耗?来配对,若有剩余的?再进行配对;代码写的有点长,但都是ctrl c+ctrl v,四部分内容是一样的

加入位置信息用vector来实现,方便删减;

AC代码

#include<iostream>
#include<stdio.h>
#include<string>
#include<map>
#include<algorithm>
#include<set>
#include<vector>
using namespace std;
struct node{
	int cnt;
	vector<int>v;
}u[27], v[27];
set<pair<int, int>>s;
int main(){
	int n;
	string a, b;
	cin >> n >> a >> b;
	for (int i = 0; i < n; i++)   //存入字母和?的信息
	{
		if (a[i] == '?')
		{
			u[26].cnt++;
			u[26].v.push_back(i + 1);
		}
		else
		{
			u[a[i] - 'a'].cnt++;
			u[a[i] - 'a'].v.push_back(i + 1);
		}
	}
	for (int i = 0; i < n; i++)
	{
		if (b[i] == '?')
		{
			v[26].cnt++;
			v[26].v.push_back(i + 1);
		}
		else
		{
			v[b[i] - 'a'].cnt++;
			v[b[i] - 'a'].v.push_back(i + 1);
		}
	}
	int cnt = 0;
	for (char i = 'a'; i <= 'z'; i++)     //字母与字母的配对
	{   
		int num1 = u[i - 'a'].cnt, num2 = v[i - 'a'].cnt;
		int MIN = min(num1, num2);
		u[i - 'a'].cnt -= MIN;
		v[i - 'a'].cnt -= MIN;
		while (MIN--)
		{
			cnt++;
			s.insert({ u[i - 'a'].v.back(), v[i - 'a'].v.back() });
			u[i - 'a'].v.pop_back(), v[i - 'a'].v.pop_back();
		}
	}
	for (char i = 'a'; i <= 'z'; i++)   //字母与?的配对
	{
		int MIN = min(u[26].cnt, v[i - 'a'].cnt);
		u[26].cnt -= MIN;
		v[i - 'a'].cnt -= MIN;
		while (MIN--)
		{
			cnt++;
			s.insert({ u[26].v.back(), v[i - 'a'].v.back() });
			u[26].v.pop_back(), v[i - 'a'].v.pop_back();
		}
		MIN = min(v[26].cnt, u[i - 'a'].cnt);
		v[26].cnt -= MIN;
		u[i - 'a'].cnt -= MIN;
		while (MIN--)
		{
			cnt++;
			s.insert({ u[i - 'a'].v.back(), v[26].v.back() });
			u[i - 'a'].v.pop_back(), v[26].v.pop_back();
		}
	}
	int MIN = min(v[26].cnt, u[26].cnt);
	while (MIN--)    //?与?的配对
	{
		cnt++;
		s.insert({ u[26].v.back(), v[26].v.back() });
		u[26].v.pop_back(), v[26].v.pop_back();
	}
	cout << cnt << endl;
	for (auto i = s.begin(); i != s.end(); i++)
		cout << i->first << " " << i->second << endl;
	return 0;
}

 

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值