原题目

A Magic Lamp

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 3964    Accepted Submission(s): 1605

Problem Description
Kiki likes traveling. One day she finds a magic lamp, unfortunately the genie in the lamp is not so kind. Kiki must answer a question, and then the genie will realize one of her dreams. 
The question is: give you an integer, you are allowed to delete exactly m digits. The left digits will form a new integer. You should make it minimum.
You are not allowed to change the order of the digits. Now can you help Kiki to realize her dream?
Input
There are several test cases.
Each test case will contain an integer you are given (which may at most contains 1000 digits.) and the integer m (if the integer contains n digits, m will not bigger then n). The given integer will not contain leading zero.
Output
For each case, output the minimum result you can get in one line.
If the result contains leading zero, ignore it. 
Sample Input
178543 4 1000001 1 100001 2 12345 2 54321 2
Sample Output
13 1 0 123 321

********************************************************************************************************************************************************************************************

题意:给你一个数字串 (每一个串有n个数字), 和一个m (要你删除m个数字 ), 使得剩下的数字组成的整数最小。

一开始想了一下删除m个,等价与在数字串中选择 n-m 个数字组成最小整数。

一开始一直做不出,贪心有点问题。后来被dalao教一下就懂了;

贪心的方法:

我们先假设 n 是串的个数, m 是要删除的个数, need 是需要选的个数;

1) 因为是按照顺序组合数字的, 所以第一个删除的数字一定在  [0, n-need] 之内;

证明: 假设数字有 a0, a1, a2, a3, a4, a5 个,要删除2个,n=6, m=2, need = 4;

因为是连续选择的,我们假设在 [0, 3]之内选择第一个,那么我们只剩下a4, a5了,无法构成4个;

所以假设最后的need-1个我们都要, 那么 从 0 ~ (n - 1) - ( need - 1 )==[0, n-need]之内第一个一定在里面(数组从0开始,所以 n-1);

2) 当在[0, n-need]之内选好了第一个,找到第一个的位置pre,那么下一个 数字一定在 [ pre + 1, n-need + 1 ], 就可以一直找下去了,记录你找到的数;

3) 最后在你找到的数里面讨论前导0的情况。

 1 #include <iostream>
 2 #include <cstring>
 3 #include <cstdio>
 4 #include <algorithm>
 5 #include <vector>
 6 #include <deque>
 7 using namespace std;
 8 const int maxn = 1000+10;
 9 int d[maxn][maxn];
10 int a[maxn];
11 void RMQ_init(string s)
12 {
13     memset(d, 0, sizeof(d));
14     memset(a, 0, sizeof(a));
15     int n = s.size();
16     for(int i=0;i< n ;i++)  a[i]=d[i][0]=s[i]-'0';
17     for(int j = 1; (1<<j) <= n ;j++)
18         for(int i = 0 ; i+(1<<j ) -1 < n ;i++)
19             d[i][j] = min(d[i][j-1] , d[i+(1<<(j-1))][j-1] );
20 }
21 int RMQ(int L, int R)
22 {
23     int k=0;
24     while((1<<(k+1)) <= R-L+1)  k++;
25     return min(d[L][k], d[R-(1<<k) + 1][k]);
26 }
27 int findmin(int L, int R, int Min)
28 {
29     for(int i=L; i <= R ;i++){
30         if(a[i] == Min) return i;
31     }
32 }
33 int main()
34 {
35     string s;
36     int m, n, need;
37     while(cin >> s >> m){
38     RMQ_init(s);
39     n = s.size() ;
40     need = n - m;
41     int Left=0, Right = n - need;
42 //    cout << Left << " " << Right << endl;
43 //    cout << RMQ(Left, Right) << endl;
44     deque <int > q;
45     while(1){
46         if(need ==0 )   break;
47         int Min = RMQ(Left, Right);
48 //        cout << Min << endl;
49         q.push_back(Min);
50         int pre = findmin(Left, Right, Min);
51         Left = pre+1;
52         Right ++ ;
53         need --;
54     }
55     while(!q.empty()){
56         if(q.front() == 0)  q.pop_front();
57         else break;
58     }
59     if(q.empty()){
60         cout << "0";
61     }
62     else{
63         while(!q.empty()){
64             cout << q.front();
65             q.pop_front();
66         }
67     }
68     cout << endl;
69     }
70     return 0;
71 }

转载于:https://www.cnblogs.com/denghaiquan/p/6666168.html

