魔咒词典

2023-05-16

题目描述

 哈利波特在魔法学校的必修课之一就是学习魔咒。据说魔法世界有100000种不同的魔咒,哈利很难全部记住,但是为了对抗强敌,他必须在危急时刻能够调用任何一个需要的魔咒,所以他需要你的帮助。

 给你一部魔咒词典。当哈利听到一个魔咒时,你的程序必须告诉他那个魔咒的功能;当哈利需要某个功能但不知道该用什么魔咒时,你的程序要替他找到相应的魔咒。如果他要的魔咒不在词典中,就输出“what?”

输入

 首先列出词典中不超过100000条不同的魔咒词条,每条格式为:

[魔咒] 对应功能
其中“魔咒”和“对应功能”分别为长度不超过20和80的字符串,字符串中保证不包含字符“[”和“]”,且“]”和后面的字符串之间有且仅有一个空格。词典最后一行以“@END@”结束,这一行不属于词典中的词条。
词典之后的一行包含正整数N(<=1000),随后是N个测试用例。每个测试用例占一行,或者给出“[魔咒]”,或者给出“对应功能”。

输出

 每个测试用例的输出占一行,输出魔咒对应的功能,或者功能对应的魔咒。如果魔咒不在词典中,就输出“what?”

 

以下借鉴了几位网友的思路和代码

样例输入


[expelliarmus] the disarming charm
[rictusempra] send a jet of silver light to hit the enemy
[tarantallegra] control the movement of one's legs
[serpensortia] shoot a snake out of the end of one's wand
[lumos] light the wand
[obliviate] the memory charm
[expecto patronum] send a Patronus to the dementors
[accio] the summoning charm
@END@
4
[lumos]
the summoning charm
[arha]
take me to the sky  

样例输出


light the wand
accio
what?
what?  

提示

 使用stl当中的map将对应的字符串建立一一映射。 

代码如下:


 1 #include<iostream>
 2 #include<string>
 3 #include<algorithm>
 4 #include<cstdio>
 5 #include<cstring>
 6 #include<cmath>
 7 #include<map>
 8 using namespace std;
 9 int main()
10 {
11     map<string,string>a;
12     int n,l,r;
13     string s,index,value;
14     while(getline(cin,s) && s!="@END@")
15     {
16         l=s.find('[');r=s.find(']');//找出左右括号所处的下表
17         index=s.substr(l+1,r-l-1);//在字符串中提取出魔咒,具体怎么计算的,可以自己选个测试用例在草稿纸上画画便一目了然
18         value=s.substr(r+2,s.size()-r);//提取出魔咒对应的内容
19         a[index]=value;//将魔咒作为索引
20         a[value]=index;//将内容作为索引
21     }
22     cin>>n;getline(cin,s);//读取走缓冲区中讨厌的回车键
23     for(int i=0;i<n;i++)
24     {
25         getline(cin,s);
26         if(s[0]=='[')
27         {
28             s.erase(s.size()-1,1);//如果以魔咒为索引,就把讨厌的左右中括号去掉
29             s.erase(0,1);
30         }
31         if(a.find(s)==a.end()) cout<<"what?"<<endl;//判断在map里是否有这个索引
32         else cout<<a[s]<<endl;
33     }
34     return 0;
35 }  

 

题解:如果数据量小,每个字符串的长度小的话,直接使用map<string,string>轻松解决,但是数据量大,字符串的长度可能很长,这样会造成超时和超内存的问题,因此使用字符串hash和映射解决


 1 #include<iostream>
 2 #include<string> 
 3 #include<map>
 4 #include<utility>
 5  
 6 using namespace std;
 7 typedef unsigned long long ull;
 8 const int N=200010;
 9 const ull p=1e9+13;
