散列相关问题

1029 旧键盘 (20 分)

旧键盘上坏了几个键,于是在敲一段文字的时候,对应的字符就不会出现。现在给出应该输入的一段文字、以及实际被输入的文字,请你列出肯定坏掉的那些键。

输入格式:

输入在 2 行中分别给出应该输入的文字、以及实际被输入的文字。每段文字是不超过 80 个字符的串,由字母 A-Z(包括大、小写)、数字 0-9、以及下划线 _(代表空格)组成。题目保证 2 个字符串均非空。

输出格式:

按照发现顺序,在一行中输出坏掉的键。其中英文字母只输出大写,每个坏键只输出一次。题目保证至少有 1 个坏键。

#include <cstdio>

#include <vector>

#include <map>

using namespace std;

map<char,int> m;

int main()

{

vector <char> keys;//存坏键

char a[111],b[111];

scanf("%s %s",a,b);

int i=0,j=0;

while(a[i]!='\0')//two pointers

{

if(a[i]==b[j])//没坏

{

i++;j++;

}else//坏了

{

if(m[a[i]]==0)//之前没遇到

{

if(a[i]>='a'&&a[i]<='z')//如果是小写记录下并将对应大写设为已坏

{

keys.push_back(a[i]-32);

m[a[i]-32]=1;

}else keys.push_back(a[i]);//其余的直接记录

if(a[i]>='A'&&a[i]<='Z') m[a[i]+32]=1;//大写字母也要将对应小写设置

m[a[i]]=1; //统一设置为已坏

}

i++;//只移动输入的字符串的指针

}

}

vector <char>::iterator p=keys.begin();

int r=keys.size();

for(i=0;i<r;i++) printf("%c",p[i]);

return 0;

}

1033 旧键盘打字 (20 分)

旧键盘上坏了几个键,于是在敲一段文字的时候,对应的字符就不会出现。现在给出应该输入的一段文字、以及坏掉的那些键,打出的结果文字会是怎样?

输入格式:

输入在 2 行中分别给出坏掉的那些键、以及应该输入的文字。其中对应英文字母的坏键以大写给出;每段文字是不超过 10​5​​ 个字符的串。可用的字符包括字母 [a-z, A-Z]、数字 0-9、以及下划线 _(代表空格)、,、.、-、+(代表上档键)。题目保证第 2 行输入的文字串非空。

注意:如果上档键坏掉了,那么大写的英文字母无法被打出。

#include <stdio.h>

#include <string.h>

int a[256];

char b[100010];

int main()

{

memset(a,1,sizeof(a));//初始化

gets(b);

int l=strlen(b);

for(int i=0;i<l;i++)

{

if(b[i]>='A'&&b[i]<='Z') b[i]=b[i]-'A'+'a';//化为小写

a[b[i]]=0;//置0

}

gets(b);

l=strlen(b);

for(int i=0;i<l;i++)

{

if(b[i]>='A'&&b[i]<='Z')

{

int low=b[i]-'A'+'a';

if(a[low]&&a['+'])  printf("%c",b[i]);

}else if(a[b[i]]) printf("%c",b[i]);

}

printf("\n");

return 0;

}

1038 统计同成绩学生 (20 分)
1039 到底买不买 (20 分)

大致的思路的差不多,建立表存入信息再根据输入判断或更新

1042 字符统计 (20 分)
1043 输出PATest (20 分)

这两题都是统计特定字符

#include <cstdio>

int s[6]={0};

char ss[6]={'P','A','T','e','s','t'};//按顺序存个数

using namespace std;

int main()

{

char c;

c=getchar();

while(c!='\n')

{

switch (c)//记录

{

case 'P': s[0]++;break;

case 'A': s[1]++;break;

case 'T': s[2]++;break;

case 'e': s[3]++;break;

case 's': s[4]++;break;

case 't': s[5]++;break;

}

c=getchar();

}

while(true)

{

int flag=0;

for(int i=0;i<6;i++)//按顺序输出

{

if(s[i])

{

printf("%c",ss[i]);s[i]--;

}else flag++;

}

if(flag==6) break;

}

return 0;

}

1047 编程团体赛 (20 分)
1059 C语言竞赛 (20 分)

C 语言竞赛是浙江大学计算机学院主持的一个欢乐的竞赛。既然竞赛主旨是为了好玩,颁奖规则也就制定得很滑稽:

  • 0、冠军将赢得一份“神秘大奖”(比如很巨大的一本学生研究论文集……)。
  • 1、排名为素数的学生将赢得最好的奖品 —— 小黄人玩偶!
  • 2、其他人将得到巧克力。

