[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#实现访问网络共享文件夹
C#实现访问网络共享文件夹,使用 WNetAddConnection2A 和 WNetCancelConnection2A. 在目标服务器建立共享文件夹,建立访问账号test; public enum ...
- osm2pgsql导入duplicate key error in slim mode
This is a well known issue and one that is likely not going to be addressed any time soon. While the ...
- osm2pgsql windows “illegal option -W” error
新版本不支持 解决: 修改pg_hba.conf的METHOD为trust 参考:http://stackoverflow.com/questions/15510428/osm2pgsql-windo ...
- Python中常用模块二
一.hashlib (加密) hashlib:提供摘要算法的模块 1.正常的md5算法 import hashlib # 提供摘要算法的模块 md5 = hashlib.md5() md5.upd ...
- servlet模板
package ${enclosing_package};import java.io.IOException;import javax.servlet.ServletException;import ...
- VMware下Ubuntu虚拟机NAT模式 连接Xshell
1.Edit->Virtual NetWork Editor 2.选择NAT->NAT Settings 配置端口映射关系 很重要! Host port: 真实主机 一般是Windows端 ...
- 明码——第九届蓝桥杯C语言B组(省赛)第二题
原创 标题:明码 汉字的字形存在于字库中,即便在今天,16点阵的字库也仍然使用广泛.16点阵的字库把每个汉字看成是16x16个像素信息.并把这些信息记录在字节中. 一个字节可以存储8位信息,用32个字 ...
- C#ADO.NET基础一
简介 使用SQLite进行讲解 1.基础类: SQLiteConnection 连接数据库 SQLiteCommand 执行命令(增,删,改,查),或存储过程 SQLiteDataReader 读取查 ...
- c# 两个字符串,s="aeiou",s2="welcome to Quantum Asia"
c# 两个字符串,s="aeiou",s2="welcome to Quantum Asia" 方案一: 使用while循环: static void Mai ...
- linux系统上安装mysql5.6(详细步骤)
为了学习mycat 尝试在虚拟机上装mysql(看了别人的博客比划着安装),但装了两次都没成功.因此总结了如下步骤 有需要的朋友可以试下(linux需要联网) mysql-5.6.26.tar.gz百 ...