[洛谷]P1323 删数问题 (#单调队列+优先队列)

31 篇文章 0 订阅
23 篇文章 0 订阅

题目描述

描述:

一个集合有如下元素:1是集合元素;若P是集合的元素,则2 * P +1,4*P+5也是集合的元素,

取出此集合中最小的K个元素,按从小到大的顺序组合成一个多位数,现要求从中删除M个数位上的数字,

使得剩下的数字最大,编程输出删除前和删除后的多位数字。

注:不存在所有数被删除的情况

输入输出格式

输入格式:

输入的仅一行,K,M的值,K,M均小于等于30000。

输出格式:

输出为两行,第一行为删除前的数字,第二行为删除后的数字。

输入输出样例

输入样例#1

5  4

输出样例#1

137915
95

说明

数据范围:

对于30%的数据,有1 ≤ K ,M≤ 300;

对于100%的数据,有1 ≤ K,M ≤ 30000。

这么水的题目,就不要打表了。


思路

水你mmp啊......

这个我看了题解,看到了一大堆用优先队列的..都那么神犇的吗。。还有用链表,to_string操作,太强啦,先%各位大佬。

//看了优先队列的题解,然后自己写了一个优先队列+单调队列的 
#include <stdio.h>
#include <iostream>
#include <queue>
using namespace std;
int n,m,s[300001],que[600001],position[300001],cnt;
//s取每一位上的数,que是单调队列,position是单调队列中每一位的位置 
struct lxydl
{
	int data;
	bool operator < (const lxydl a)const{return data>a.data;}//优先级 
};
lxydl a,b;
priority_queue<lxydl> p_que;//优先队列 
inline void xsyation(int a)//将分开的数放进s 
{
	int b=cnt+1,c;
	while(a)//注意这里是顺着搞的,也就是说数组是从高位到低位存 
	{
		s[++cnt]=a%10;
		a/=10;
	}
	c=cnt;
	while(b<c)//所以要把它反转 
	{
		swap(s[b++],s[c--]); 
	}
}
int main()
{
	ios::sync_with_stdio(false);
	cin.tie(0);
	cout.tie(0);
	register int i,j;
	cin>>n>>m;
	a.data=1;
	p_que.push(a);
	while(n--)
	{
		a=p_que.top();
		xsyation(a.data);
		cout<<a.data;
		p_que.pop();
		b.data=a.data*2+1; p_que.push(b);
		b.data=a.data*4+5; p_que.push(b); 
	}
	cout<<endl;
	int head(1),tail(0),l=m+1;
	for(i=1;i<=cnt;i++)//删数操作,可以用单调队列 
	{
		while(head<=tail && s[i]>que[tail]) tail--;
		que[++tail]=s[i];
		position[tail]=i;
		if(i-position[head]+1>l) head++;
		if(i>=l)
		{
			cout<<que[head];
			head++;
		}
	}
	cout<<endl;
	return 0;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值