给定比赛的最终排名以及一系列参赛者的 ID,你要给出这些参赛者应该获得的奖品。

输入格式:

输入第一行给出一个正整数 N(≤10​4​​),是参赛者人数。随后 N 行给出最终排名,每行按排名顺序给出一位参赛者的 ID(4 位数字组成)。接下来给出一个正整数 K 以及 K 个需要查询的 ID。

输出格式:

对每个要查询的 ID,在一行中输出 ID: 奖品,其中奖品或者是 Mystery Award(神秘大奖)、或者是 Minion(小黄人)、或者是Chocolate(巧克力)。如果所查 ID 根本不在排名里,打印 Are you kidding?(耍我呢?)。如果该 ID 已经查过了(即奖品已经领过了),打印 ID: Checked(不能多吃多占)。

#include <stdio.h>

#include <math.h>

using namespace std;

int id[10000]={0};

bool su(int k)//判断素数

{

if(k==2) return true;

for(int i=2;i<=(int)sqrt(k)+1;i++)

{

if(k%i==0) return false;

}

return true;

}

void judge(int i)//判断得奖

{

if(id[i]==0)//不存在

{

printf("%04d: Are you kidding?\n",i);return;

}

if(id[i]<0) printf("%04d: Checked\n",i);//已经出现过

else if(id[i]==1)//第一

{

printf("%04d: Mystery Award\n",i);

}else if(su(id[i]))//小黄人

{

printf("%04d: Minion\n",i);

}else//巧克力

{

printf("%04d: Chocolate\n",i);

}

id[i]=-1;//记为已查

}

int main()

{

int x,n,k;

scanf("%d",&n);

for(int i=1;i<=n;i++)

{

scanf("%d",&x);

id[x]=i;

}

scanf("%d",&k);

while(k--)

{

scanf("%d",&x);

judge(x);

}

return 0;

}

1065 单身狗 (25 分)

“单身狗”是中文对于单身人士的一种爱称。本题请你从上万人的大型派对中找出落单的客人,以便给予特殊关爱。

输入格式:

输入第一行给出一个正整数 N(≤ 50 000),是已知夫妻/伴侣的对数;随后 N 行,每行给出一对夫妻/伴侣——为方便起见,每人对应一个 ID 号,为 5 位数字(从 00000 到 99999),ID 间以空格分隔;之后给出一个正整数 M(≤ 10 000),为参加派对的总人数;随后一行给出这 M 位客人的 ID,以空格分隔。题目保证无人重婚或脚踩两条船。

输出格式:

首先第一行输出落单客人的总人数;随后第二行按 ID 递增顺序列出落单的客人。ID 间用 1 个空格分隔,行的首尾不得有多余空格。

#include <stdio.h>

#include <set>

using namespace std;

int cp[100000]={-1},vistor[10010]={-1};//初始化

int main()

{

int n,m,id1,id2;

set<int> single;//存入单身狗

scanf("%d",&n);

for(int i=0;i<n;i++)

{

scanf("%d %d",&id1,&id2);

cp[id1]=id2;//每一对互相映射

cp[id2]=id1;

}

scanf("%d",&m);

for(int i=0;i<m;i++)

{

scanf("%d",&vistor[i]);

if(cp[vistor[i]]==-1) single.insert(vistor[i]);//绝对单身直接记录

else cp[vistor[i]]+=100000;//有对象但不知道对象来没来,所以做标记

}

for(int i=0;i<m;i++)

{

if(cp[vistor[i]]>=100000)//有对象的

{

if(cp[cp[vistor[i]]-100000]>=100000)//根据映射,检验其对象是否来了

{

cp[vistor[i]]-=100000;

cp[cp[vistor[i]]]-=100000;

}

else single.insert(vistor[i]);//没来的话就算单身(?黑人问号脸?)

}

}

printf("%d\n",single.size());

for(set<int>::iterator it=single.begin();it!=single.end();it++)

{

if(it==single.begin()) printf("%05d",*it);

else printf(" %05d",*it);

}

return 0;

}

1068 万绿丛中一点红 (20 分)

对于计算机而言,颜色不过是像素点对应的一个 24 位的数值。现给定一幅分辨率为 M×N 的画,要求你找出万绿丛中的一点红,即有独一无二颜色的那个像素点,并且该点的颜色与其周围 8 个相邻像素的颜色差充分大。

输入格式:

输入第一行给出三个正整数,分别是 M 和 N(≤ 1000),即图像的分辨率;以及 TOL,是所求像素点与相邻点的颜色差阈值,色差超过 TOL 的点才被考虑。随后 N 行,每行给出 M 个像素的颜色值,范围在 [0,2​24​​) 内。所有同行数字间用空格或 TAB 分开。