10 const ull mod=1e9+7;
11 string a[N]; 
12 map<ull,int>d;
13  
14 ull hashstr(string s)  //hash函数
15 {
16     ull v=0;
17     for(int i=0;s[i];i++)
18       v=(v*p+(int)s[i])%mod;
19     return v;
20 }
21  
22 int main()
23 {
24     int k=0,n;
25     for(;;){
26         string s1,s2,s;
27         getline(cin,s);
28         int p1=s.find("[",0);
29         int p2=s.find("] ",p1+1);
30         if(p1==string::npos||p2==string::npos)
31           break;
32         s1=s.substr(p1+1,p2-p1-1);//s1为名字
33         s2=s.substr(p2+2);//s2为功能
34         a[++k]=s1;
35         ull v=hashstr(s2);
36         d.insert(make_pair(v,k));
37         a[++k]=s2;
38         v=hashstr(s1);
39         d.insert(make_pair(v,k));
40     }
41     scanf("%d",&n);
42     getchar();
43     for(int i=1;i<=n;i++){
44         string s;
45         getline(cin,s);
46         if(s[0]=='['){
47             s.erase(s.end()-1);
48             s.erase(s.begin());
49         }
50         ull v=hashstr(s);
51         if(d.find(v)!=d.end()){
52             int pos=(d.find(v))->second;;
53             cout<<a[pos]<<endl;
54         }
55         else
56           cout<<"what?"<<endl;
57     }
58     return 0;
59 }  

 

思路
模拟赛上拉的题目,意思很简单,给你多行“【咒语】功能”以“@END@”结尾,最后输入字符串查询,要求输入的如果是功能输出咒语,如果是咒语输出功能,都不存在则输出“what?”,首先想到的就是直接用map映射但是数据量太大有10w想着肯定会超内存,就没有用后来想到用Hash+二分快速查找,但是没学会字符串Hash无奈只能对着字符串进行二分查找结果浪费很多时间还wa掉了,后来看榜单发现大家基本都写出来了,就猜肯定是hd数据弱果断暴力遍历着查找,虽然在数组范围上坑了几次,最后还是AC,果然是弱数据。赛完就赶紧学了字符串Hash把我比赛中没实现的想法给实现了,效率果然比暴力高了很多。就是将每一个串Hash一个值(这里用BKDRHash是java用的Hash算法可以达到10W数据无冲突hash值是32位整数),用结构体将这个Hash值与字符串的位置对应起来,来达到随机读取的目的,把输入的字典全部Hash后将这些Hash值排序,方便二分查找。
PS:后来有找来了几个10W的数据结果果断wa掉,看来不解决Hash冲突还是不行的,可能真的是hd数据水的没有冲突把,或者后来找到数据专门对几个Hash算法进行的针对,看来对于大数据的题目还是要解决问题


 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstring>
 4 #include <cmath>
 5 #include <algorithm>
 6 #define N 100005
 7 using namespace std;
 8 char spell[N][85];//储存功能
 9 char magic[N][25];//储存咒语
10 int num=1;
11 struct node//Hash值对应的数组下标
12 {
13     int Hash;
14     int l;
15 } key[N],tail[N];//咒语和功能
16 int cmp(node a,node b)
17 {
18     return a.Hash<b.Hash;//按照Hash排序
19 }
20 
21 unsigned int BKDRHash(char *str)//Hash算法
22 {
23     unsigned int seed = 131;
24     unsigned int hash = 0;
25 
26     while (*str)
27     {
28         hash = hash * seed + (*str++);
29     }
30     return (hash & 0x7FFFFFFF);
31 }
32 void init()//初始化,计算所有串的Hash值
33 {
34     for(int i=1; i<num; i++)
35     {
36         key[i].Hash=BKDRHash(magic[i]);
37         key[i].l=i;
38         tail[i].Hash=BKDRHash(spell[i]);
39         tail[i].l=i;
40     }
41     sort(key,key+num,cmp);//按照Hash值排序,方便二分查找
42     sort(tail,tail+num,cmp);
43 }
44 int main()
45 {
46 
47     while(scanf("%s",magic[num]))
48     {
49         if(!strcmp(magic[num],"@END@"))//终止条件
50             break;
51         getchar();//吞掉空格
52         gets(spell[num++]);//第二个串
53     }
54     init();//初始化
55     int n;
56     scanf("%d",&n);
57     getchar();//吞回车
58     while(n--)//n次查询
59     {
60         node tmp;
61         char str[120];
62         gets(str);
63         if(str[0]=='[')//是咒语
64         {
65             tmp.Hash=BKDRHash(str);//计算Hsah值
66             int mid=lower_bound(key,key+num,tmp,cmp)-key;二分查找位置(如果没找到,返回值为假设这个值存在应该在的位置)
67             if(key[mid].Hash!=tmp.Hash)//比较两者是否相同(是否存在)
68                 printf("what?\n");
69             else
70                 printf("%s\n",spell[key[mid].l]);//输出
71         }
72         else
73         {
74             tmp.Hash=BKDRHash(str);
75             int mid=lower_bound(tail,tail+num,tmp,cmp)-tail;
76             if(tail[mid].Hash!=tmp.Hash)
77                 printf("what?\n");
78             else
79             {
80                 int l=strlen(magic[tail[mid].l]);//处理咒语两边的方括号
81                 magic[tail[mid].l][l-1]=0;
82                 printf("%s\n",magic[tail[mid].l]+1);
83             }
84         }
85     }
86     return 0;
87 }  

 