HDU 3183 A Magic Lamp(RMQ问题, ST算法)相关推荐

  1. hdu 3183 A Magic Lamp (rmq)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3183 #include <stdio.h> #include <stdlib.h&g ...

  2. hdu 3183 A Magic Lamp(给一个n位的数,从中删去m个数字,使得剩下的数字组成的数最小(顺序不能变),然后输出)...

    1.题目大意是,给你一个1000位的数,要你删掉m个为,求结果最小数. 思路:在n个位里面删除m个位.也就是找出n-m个位组成最小数 所以在区间 [0, m]里面找最小的数.相应的下标标号i 接着找区 ...

  3. HDU - 3183 A Magic Lamp 线段树

    题目链接 题意:数字字符串删除k个值后的数字最小不算前导0 思路:其实就是最小字典序,那么我们肯定一位一位的确定最小的值,那么对于i位,其实找i+1~i+k位中最小的数有没有比当前位置的数小,如果有的 ...

  4. RMQ的ST算法(区间最值)

    ST算法求解RMQ问题(区间最值) 效率:O(n log n)预处理,O(1)询问 思想: 用 f [ i ][ j ] 表示 以i 开头的区间,包括2^j 个元素的一段区间的最值 那么有初始化的初始 ...

  5. 动态规划-RMQ问题(ST算法)

    文章目录 RMQ问题 ST算法 模板 例题 P2251 质量检测 P1816 忠诚 P2216 [HAOI2007]理想的正方形 RMQ问题 RMQ(Range Minimum/Maximum Que ...

  6. 一二三系列之优先队列、st表——Battle,Heapsort,A Magic Lamp

    文章目录 Battle Heapsort A Magic Lamp Battle source 如果怪兽先死,那么英雄血量不足也没关系 反悔贪心 每次都先杀怪兽再说,如果血量不够了,就倒回去从怪兽打出 ...

  7. HDU3183(RMQ问题,ST算法)

    题目:A Magic Lamp 题意: 对于一个序列A[1...N],一共N个数,除去M个数使剩下的数组成的整数最小. 也就是说在A[1...N]中顺次选取N-M个数,使值最小. 本题很有技巧性,一开 ...

  8. RMQ问题(线段树算法,ST算法优化)

    RMQ (Range Minimum/Maximum Query)问题是指: 对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A中下标在[i,j]里的最小(大)值 ...

  9. RMQ ST算法简介

    RMQ (Range Minimum/Maximum Query)问题是指:对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A中下标在i,j里的最小(大)值,也就 ...

最新文章

  1. EOS/普元:中国IT业的悲哀
  2. OA项目14:权限管理功能分析
  3. 动态规划--最长公共子串
  4. Java开发环境!我总结了所有面试题
  5. 但行好事,莫问前程!
  6. 使用Maven的jaxws-maven-plugin插件,将wsdl生成java
  7. 小程序(仿微信发布说说功能)
  8. vue项目中使用lib-flexible解决移动端适配
  9. php曲线,PHP生成曲线图的函数
  10. 排序算法python实现_合并排序算法– Java,C和Python实现
  11. 游戏设计规则探秘之宾语
  12. Codeforces 659F Polycarp and Hay【BFS】
  13. 灵活的Zend Framework之使用自定义的Frontcontroller
  14. 双色球选号--python
  15. 小米路由器4C从0到自编译以及刷固件
  16. vm15安装mac10.14提取ipa包
  17. 如何理解类型geometry和geography以及4326、3857坐标系
  18. Maven: Non-resolvable import POM:Failure to find *** in *** was cached in the local repository.
  19. 【ensp】单臂路由的配置
  20. 【正点原子STM32连载】 第二十九章 低功耗实验 摘自【正点原子】MiniPro STM32H750 开发指南_V1.1

热门文章

  1. 关于C语言中的结构体内存对齐与位段问题
  2. 一个方法搞定安卓路由跳转
  3. 语音识别:时间序列的Smith–Waterman对齐算法
  4. c语言中说取消标识符是,2019年全国计算机二级C语言考试考点解析(3)
  5. we自动化po模式_Web自动化测试—PO设计模式(一)
  6. linux 监控进程是否存在并重启进程、打印进程日志
  7. boot jpa mysql postman spring_springboot使用spring-data-jpa操作MySQL数据库
  8. python随机生成中文字符串_用Python生成随机UTF-8字符串
  9. MySQL不能查看表_mysql root用户登录后无法查看数据库全部表
  10. 【深度学习】sigmoid - 二次代价函数 - 交叉熵 - logistic回归 - softmax