[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是可以还原系 ...
随机推荐
- c++多线程编程(二)
这是道面试题目:有三个线程分别打印A.B.C,请用多线程编程实现,在屏幕上循环打印10次ABCABC… 见代码: #include <iostream> #include <Wind ...
- laravel 验证机制validation
Laravel 中 validation 验证 返回中文提示 全局设置 自己建一个zn文件夹,然后把en的4个文件全复制过去,修改validation.php的代码为下面的内容,然后在app.php修 ...
- Ubuntu16 install Anaconda
gbt@gbt-Precision-7720:~$ gbt@gbt-Precision-7720:~$ cd Anacondagbt@gbt-Precision-7720:~/Anaconda$ gb ...
- opennebula kvm日志
Fri Jul :: [InM][I]: Command execution fail: 'if [ -x "/home/oneadmin/tmp/one/im/run_probes&quo ...
- Docker学习笔记_网上资源参考
Docker学习,网上资源参考 1.菜鸟教程: http://www.runoob.com ...
- PHP数组的详细解读
数组的定义 数组的本质是管理和操作一组变量,数组中可以存储任意长度的数据,也可以存储任意类型的数据.数组中的单元称为元素,每个元素包括下标(键)和值,访问元素的时候,是通过下标来访问,包括一维数组,二 ...
- WEB前端--JavaScript
JavaScript JavaScript基础 一.JavaScript简介 JavaScript是一种嵌入到HTML文件中的描述性语言,由浏览器的解释器将其动态地处理成可执行的代码,能独立地完成与客 ...
- 运行monitor提示需要安装旧JAVA SE 6运行环境
MAC系统下运行monitor命令 ➜ tools git:(master) ✗ monitor 提示如下: 若要打开Eclipse.app,您需要Java SE 6 runtime,您想现在安装一个 ...
- ios PNG Crush error (PNG图片错误)
我是这么解决的: I had the same problem. How to fix : Open up image with Preview -> File > Export > ...
- Maven 项目使用开源中国镜像
从maven中央库下载jar非常缓慢甚至有时候会下载不下来. 可以采用中国的maven镜像.目前主要是 开源中国的镜像. 找到maven配置文件setting.xml,打开 中间添加开源中国的配置: ...