题目描述
描述:
一个集合有如下元素: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;
}