[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. Jmeter调度器配置

    Jmeter的线程组设置里有一个调配器设置,用于设置该线程组下脚本执行的开始时间.结束时间.持续时间及启动延迟时间.当需要半夜执行性能测试时会用到这个功能. ps:设置调度器配置,需要将前面的循环次数 ...

  2. strstr and strpos

    啥也不说  直接上代码: <?php $email = 'name@example.com'; $domain = strstr($email, '@'); echo $domain; // 打 ...

  3. c语言实践 给三个数输出最大的那个数

    我是怎么想的,我前面学过两个数比大小,比如有三个数,a b c,先比较a和b的大小,然后用那个较大的和c比较就得出最大的那个了.这个求三个数比大小的问题最后变化成 了两个数比大小了. int main ...

  4. Windows下用Nginx配置遇到的问题

    Nginx是一款轻量级的web服务器/反向代理服务器,更详细的释义自己百度了.目前国内像新浪.网易等都在使用它.先说下我的服务器软件环境: 系统:Windows Server + IIS + ngin ...

  5. PHP中static与self

    一直搞不清楚,今天百度自己也测试了一下. <?php class A { public static function closure(){ echo __CLASS__."<b ...

  6. Jquery delegate的理解

    $(".step4TagList .albumShow", "#divStep4").delegate(" .abmFct .icoRt", ...

  7. Django不能使用ip方式访问的解决办法

    问题: 启动服务后,使用http://127.0.0.1:8000/showcase/或者http://localhost:8000/showcase/都能访问, 但是使用http://192.168 ...

  8. C#帮助类:MD5加密

    /// <summary> /// MD5加密 /// </summary> public class Md5 { /// <summary> /// MD5加密 ...

  9. 使用 typescript ,提升 vue 项目的开发体验(1)

    此文已由作者张汉锐授权网易云社区发布. 欢迎访问网易云社区,了解更多网易技术产品运营经验. 前言:对于我们而言,typescript 更像一个工具 官方指南 从 vue2.5 之后,vue 对 ts ...

  10. 数组中 reduce累计运算

    let arr = [1,2,3,4]; let sum = (a, b) => a + b; arr.reduce(sum, 0); 最后输出10