分析:
此题其实就是字符串哈希的模板题,将字符串s1映射到s2,输入s1求s2,输入s2求s1
第一反应是用STL,但此题用map会MLE或TLE,所以手写哈希表
字符串哈希的方法如下:
1.将前8位字母的ASCII码乘积作为key(用long long存),转换为数字的哈希问题
2.用数组和链表存储,key对应的字符串存在数组下标为key%c(c为常数)的地方
若key%c相同,则存在链表里,下图直观展示了该过程(图中数字为key)

3.查找时先到key对应的数组,如果元素不对,则继续遍历链表,直到查到该元素


 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 using namespace std;
 5 const int c=10000;
 6 struct myhash{//评测机上hash是关键字
 7     struct node {
 8         long long key;//若key%c相同辅助判重
 9         string s1;//若key相同辅助判重
10         string s2;//要查找的东西,字符串s1对应的字符串s2
11         node* next;//链表用指针
12     }a[c+5];
13     void push(long long x,string s1,string s2) {//插入函数
14         if(a[x%c].key==0) {//如果数组x%c处没有元素,直接插入
15             a[x%c].key=x;
16             a[x%c].s1=s1;
17             a[x%c].s2=s2;
18         } else {//如果有,链表插入新节点(头插入)
19             node *tmp=new node();
20             tmp->key=x;
21             tmp->s1=s1;
22             tmp->s2=s2;
23             tmp->next=a[x%c].next;//此处相当于head
24             a[x%c].next=tmp;
25         }
26     }
27     string found(long long x,string s1) {//查找函数
28         if(a[x%c].key==x&&a[x%c].s1==s1) return a[x%c].s2;//如果字符串在数组里,直接返回对应的字符串s2
29         if(a[x%c].key==0) return "what?";//如果数组里没有元素,链表里肯定也没有,返回what?
30         else {
31             node* tmp=a[x%c].next;//遍历链表查找
32             while(tmp!=NULL) {
33                 if(tmp->key==x&&tmp->s1==s1) return tmp->s2;
34                 tmp=tmp->next;
35             }
36             return "what?";
37         }
38     }
39 };
40 myhash H1,H2;//H1存储名字对应的内容,H2存储内容对应的名字
41 long long to_num(string str){//前8位的乘积hash 
42     long long ans=1;
43     int len=str.length();
44     if(len>8) len=8;
45     for(int i=0;i<len;i++) ans*=str[i];
46     return ans;
47 }
48 int main() {
49     int t; 
50     string str,s1,s2;
51     while(1) {
52         int i;
53         getline(cin,str);//有空格,必须用getline
54         if(str=="@END@") break;
55         for(i=0; i<str.length(); i++) {
56             if(str[i]==']') break;//找到咒语名字和内容的分界点
57         }
58         s1=str.substr(0,i+1);
59         s2=str.substr(i+2,str.length()-i-1);
60         //将str分为两部分,s1是名字,s2是内容
61         H1.push(to_num(s1),s1,s2);
62         H2.push(to_num(s2),s2,s1);
63     }
64     cin>>t;
65     getchar();//输入一个换行符,否则换行符会被getline读入到str中,导致在哈希表中查找'\n',多输出一个"what?"
66     while(t--){
67         getline(cin,str);
68         bool have_kuo=false;//区分输入的是内容还是名字
69         for(int i=0;i<str.length();i++){
70             if(str[i]==']'){
71                 have_kuo=true;
72                 break;
73             }
74         } 
75         string tmp;
76         if(have_kuo) tmp=H1.found(to_num(str),str);
77         else tmp=H2.found(to_num(str),str);
78         if(tmp[0]=='['){
79             for(int i=1;i<tmp.length()-1;i++)  cout<<tmp[i];//输出咒语名字时不带括号
80             cout<<endl;
81         }
82         else cout<<tmp<<endl;
83     }
84 }  

 

