[Luogu3234] [LOJ2208]

题解及代码

锻炼哈希码力的一道题 , 具体细节见代码

#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]抄卡组的更多相关文章

  1. bzoj3574[Hnoi2014]抄卡组

    http://www.lydsy.com/JudgeOnline/problem.php?id=3574 我们发现如果所有的字符串都有*,那么只需要比较他们的“前缀”和“后缀”相同即可.“前缀”指第一 ...

  2. luogu P3234 [HNOI2014]抄卡组

    传送门 nmdwsm 自己看吧,不想写了qwq 垃圾代码如下 和题解完全不一样 #define LL long long #define uLL unsigned long long #define ...

  3. 【LG3234】[HNOI2014]抄卡组

    题面 题解 分三种情况: 若所有串都没有通配符,直接哈希比较即可. 若所有串都有通配符, 把无通配符的前缀 和 无通配符的后缀哈希后比较即可. 中间部分由于通配符的存在,一定可以使所有串匹配. 若部分 ...

  4. BZOJ3574 HNOI2014抄卡组(哈希)

    容易发现通配符中间的部分可以任意匹配,会造成的无法匹配的仅仅是前后缀,前缀和后缀可以分别独立处理.如果字符串均有通配符,只需要按前/后缀长度排序然后暴力匹配就可以了. 问题在于存在无通配符的字符串.显 ...

  5. 洛谷P3234 抄卡组 [HNOI2014] 字符串hash

    正解:字符串hash 解题报告: 传送门! 字符串hash是字符串匹配中很常见的一个方法,原理也很好懂,这里就不做太多阐述辣有时间放到hash笔记里面去QAQ 题意不说了挺好理解的,自带一句话概括好评 ...

  6. 【HNOI2014】抄卡组

    题面 题解 如果所有的字符串都有通配符,那么只要比较不含通配符的前缀和后缀就可以了. 否则一定有一个串没有通配符.找出这个字符串,然后将所有串与这个串匹配,通配符将\(B\)分成一段一段在\(A\)上 ...

  7. 【LOJ6254】最优卡组 堆(模拟搜索)

    [LOJ6254]最优卡组 题面 题解:常用的用堆模拟搜索套路(当然也可以二分).先将每个卡包里的卡从大到小排序,然后将所有卡包按(最大值-次大值)从小到大排序,并提前处理掉只有一张卡的卡包. 我们将 ...

  8. HearthBuddy卡组

    偶数萨 手打两天已上传说,各位加油  欧洲牧羊人 ### 火元素换艾雅# 职业:萨满祭司# 模式:狂野模式## 2x (2) 图腾魔像        # 2x (2) 大漩涡传送门   # 2x (2 ...

  9. 服务器&阵列卡&组raid 5

    清除raid信息后,机器将会读不到系统, 后面若进一步操作处理, raid信息有可能会被初始化掉,那么硬盘数据就有可能会被清空, 导致数据丢失, 否则如果只是清除raid信息,重做raid是可以还原系 ...

随机推荐

  1. sql server 错误总结

    1>无法访问sql server2000数据库 1.1>安装sql server2000 sp1的补丁包. 1.2>sql server 数据库开启了允许远程访问. 1.3>s ...

  2. #Pragma Pack与内存分配

    博客转载自:https://blog.csdn.net/mylinx/article/details/7007309 #pragma pack(n) 解释一: 每个特定平台上的编译器都有自己的默认“对 ...

  3. pandas dataframe 满足条件的样本提取

    pandas 的dataframe 对 数据查询可以通过3种方式 . 预备知识: 1. pandas 的索引和label都是从0开始的计数的 2. 时间切片都是左闭右开的. [5:6,:]  只会输出 ...

  4. PHP防止木马攻击的措施

    防止跳出web目录 只允许你的PHP脚本在web目录里操作,针对Apache,还可以修改httpd.conf文件限制PHP操作路径. 例如:php_admin_value  open_basedir( ...

  5. Mr_matcher的细节3

    主要是订阅了playbag发布的scan话题和odom话题 其类型分别为 //cache the static tf from base to laser getBaseToLaserTf(scan_ ...

  6. Python基础入门-For循环

    For循环的功能比较强大,他可以帮助我们实现很多重复性的工作.而且for循环能迭代不同的数据结构.他的应用也十分的广泛,作为初学者,我们需要对循环的概念多加理解和练习.接下来我们就来学习for循环的一 ...

  7. Linq学习<三> linq to entity

    之前一直用sql选择出数据放在一个集合中,然后再用Linq或者lambda去操作数据,今天学了Linq to entity 才知道原来linq产生是为了Entity.也就是EDM(实体数据模型) 关于 ...

  8. MongoDB整理笔记の安全访问

    MongoDB安全访问将从以下三个方面得到控制!   1.绑定IP内网地址访问MongoDB服务     2.设置监听端口     3.使用用户名和密码 绑定IP内网地址访问MongoDB服务 Mon ...

  9. Sharepoint2013搜索学习笔记之自定义结果精简分类(八)

    搜索结果页左边的结果精简分类是可以根据搜索结果自定义的,在搜索的部门日志结果集页面上我搜索测试关键字,左边分类导航在默认分类的基础上增加了一个日志类型的分类,如下图: 要实现这个效果,导航到之前定义的 ...

  10. go tcp使用

    TCP clientThere have been countless times during penetration tests that I've neededto whip up a TCP ...