输出格式:

在一行中按照 (x, y): color 的格式输出所求像素点的位置以及颜色值,其中位置 x 和 y 分别是该像素在图像矩阵中的列、行编号(从 1 开始编号)。如果这样的点不唯一,则输出 Not Unique;如果这样的点不存在,则输出 Not Exist。

#include <iostream>

#include <map>

#include <algorithm>

using namespace std;

int color[1010][1010]={0};

int m,n,tol;

struct sp

{

int co;

int x,y;

}ans;

map<int,int> num;

bool judge(int i,int j)

{

for(int y=max(1,i-1);y<=min(i+1,n);y++)

{

for(int x=max(1,j-1);x<=min(m,j+1);x++)

{

if(x!=j&&i!=y)

{

if(color[i][j]-color[y][x]<=tol&&color[i][j]-color[y][x]>=-tol) return false;

}

}

}

return true;

}

int main()

{

cin>>m>>n>>tol;

for(int i=1;i<=n;i++)

{

for(int j=1;j<=m;j++)

{

cin>>color[i][j];num[color[i][j]]++;

}

}

int count=0;

for(int i=1;i<=n;i++)

{

for(int j=1;j<=m;j++)

{

if(judge(i,j))

{

if(num[color[i][j]]==1)

{

ans.co=color[i][j];ans.x=j;ans.y=i;

count++;

}

}

}

}

if(count>1) cout<<"Not Unique";

else if(count==0) cout<<"Not Exist";

else cout<<'('<<ans.x<<", "<<ans.y<<"): "<<ans.co;

return 0;

}

第三个点老是不过,实在不知道哪里有问题了。。

1072 开学寄语 (20 分)1082 射击比赛 (20 分)1083 是否存在相等的差 (20 分)
记录并统计,一般的方法
1090 危险品装箱 (25 分)

集装箱运输货物时,我们必须特别小心,不能把不相容的货物装在一只箱子里。比如氧化剂绝对不能跟易燃液体同箱,否则很容易造成爆炸。

本题给定一张不相容物品的清单,需要你检查每一张集装箱货品清单,判断它们是否能装在同一只箱子里。

输入格式:

输入第一行给出两个正整数:N (≤10​4​​) 是成对的不相容物品的对数;M (≤100) 是集装箱货品清单的单数。

随后数据分两大块给出。第一块有 N 行,每行给出一对不相容的物品。第二块有 M 行,每行给出一箱货物的清单,格式如下:

K G[1] G[2] ... G[K]

其中 K (≤1000) 是物品件数,G[i] 是物品的编号。简单起见,每件物品用一个 5 位数的编号代表。两个数字之间用空格分隔。

输出格式:

对每箱货物清单,判断是否可以安全运输。如果没有不相容物品,则在一行中输出 Yes,否则输出 No。

和单身狗那一题不一样,这里可能存在多对一的情况,如果直接储存会使得数据覆盖。我先是用字符串存的map,虽然不错,但是判断时是遍历所有的情况,果然超时了。参考了大神方法,于是为了解决一对多的问题,映射到vector,原本想到用数组,但是太大。然后判断时可以直接跳过不存在于不相容列表的物品,再一一比较,虽然有多重循环,却没有超时。

#include <iostream>

#include <map>

#include <vector>

using namespace std;

int n,m,k;

map<string,vector<string> > cp;

string temp[10010];

bool judge()

{

for(int i=0;i<k;i++)

{

if(cp[temp[i]].size()>0)

{

for(int j=i+1;j<k;j++)

{

for(int l=0;l<cp[temp[i]].size();l++)

{

if(temp[j]==cp[temp[i]][l]) return false;

}

}

}

}

return true;

}

int main()

{

cin>>n>>m;

string a,b;

for(int i=0;i<n;i++)

{

cin>>a>>b;

cp[a].push_back(b);

cp[b].push_back(a);

}

for(int i=0;i<m;i++)

{

cin>>k;

for(int j=0;j<k;j++) cin>>temp[j];

if(judge()) cout<<"Yes"<<endl;

else cout<<"No"<<endl;

}

return 0;

}