转载于:https://www.cnblogs.com/jiamian/p/10634151.html

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

魔咒词典 的相关文章

  • ios设备备份,更新路径(mac os)

    备份路径 xff1a Library Application Support MobileSync Backup 固件更新下载路径 xff1a Library iTunes iPhone Software Updates 转载于 https
  • Flutter FormatException: Bad UTF-8 encoding 0xc3 (at offset 172)

    是文件编码问题 1 xff0c 打开 Android Studio 进入设置界面 Ctrl 43 Alt 43 S 或 File gt Settings 打开如下界面 xff1a 2 xff0c 将 Project Encodeing 设置
  • 数据库系统原理及应用教程复习笔记(第3 版)

    最近在复习数据库相关知识点 xff0c 过几天就要考试了 xff1b 第一章 数据库基础知识 1 数据库管理是数据处理的基础工作 xff0c 数据库是数据管理的技术和手段 数据库中的数据具有整体性和共享性 1 2 数据库系统的核心 xff1
  • ubuntu 由于没有公钥,无法验证下列签名

    W GPG 错误 xff1a http deb opera com stable InRelease 由于没有公钥 xff0c 无法验证下列签名 xff1a NO PUBKEY 63F7D4AFF6D61D45 没有公钥的此类错误只要在终端
  • mysql2 没有配置文件怎么办

    1111mysql没有配置文件会用默认的配置 启动时没有使用配置文件 如果没有设置使用指定目录my cnf文件及默认读取目录没有my cnf文件 xff0c 表示mysql启动时并没有加载配置文件 xff0c 而是使用默认配置 需要修改配置
  • repo init 时gpg: 无法检查签名:找不到公钥

    i found a solution here http www marshut com wrrts repo release 1 12 4 html Sorry I realized today that we didn 39 t upl
  • 使用Java对字符串进行升序排序

    Java对字符串的很多API和功能是JavaWeb能广泛发展的基础 xff0c 下面是一道经典的字符串操作题 xff0c 需要边查JAVASE的API对每个步骤进行操作 题目 xff1a 给一个字符串 xff0c 34 34 12 8 0
  • 2.2 关系代数运算

    2 2 1 关系代数的五个基本操作 考核要求 xff1a 达到 简单应用 层次 知识点 xff1a 五个基本操作的含义和运算应用 1 并 xff1a 两个关系需有相同的关系模式 xff0c 并的对象是元组 xff0c 由两个关系所有元组构成
  • FTP 之 550 permission denied

    做案子時 將WinInet dll封裝 用C 調用 做了四個函數 分別是上傳 下載 刪除 清空文件夾 在開發環境中測試沒有任何問題 但拿到正式環境中卻出現 34 550 c dh MPF dir ABC 34 不存在的錯誤 但實際上的FTP
  • Go——值、指针和引用

    传值还是传引用 在函数和接口章节 xff0c 我们知道Go只有一种参数传递规则 xff0c 那就是值拷贝 xff0c 这种规则包括两种含义 xff1a 函数参数传递时使用的是值拷贝 实例赋值给接口变量 xff0c 接口对实例的引用是值拷贝
  • 读取PC版微信数据库(电脑版微信数据库)内容

    原始网址 https www cnblogs com Charltsing p WeChatPCdb html 联系QQ xff1a 564955427 1 PC版微信的密钥是32位byte xff0c 不同于安卓版 xff08 7位字符串
  • Pytorch-属性统计

    引言 本篇介绍Pytorch属性统计的几种方式 统计属性 求值或位置 normmean sumprodmax min argmin argmaxkthvalue topk norm norm 与 normalize norm指的是范数 xf
  • 高性能异步爬虫

    背景 其实爬虫的本质就是client发请求批量获取server的响应数据 xff0c 如果我们有多个url待爬取 xff0c 只用一个线程且采用串行的方式执行 xff0c 那只能等待爬取一个结束后才能继续下一个 xff0c 效率会非常低 需
  • [operator]deepin 卸载自带搜狗输入法后,输入法消失

    解决这个问题我先是升级了官方的im config套件 xff0c 升级后发现并没有什么用 xff0c 然后使用以下方式 xff0c 做个记录 命令行操作 删除搜狗的残留文件 cd config rm rf SogouPY users rm
  • DPK

    一 概念 dpk文件是Delphi的包文件 xff0c 有dpk文件的组件安装比较方便 一般来说 xff0c 支持不同版本Delphi的组件会有不同的dpk文件 xff0c 一般以7结尾的dpk文件是支持Delphi 7的 如果没有支持De
  • free -g 说明

    free g 说明 xff1a free g 43 buffers cache 说明 xff1a buffer 写缓存 xff0c 表示脏数据写入磁盘之前缓存一段时间 xff0c 可以释放 sync命令可以把buffer强制写入硬盘 cac
  • Google Drive 里的文件下载的方法

    Google Drive 里并不提供创建直接下载链接的选项 xff0c 但是可以通过小小的更改链接形式就能把分享的内容保存到本地 例如 xff0c 一份通过 Google Drive 分享的文件链接形式为 xff1a https drive
  • 关于虚拟机VMware Tools安装中出现的无法自动安装VMCI驱动程序的问题

    问题 解决方法 根据配置文件信息找到所在的虚拟机位置 找到后缀名为vmx的文件 xff0c 右键打开方式中选择使用记事本打开 选择左上角编辑中的查找功能输入图中的查找内容后 xff0c 点击查找下一个 将其原先的TRUE值改为false即可

