题目的意思是对于一个n位数,删除m个位后,得到的最小数是什么,比如12345 2,删除两个位,得到最小的就是123.实际上这题目解法很多,好像有贪心,线段树,RMQ等,因为我最近在学习RMQ,所以就用RMQ了。
这题目用了一个鸽巢原理,得到的m-n位数的第一位,必然出现在1~m-n+1,这个由鸽巢原理就十分明显了(如果1~n-(m-n)+1都没有的话,剩下的m-n-1个位是不可能凑出m-n个位的数的!)这样我们就可以从[1,n-(m-n)+1]中作RMQ取得最小值下标i,之后对于i+1后,m-n-1求第二个最小,依次递推即可。
我的RMQ写得很挫,唯有参考大神的代码了!
╮(╯▽╰)╭,身上有的东西基本都是从别人那里学来的,因为原创是很困难的。
/***********************************************************
> OS : Linux 3.2.0-60-generic #91-Ubuntu
> Author : yaolong
> Mail : dengyaolong@yeah.net
> Time : 2014年06月06日 星期五 07:19:57
**********************************************************/
#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<cmath>
using namespace std;
int mmin[1234][20];
char str[1234];
char ans[1234];
int min(int i,int j){
return str[i]<=str[j]?i:j;
}
void rmq_st(int n){
for(int i=0;i<=n;i++){
mmin[i][0]=i;
}
int endj=(int)(log(n+0.0)/log(2.0));
for(int j=1;j<=endj;j++){
int endi=n+1-(1<<j);
for(int i=0;i<endi;i++){
mmin[i][j]=min(mmin[i][j-1],mmin[i+(1<<(j-1))][j-1]);
}
}
}
int query_min(int l,int r){
int k=(int)(log(r-l+1.0)/log(2.0));
return min(mmin[l][k],mmin[r-(1<<k)+1][k]);
}
int main(){
int a;
while(scanf("%s %d",str,&a)){
int len=strlen(str);
rmq_st(len);
int need=len-a;
int k=0,i=0;
while(need--){
k =query_min(k,len-need-1);//取得下标
ans[i]=str[k];
i++,k++;
}
for( k=0;k<i;k++){//过滤0
if(ans[k]>'0')
break;
}
if(i==k){
puts("0");
}else{
for(;k<i;k++){
printf("%c",ans[k]);
}
puts("");
}
}
return 0;
}
HDU3183(RMQ+鸽巢原理),布布扣,bubuko.com
原文地址:http://blog.csdn.net/dengyaolongacmblog/article/details/28852007