HDU - 4821 String(hash+尺取)

Given a string S and two integers L and M, we consider a substring of S as “recoverable” if and only if
(i) It is of length M*L;
(ii) It can be constructed by concatenating M “diversified” substrings of S, where each of these substrings has length L; two strings are considered as “diversified” if they don’t have the same character for every position.

Two substrings of S are considered as “different” if they are cut from different part of S. For example, string “aa” has 3 different substrings “aa”, “a” and “a”.

Your task is to calculate the number of different “recoverable” substrings of S.
Input
The input contains multiple test cases, proceeding to the End of File.

The first line of each test case has two space-separated integers M and L.

The second ine of each test case has a string S, which consists of only lowercase letters.

The length of S is not larger than 10^5, and 1 ≤ M * L ≤ the length of S.
Output
For each test case, output the answer in a single line.
Sample Input
3 3
abcabcbcaabc
Sample Output
2
题意: 题意就比较难理解,通俗来说就是给出m,l,和一段字符串,所求为该字符串中某种子串的个数;
某种子串为:

  1. 长度为m*l;
  2. 由m个长为l的子子串构成并且每个子子串互不相同。

**思路:**思路的话,刚开始考虑的为一个字符一个字符的移动,而这种思路当l为1并且字符串长度很长的时候最坏复杂度为O(n^2)肯定不行;
这里就要用一种巧妙的移动方式,尺取。
之前做过一个尺取字符串的题,和这个处理方式很像,这里是博客: https://blog.youkuaiyun.com/nb_weige/article/details/101940000
尺取方法在注释
代码:

//#include<bits/stdc++.h>
#include<stdio.h>
#include<algorithm>
#include<map>
#include<string.h>
using namespace std;
const int maxn=1e5+7;
#define ll long long
#define ull unsigned long long
ull base[maxn];
ull Hash[maxn];
char s[maxn];
void gethash(char *p)
{
	Hash[0]=0;
	ll len=strlen(p);
	for(int i=1;i<=len;i++)
	{
		Hash[i]=Hash[i-1]*131+p[i];
	}
}
ull fun(int l,int r)
{
	return Hash[r]-Hash[l-1]*base[r-l+1];
}
map<ull,int>mp;
int main()
{
	base[0]=1;
	for(int i=1;i<=maxn;i++)
		base[i]=base[i-1]*131;
	ll l,m;
	while(~scanf("%lld%lld",&m,&l))
	{
		ll num=0;
		scanf("%s",s+1);
		ll len=strlen(s+1);
		Hash[0]=0;
		for(int i=1;i<=len;i++)
			Hash[i]=Hash[i-1]*131+s[i];
		for(int i=1;i<=l&&i+m*l-1<=len;i++)     //只需要遍历到 l ,因为后面的尺取已经遍历过
		{
			mp.clear();
			for(int j=i;j<i+l*m;j+=l)
				mp[fun(j,j+l-1)]++;
			if(mp.size()==m)                    //找到第一个满足条件的,然后尺取
				num++;
			for(int j=i+m*l;j+l-1<=len;j+=l)
			{
				mp[fun(j,j+l-1)]++;                     //尺取: 向后延伸,加上后面的长 l 的串
				mp[fun(j-m*l,j-m*l+l-1)]--;             //尺取:向前看,去掉前面的一个长 l 的串
				if(mp[fun(j-m*l,j-m*l+l-1)]==0)
					mp.erase(fun(j-m*l,j-m*l+l-1));
				if(mp.size()==m)
					num++;
			}
		}
		printf("%lld\n",num);
	}
	return 0;
}

《C++编程实例100篇》是一本深入实践、极具价值的编程教程,它针对C++编程语言提供了丰富的实例,旨在帮助读者更好地理解和掌握C++的各项特性与编程技巧。这本书的经典之处在于它将理论与实践相结合,通过100个精心设计的编程实例,覆盖了C++的各个核心领域,包括基础语法、面向对象编程、模板、异常处理、STL(标准模板库)等。 我们来探讨C++的基础语法。C++是C语言的增强版,它保留了C语言的高效性和灵活性,并引入了类、对象和继承等面向对象编程概念。基础语法包括变量声明、数据类型、运算符、控制结构(如if语句、for循环、while循环)、函数的定义和调用等。在实例中,你可能会遇到如何编写简单的程序,如计算两个数的和,或者实现一个简单的猜数字游戏。 C++的面向对象编程是其一大特色。通过类和对象,你可以构建复杂的软件系统。类是对象的蓝图,它定义了对象的属性和行为。实例化一个类,就是创建一个具体的对象。继承允许你创建新的类,这些类从现有的类派生,共享其属性和方法,同时可以添加新的功能。多态性是面向对象的另一个关键特性,它使得不同类型的对象可以对同一消息作出不同的响应。这些概念在实例中会以各种形式展现,例如设计一个图形界面的类层次,或实现一个简单的模拟游戏。 接下来是模板,C++的模板功能让代码更加通用,可以处理不同类型的数据。模板分为函数模板和类模板,前者可以创建泛型函数,后者可以创建泛型类。通过模板,你可以编写出高效且灵活的代码,比如实现一个通用的排序算法。 异常处理是C++中用于处理程序运行时错误的机制。当程序出现异常情况时,可以抛出一个异常,然后在适当的点捕获并处理这个异常。这使得代码能够优雅地处理错误,而不是让程序崩溃。实例中可能会有涉及文件操作或网络通信时可能出现的异常处理示例。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值