随机推荐

  • 服务器系统运行内存,服务器系统运行内存使用情况

    服务器系统运行内存使用情况 内容精选 换一换 包年 包月的计费模式也称为包周期计费模式 xff0c 是一种预付费方式 xff0c 按订单的购买周期计费 xff0c 适用于可预估资源使用周期的场景 xff0c 价格比按需计费模式更优惠 包年
  • Ubuntu 18.04 上使用xrdp远程桌面登录蓝屏解决

    所有工具方法来自 http c nergy be blog p 61 13663 免责声明 xff1a 像往常一样 xff0c 使用此风险自负 xff01 本地有台机器装了乌班图18 04版本系统 我们想远程图形化访问它 我第一想法是xrd
  • Go——习惯用法

    1 干净与强迫症 Go在代码干净上有了近乎苛刻的要求 xff0c 主要体现在如下几个方面 xff1a 编译器不能通过未使用的局部变量 xff08 包括未使用的标签 xff09 import 未使用的包同样通不过编译 所有的控制结构 函数和方
  • snprintf()函数使用方法

    众所周知 sprintf不能检查目标字符串的长度 xff0c 可能造成众多安全问题 所以都会推荐使用snprintf 自从snprintf代替了sprintf xff0c 相信大家对snprintf的使用都不会少 xff0c 函数定义如下
  • Openwrt无线中继设置并访问外网

    Openwrt无线中继设置并访问外网 本篇博文参考来自 xff1a http blog csdn net pifangsione article details 13162023 配置目标 主路由器使用AP模式发射Wifi从路由器使用Cli
  • 在 Windows 7 中禁用IPv6协议/IPv6隧道

    How to disable certain Internet Protocol version 6 IPv6 components in Windows Vista Windows 7 and Windows Server 2008 ht
  • python matplotlib绘图大全(散点图、柱状图、饼图、极坐标图、热量图、三维图以及热图)...

    2019 7 14晚 matplotlib七种常见图像输出编程大全 七种图形汇总输出如下 xff1a import numpy as np 导入数据结构nmupy模块 import matplotlib pyplot as plt 导入ma
  • 光纤模式分布 matlab,matlab计算单模光纤模式分布(公布源代码及参考文献)

    最近在使用matlab计算单模光纤纤芯模及包层模模场分布时 xff0c 有一些问题一直悬而未决 xff0c 多次咨询原作者后虽解决了部分问题 xff0c 但是余下的问题原作者也不理我了 xff0c 特发此贴以广交学习光纤方面的同学 老师及科
  • Ubuntu下编译安装MySQL5.7

    tar zxvf mysql 5 7 14 tar gz cd mysql 5 7 14 第一步 xff1a cmake DCMAKE INSTALL PREFIX 61 usr local mysql DMYSQL DATADIR 61
  • UNICODE使用的一些知识和技巧

    UNICODE宏和 UNICODE宏的关系 在windows编程中 经常要编译Unicode版本的程序 方法是工程文件的配置中加上UNICODE或者 UNICODE编译条件 那么到底是用哪一个呢 Jeffrey Richter在 Windo
  • cmake 常用命令

    1 使用日期 获取时间 string TIMESTAMP DATE TIME 34 y m d H M 34 获取日期 string TIMESTAMP DATE VERSION 34 m d 34 转载于 https www cnblog
  • QQ2008 msg.db,user.db读取

    Saturday November 27 2010 msg db读取 下载 user db读取 下载 转载于 https www cnblogs com ycdx2001 archive 2010 11 27 1889498 html
  • MongoDB——Mac环境搭建

    1 下载 官网地址 xff1a https www mongodb com 2 解压并配置 解压到 usr local 配置Path xff0c vim打开 bash profile添加export PATH 61 PATH usr loc
  • Django模型

    模型是你的数据的唯一的 权威的信息源 它包含你所储存数据的必要字段和行为 通常 xff0c 每个模型对应数据库中唯一的一张表 1 基础 每个模型都是django db models Model 的一个Python 子类 模型的每个属性都表示
  • 速度之王 — LZ4压缩算法(二)

    LZ4 Extremely Fast Compression algorithm 项目 xff1a http code google com p lz4 作者 xff1a Yann Collet 本文作者 xff1a zhangskd 64
  • dpkg

    dpkg error dpkg status database is locked by another process 无法获得锁 var lib apt lists lock open ubuntu升级错误或强制中断后容易爆出上面两个错
  • html5中加一个链接,HTML5教程—链接的添加方式_HTML5教程_链接添加_HTML5运用_课课家...

    HTML5的强大功能有很多 xff0c 在图像的修改中 xff0c 我们可见其强大 xff0c 然而其中有一个功能仍能可以运用于广告中的 xff0c 因为在广告主的需求中 xff0c 有很多情况下需要在动画中添加一些外部链接 而这份文档就在
  • Django--初始化

    1 Django介绍 它是一个WEB框架 Django 大而全tornado flask 小而精 2 Django安装 https www djangoproject com download 3 创建django程序 手动创建 file
  • ubuntu更换源后报错:W: GPG error: (转载)

    From xff1a http www njava com njava 626 html 更换163源后 xff0c 更新源时出现错误 apt get update W GPG error http extras ubuntu com pr
  • 魔咒词典

    题目描述 哈利波特在魔法学校的必修课之一就是学习魔咒 据说魔法世界有100000种不同的魔咒 xff0c 哈利很难全部记住 xff0c 但是为了对抗强敌 xff0c 他必须在危急时刻能够调用任何一个需要的魔咒 xff0c 所以他需要你的帮助