九度oj 题目1397:查找数段

题目描述:

在BaiDu搜索引擎里,如何提高搜索效率是研发人员为之奋斗的目标。现在,JOBDU密码库里也有一段数字片段S(0<长度<=100,000),HQ想通过智能搜索得到包含关键字P(0<长度<=100,000)的某个数段长度,如果存在多个这样的数段,则选择长度最小的。例如,数字片段123456789,关键字为257.显然S本身就包含257,所以长度9是一个符合的数段,但是HQ从S中找到子串234567也包含关键字,并且无法找到更短的子串满足条件,因此返回结果6。PS:JOBDU密码库里的数字片段可能包含“*”,表示这一位可以是(0~9)中任意1个,具体见案例2。

输入:

输入有多个测试案例,每个测试案例1行,包括两个字串。

第一个为数字片段S(0<长度<=100,000),第二个为关键字P(0<长度<=100,000)。

输出:

根据输入案例返回查找结果,如果不存在包含关键字的数字片段则返回0。

样例输入:
123456789 257
33**2*** 113
样例输出:
6
3

题目要求找到包含关键字的最短字段的长度。需要用尺取法求解。至于什么是尺取法,搜索一下,网上讲的非常好

代码如下
 1 #include <cstdio>
 2 #include <cstdlib>
 3 #include <cstring>
 4 #include <iostream>
 5 using namespace std;
 6
 7 char str[100002], test[100002];
 8 int goal[12];
 9 int has[12];
10
11
12 bool isOk(int help) {
13     for(int i = 0; i <= 9; i++) {
14         if(has[i] < goal[i]) {
15             int cha = goal[i] - has[i];
16             if(help < cha) {
17                 return false;
18             }
19             help = help - cha;
20         }
21     }
22     return true;
23 }
24
25 int main(int argc, char const *argv[])
26 {
27     while(scanf("%s %s",str, test) != EOF) {
28         memset(goal, 0, sizeof(goal));
29         memset(has, 0, sizeof(has));
30         int any = 0;
31
32         int lent = strlen(test);
33         for(int i = 0; i < lent; i++) {
34             goal[test[i] - ‘0‘]++;
35         }
36         int ans = 100002;
37         int from = 0, to = 0;
38         int lens = strlen(str);
39         bool isSuccess = true;
40         while(to < lens) {
41             while(to==0 || (!isOk(any) && to < lens)) {
42
43                 if(str[to] == ‘*‘) {
44                     any++;
45                 }
46                 else {
47                     has[str[to] - ‘0‘]++;
48                 }
49                 to++;
50             }
51             if(from == 0 && to == lens && !isOk(any)) {
52                 isSuccess = false;
53                 break;
54             }
55             int fir = true;
56             while(fir || isOk(any)) {
57
58                 fir = false;
59                 if(str[from] == ‘*‘) {
60                     any--;
61                 }
62                 else {
63                     has[str[from] - ‘0‘]--;
64                 }
65                 from++;
66
67             }
68             int len = to - from + 1;
69             ans = min(ans,len);
70
71         }
72         if(!isSuccess) {
73             ans = 0;
74         }
75         printf("%d\n", ans);
76     }
77     return 0;
78 }

时间: 2024-08-28 03:38:45

九度oj 题目1397:查找数段的相关文章

九度oj 题目1007:奥运排序问题

九度oj 题目1007:奥运排序问题   恢复 题目描述: 按要求,给国家进行排名. 输入:                        有多组数据. 第一行给出国家数N,要求排名的国家数M,国家号从0到N-1. 第二行开始的N行给定国家或地区的奥运金牌数,奖牌数,人口数(百万). 接下来一行给出M个国家号. 输出:                        排序有4种方式: 金牌总数 奖牌总数 金牌人口比例 奖牌人口比例 对每个国家给出最佳排名排名方式 和 最终排名 格式为: 排名:排名

九度oj 题目1546:迷宫问题 (概率dp guess消元)

