[HNOI2014]抄卡组
锻炼哈希码力的一道题 , 具体细节见代码
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<vector>
#include<string>
#define debug(...) fprintf(stderr,__VA_ARGS__)
#define Debug(x) cout<<#x<<"="<<x<<endl
using namespace std;
typedef unsigned long long ULL;
const int INF=1e9+7;
const int N=1e5+5;
const int M=1e7+5;
const int base=2333;
ULL pow[M];//使用unsigned long long的自动溢出
int T,n;bool flag;
struct String{
int len,num;
vector<ULL> hash;//记录hash值
vector<int> word;//记录每个通配符的出现位置
inline friend bool cmp1(const String a,const String b){return a.word[1]<b.word[1];}
inline friend bool cmp2(const String a,const String b){return a.len-a.word[a.num]<b.len-b.word[b.num];}
inline friend bool operator< (String a,String b){
if(flag) return cmp1(a,b);
else return cmp2(a,b);
}
inline void init(){
len=0,num=0;
hash.clear();hash.push_back(0);
word.clear();word.push_back(0);//防止玄学错误,也是为了把数组下标处理为都+1
}
inline void build(string s){//计算hash值存下来
for(string::iterator it=s.begin();it!=s.end();it++){
hash.push_back(hash.back()*base+(*it));
len++;
if((*it)=='*'){
num++;
word.push_back(len);
}
}
}
inline ULL get_hash(int l,int r){return hash[r]-hash[l-1]*pow[r-l+1];}
inline int get_suffix()const{return len-word[num];}
inline bool match(String s){//有通配符匹配没有通配符
int suf=get_suffix();//后缀长度
if(word[1]-1+suf>s.len) return false;
if(get_hash(1,word[1]-1)!=s.get_hash(1,word[1]-1)) return false;
if(get_hash(word[num]+1,len)!=s.get_hash(s.len-suf+1,s.len)) return false;
int l=word[1],r=s.len-suf;
for(int i=1;i<num;i++){
int length=word[i+1]-word[i]-1;
ULL t=get_hash(word[i]+1,word[i+1]-1);
while(1){
if(l+length-1>r) return false;
if(s.get_hash(l,l+length-1)==t){l+=length;break;}
l++;
}
}
return true;
}
}s[N];
inline bool solve(){
ULL hashval=0;int pos=0;
for(int i=1;i<=n;i++){
if(s[i].num) continue;//有通配符就跳过
if(!pos){
pos=i;
hashval=s[i].hash[s[i].len];
}
else if(s[i].hash[s[i].len]!=hashval) return false;//没有通配符之间的
}
if(pos){
for(int i=1;i<=n;i++)
if(s[i].num&&(!s[i].match(s[pos]))) return false;//有通配符和没有通配符的
}
else{
flag=true;sort(s+1,s+n+1);//按word[1]排序
for(int i=1;i<n;i++){
if(s[i].get_hash(1,s[i].word[1]-1)!=s[i+1].get_hash(1,s[i].word[1]-1)) //判断前缀
return false;
}
flag=false;sort(s+1,s+n+1);//按word长度排序
for(int i=1;i<n;i++){
if(s[i].get_hash(s[i].word[s[i].num]+1,s[i].len)!=s[i+1].get_hash(s[i+1].len-s[i].get_suffix()+1,s[i+1].len)) //判断后缀
return false;
}
}
return true;
}
inline int read(){
int x;cin>>x;
return x;//...
}
int main(){
ios::sync_with_stdio(false);//关同步cin都很快
pow[0]=1;
for(int i=1;i<M;i++)
pow[i]=pow[i-1]*base;
T=read();
while(T--){
cin>>n;
for(int i=1;i<=n;i++){
string ch;
cin>>ch;
s[i].init();
s[i].build(ch);
}
puts(solve()?"Y":"N");
}
}
[HNOI2014]抄卡组的更多相关文章
- bzoj3574[Hnoi2014]抄卡组
http://www.lydsy.com/JudgeOnline/problem.php?id=3574 我们发现如果所有的字符串都有*,那么只需要比较他们的“前缀”和“后缀”相同即可.“前缀”指第一 ...
- luogu P3234 [HNOI2014]抄卡组
传送门 nmdwsm 自己看吧,不想写了qwq 垃圾代码如下 和题解完全不一样 #define LL long long #define uLL unsigned long long #define ...
- 【LG3234】[HNOI2014]抄卡组
题面 题解 分三种情况: 若所有串都没有通配符,直接哈希比较即可. 若所有串都有通配符, 把无通配符的前缀 和 无通配符的后缀哈希后比较即可. 中间部分由于通配符的存在,一定可以使所有串匹配. 若部分 ...
- BZOJ3574 HNOI2014抄卡组(哈希)
容易发现通配符中间的部分可以任意匹配,会造成的无法匹配的仅仅是前后缀,前缀和后缀可以分别独立处理.如果字符串均有通配符,只需要按前/后缀长度排序然后暴力匹配就可以了. 问题在于存在无通配符的字符串.显 ...
- 洛谷P3234 抄卡组 [HNOI2014] 字符串hash
正解:字符串hash 解题报告: 传送门! 字符串hash是字符串匹配中很常见的一个方法,原理也很好懂,这里就不做太多阐述辣有时间放到hash笔记里面去QAQ 题意不说了挺好理解的,自带一句话概括好评 ...
- 【HNOI2014】抄卡组
题面 题解 如果所有的字符串都有通配符,那么只要比较不含通配符的前缀和后缀就可以了. 否则一定有一个串没有通配符.找出这个字符串,然后将所有串与这个串匹配,通配符将\(B\)分成一段一段在\(A\)上 ...
- 【LOJ6254】最优卡组 堆(模拟搜索)
[LOJ6254]最优卡组 题面 题解:常用的用堆模拟搜索套路(当然也可以二分).先将每个卡包里的卡从大到小排序,然后将所有卡包按(最大值-次大值)从小到大排序,并提前处理掉只有一张卡的卡包. 我们将 ...
- HearthBuddy卡组
偶数萨 手打两天已上传说,各位加油 欧洲牧羊人 ### 火元素换艾雅# 职业:萨满祭司# 模式:狂野模式## 2x (2) 图腾魔像 # 2x (2) 大漩涡传送门 # 2x (2 ...
- 服务器&阵列卡&组raid 5
清除raid信息后,机器将会读不到系统, 后面若进一步操作处理, raid信息有可能会被初始化掉,那么硬盘数据就有可能会被清空, 导致数据丢失, 否则如果只是清除raid信息,重做raid是可以还原系 ...
随机推荐
- jQuery div鼠标移动效果
<head runat="server"> <meta http-equiv="Content-Type" content="tex ...
- fiddler抓包时显示Tunnel to......443
打开手机浏览器,输入http://192.168.0.65:8888/FiddlerRoot.cer
- Django rest_framework----认证,权限,频率组件
认证 from rest_framework.authentication import BaseAuthentication from rest_framework.exceptions impor ...
- Django框架 之 Pagination分页实现
Django框架 之 Pagination分页实现 浏览目录 自定义分页 Django内置分页 一.自定义分页 1.基础版自定义分页 1 2 3 4 5 6 7 8 9 10 11 12 13 14 ...
- 多线程协作 FileStream文件读写操作,读写冲突解决
using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Tex ...
- getchar() getch() getche() gets() puts() scanf()的用法及区别
getchar() putchar(ch) scanf() 头文件stdio.h getch() getche() 头文件conio.h gets() puts() 头文件stdio.h ...
- Python基础入门-For循环
For循环的功能比较强大,他可以帮助我们实现很多重复性的工作.而且for循环能迭代不同的数据结构.他的应用也十分的广泛,作为初学者,我们需要对循环的概念多加理解和练习.接下来我们就来学习for循环的一 ...
- Orace开源的异步IO编程库,特点是接口非常简单
官网:https://oss.oracle.com/projects/libaio-oracle/,正如标题所说,非常简单了,不用多解释,请直接看头文件,其中aio_poll类似于poll,重要的结构 ...
- java8之流的基本使用(二)
概述 流(stream())是java8的一个新特性,主要的作用就是将各种类型的集合转换为流,然后的方便迭代数据用的.例如: //将List类型的集合转换为流 list.stream() 转换为流之后 ...
- jQuery插件扩展extend的实现原理
相信每位前端的小伙伴对jQuery都不陌生吧,它最大的魅力之一就是有大量的插件,去帮助我们更轻松的实现各种功能. 前几天晚上,闲来无事,就自己动手写了个简单的jQuery插件,功能很简单,只是让选定的 ...