PAT乙级考前总结(四)的更多相关文章

  1. PAT乙级考前总结(一)

    数学相关的题目 1001 害死人不偿命的(3n+1)猜想 (15 分) 直接一步步计数 1005 继续(3n+1)猜想 (25 分) 卡拉兹(Callatz)猜想已经在1001中给出了描述.在这个题目 ...

  2. PAT乙级考前总结(三)

    特殊题型 1027 打印沙漏 (20 分) 题略,感觉有点像大学里考试的题.找规律即可. #include <stdio.h>#include <iostream>using ...

  3. PAT乙级考前总结(五)

    字符串处理 1003 我要通过! (20 分) “答案正确”是自动判题系统给出的最令人欢喜的回复.本题属于 PAT 的“答案正确”大派送 —— 只要读入的字符串满足下列条件,系统就输出“答案正确”,否 ...

  4. PAT乙级考前总结(二)

    简单模拟 1002 写出这个数 (20 分) 1006 换个格式输出整数 (15 分) 又是数数 1016 部分A+B (15 分) 相当于找数字 1018 锤子剪刀布 (20 分) 题目略 此处用了 ...

  5. PAT乙级:1070 结绳 (25分)

    PAT乙级:1070 结绳 (25分) 题干 给定一段一段的绳子,你需要把它们串成一条绳.每次串连的时候,是把两段绳子对折,再如下图所示套接在一起.这样得到的绳子又被当成是另一段绳子,可以再次对折去跟 ...

  6. C#版 - PAT乙级(Basic Level)真题 之 1021.个位数统计 - 题解

    版权声明: 本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. C#版 - P ...

  7. PAT乙级真题及训练题 1025. 反转链表 (25)

    PAT乙级真题及训练题 1025. 反转链表 (25) 感觉几个世纪没打代码了,真是坏习惯,调了两小时把反转链表调出来了,心情舒畅. 这道题的步骤 数据输入,数组纪录下一结点及储存值 创建链表并储存上 ...

  8. PAT 乙级 1024

    题目 题目地址:PAT 乙级 1024 题解 模拟题,重点需要考虑到各种不同情况:简单来说一下: 因为输入格式固定,所以把不同的部分分别存储和处理可以在很大程度上简化运算:其中需要考虑最多的就是小数部 ...

  9. PAT 乙级 1017

    题目 题目地址:PAT 乙级 1017 题解 粗看是一道大数除法题,实际上只不过是通过字符数组模拟除法过程,理解之后还是比较简单的: 具体分析一下本题: 因为题设中的除数(n)是一位整数,因此大幅简化 ...

随机推荐

  1. Ubuntu18.04中安装vsftpd服务/ ftp上传文件提示无权限 553 Could not create file.

    1,安装 $ sudo apt-get install vsftpd 2.配置 备份并创建新的配置文件. $ sudo mv /etc/vsftpd.conf /etc/vsftpd.conf_ori ...

  2. sql排列组合

    一个表中4条记录,如何查询所有可能组成的结果. 例如: 1  2  3  4 四条记录,最后组成(12),(13),(14),(23),(24),(34) 最后sql如下: SELECT a.name ...

  3. Dubbo源码构建

    代码签出 通过以下的这个命令签出最新的项目源码: git clone https://github.com/apache/incubator-dubbo.git dubbo 分支 我们使用 maste ...

  4. d3js可视化策略

    d3js是数据驱动图形的思路.基本可以这么理解,有什么样的图形,后面基本就有类似结构的数据.大概思路步骤如下: 一.适配数据格式 这一步主要是为第二部服务,第一步的结果作为第二部的入参. 比如,画层级 ...

  5. java算法03 - 常用的8种排序算法

    Java常用的八种排序算法: 插入排序 - 直接插入排序 每次将待排序的记录按照关键字的大小,插入到前面已经排好序的记录的适当位置.直到全部记录插入完成. 代码实现 /** * 直接插入排序 O(n^ ...

  6. springboot入门学习-helloworld实例

    今天学习了springboot,发现使用springboot创建项目确实非常方便,创建第一个springboot项目之前先看一下下面两个问题. 一.什么是springboot? Spring Boot ...

  7. DevExpress VCL Controls 2019发展路线图(No.2)

    [DevExpress VCL Controls下载] ExpressQuantumTreeList Excel-inspired Filter (v19.1) 与ExpressQuantumGrid ...

  8. 如何将Excel转换成Markdown表格[转]

    在这篇文章中,我将告诉你如何快速的将Excel转换为markdown表格,以及如何将Google Docs,Numbers,网页中的表格或其他类似Excel的程序数据转换为Markdown表格 你可能 ...

  9. ES6标准入门读书笔记

    第一章  基础 1.let和const命令 (1).let用于声明变量,所声明的变量只在当前代码块有效 特点:不存在变量提升     所以在变量声明之前就使用会报错 暂时性死区           只 ...

  10. JSP学习(1)---JSP基本原理

    一.JSP的本质 其本质是Servlet,web应用中的每个jsp页面都会由servlet容器生成对应的servlet. 在tomcat中,jsp生成的servlet在work文件夹下: 原jsp文件 ...