题目链接:点击打开链接 题目描述: 给定一个n*m的迷宫,如 S.. ..# E.E 其中,S代表开始位置,#代表不可行走的墙,E代表出口. 主人公从开始位置出发,每次等概率的随机选择下一个可以行走的位置,直到到达某一个出口为止. 现在他想知道,在这一概率事件中,它从开始位置走到某一个出口的期望步数是多少. 输入: 输入包含多组测试用例,每组测试用例由两个整数n,m(1<=n,m<=15)开始,代表迷宫的大小 接下去n行每行m个字符描述迷宫信息,具体规则如题面所述. 数据保证至少存在一个E和一

九度oj题目1009:二叉搜索树

题目描述: 判断两序列是否为同一二叉搜索树序列 输入:                        开始一个数n,(1<=n<=20) 表示有n个需要判断,n= 0 的时候输入结束. 接下去一行是一个序列,序列长度小于10,包含(0~9)的数字,没有重复数字,根据这个序列可以构造出一颗二叉搜索树. 接下去的n行有n个序列,每个序列格式跟第一个序列一样,请判断这两个序列是否能组成同一颗二叉搜索树. 输出:                        如果序列相同则输出YES,否则输出NO 样

九度oj 题目1475:IP数据包解析

题目描述: 我们都学习过计算机网络,知道网络层IP协议数据包的头部格式如下: 其中IHL表示IP头的长度,单位是4字节:总长表示整个数据包的长度,单位是1字节. 传输层的TCP协议数据段的头部格式如下: 头部长度单位为4字节. 你的任务是,简要分析输入数据中的若干个TCP数据段的头部. 详细要求请见输入输出部分的说明. 输入: 第一行为一个整数T,代表测试数据的组数. 以下有T行,每行都是一个TCP数据包的头部分,字节用16进制表示,以空格隔开.数据保证字节之间仅有一个空格,且行首行尾没有多余的

九度oj 题目1079:手机键盘

题目描述: 按照手机键盘输入字母的方式,计算所花费的时间 如:a,b,c都在“1”键上,输入a只需要按一次,输入c需要连续按三次. 如果连续两个字符不在同一个按键上,则可直接按,如:ad需要按两下,kz需要按6下 如果连续两字符在同一个按键上,则两个按键之间需要等一段时间,如ac,在按了a之后,需要等一会儿才能按c. 现在假设每按一次需要花费一个时间段,等待时间需要花费两个时间段. 现在给出一串字符,需要计算出它所需要花费的时间. 输入: 一个长度不大于100的字符串,其中只有手机按键上有的小写

九度oj 题目1171:C翻转

题目描述: 首先输入一个5 * 5的数组,然后输入一行,这一行有四个数,前两个代表操作类型,后两个数x y代表需操作数据为以x y为左上角的那几个数据. 操作类型有四种:  1 2 表示:90度,顺时针,翻转4个数  1 3 表示:90度,顺时针,翻转9个数  2 2 表示:90度,逆时针,翻转4个数  2 3 表示:90度,逆时针,翻转9个数 输入: 输入有多组数据. 每组输入一个5 * 5的数组,然后输入一行,这一行有四个数,前两个代表操作类型,后两个数x y代表需操作数据为以x y为左上角

九度oj 题目1411:转圈

题目描述: 在一个有向图有n个顶点(编号从1到n),给一个起点s,问从起点出发,至少经过一条边,回到起点的最短距离. 输入: 输入包括多组,每组输入第一行包括三个整数n,m,s(1<=n<=500,0<=m<=10000,1<=s<=n),接下来有m行,每行包括三个整数a,b,c(1<=a,b<=n,1<=c<=1000),表示有一条a到b的边,长度为c. 输出: 对每组输入.输出最短距离,如果没有这个一条路径输出"help!"

九度oj 题目1041:Simple Sorting

题目描述: You are given an unsorted array of integer numbers. Your task is to sort this array and kill possible duplicated elements occurring in it. 输入: For each case, the first line of the input contains an integer number N representing the quantity of

&lt;九度 OJ&gt;题目1012:畅通工程

题目描述: 某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇.省政府"畅通工程"的目标是使全省任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要互相间接通过道路可达即可).问最少还需要建设多少条道路? 输入: 测试输入包含若干测试用例.每个测试用例的第1行给出两个正整数,分别是城镇数目N ( < 1000 )和道路数目M:随后的M行对应M条道路,每行给出一对正整数,分别是该条道路直接连通的两个城镇的编号.为简单起见,城镇从1到N编号. 注意