L2-012. 关于堆的判断

将一系列给定数字顺序插入一个初始为空的小顶堆H[]。随后判断一系列相关命题是否为真。命题分下列几种:

  • “x is the root”:x是根结点;
  • “x and y are siblings”:x和y是兄弟结点;
  • “x is the parent of y”:x是y的父结点;
  • “x is a child of y”:x是y的一个子结点。
  • (仔细观察,发现只有第二句第二个单词是“and”,除了第二个之外的句子全是第四个单词一定可以区分出来!!此中必有隐情吧,说不好出题人就是这么造数据的!!)

输入格式:

每组测试第1行包含2个正整数N(<= 1000)和M(<= 20),分别是插入元素的个数、以及需要判断的命题数。下一行给出区间[-10000, 10000]内的N个要被插入一个初始为空的小顶堆的整数 (这里是重点,所以每次读入一个新数据存入堆里后就进行调整一次)。之后M行,每行给出一个命题。题目保证命题中的结点键值都是存在的。

输出格式:

对输入的每个命题,如果其为真,则在一行中输出“T”,否则输出“F”。

输入样例:

5 4
46 23 26 24 10
24 is the root
26 and 23 are siblings
46 is the parent of 23
23 is a child of 10

输出样例:

F
T
F
T

AC题解:(开数组模拟小顶堆即可)
提醒:判断x和y是兄弟节点时,要注意y和x可能也是的;
  注意越界,枚举父节点时,跑n/2即可(具体原因参考二叉树的性质)!
  我写的debug()函数,输出堆的时候就是按照小顶堆的层序遍历来的!
  其实用链表写着更省内存!开数组需要多开一些(一倍吧,具体自己证明多少合适!)不然会发生段错误的当数据量取上限的时候,N=1000是不够的(我试了试--段错误),见构造函数的第20行代码:“if(heap[i<<1]==-1)return ;”
  因为每次访问到根节点,我用的判断方法是判断其是否还有左孩子节点!!所以————
#include<iostream>
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<algorithm>
#include<queue>
#include<set>
#include<vector>
#include<string>
#include<map>
#define maxn 400000
#define inf 0x3f3f3f3f //L2-012. 关于堆的判断
using namespace std;
#define N 2008
int heap[N];
int flag;
int n;
void min_heap_adjust(int i){//原则:把小的元素调整上去
if(heap[i]==-)return ;
if(heap[i<<]==-)return ;
int l=i<<,r=i<<|;
if(heap[r]==-){
if(heap[i]>heap[l]){
swap(heap[i],heap[l]);
}
}else{
int minn=heap[i],j=;
if(heap[l]<minn){
j=l;minn=heap[l];
}
if(heap[r]<minn){
j=r;
}
if(j!=){
swap(heap[j],heap[i]);
min_heap_adjust(j);
}
}
}
int find_siblings(int i,int x,int y){//判断x和y是否是兄弟节点
if(flag)return ;
if(i>n/)return ;
int l=i<<,r=i<<|;
if(heap[l]==-||heap[r]==-)
return ;
if((heap[l]==x&&heap[r]==y) || (heap[r]==x&&heap[l]==y))
return flag=;
find_siblings(i<<,x,y);
find_siblings(i<<|,x,y);
return ;
}
int find_parent(int i,int x,int y){//判断x是y的父亲
if(flag)return ;
if(i>n/)return ;
int l=i<<,r=i<<|;
if(heap[i]==x){
if(heap[l]!=-){
if(heap[l]==y)return flag=;
}
if(heap[r]!=-){
if(heap[r]==y)return flag=;
}
}
find_parent(i<<,x,y);
find_parent(i<<|,x,y);
return ;
} void solve(int n){
int x,y;
flag=;//初始化一次标志数
scanf("%d",&x);
char s[];
scanf("%s",s);
if(s[]=='a'){
scanf("%d%*s%*s",&y);//求x和y是否兄弟节点
find_siblings(,x,y);
}else{
scanf("%*s%s",s);
if(s[]=='r'){
if(heap[]==x)flag=;
}
else if(s[]=='p'){//判断x是y的父亲
scanf("%*s%d",&y);
find_parent(,x,y);
}
else{
scanf("%*s%d",&y);
find_parent(,y,x);
}
}
if(flag==)printf("T\n");
else printf("F\n"); return ;
}
void debug(){
for(int i=;i<=n;i++)
printf("%d ",heap[i]);
cout<<endl;
}
int main(){
int m;
int num;
while(scanf("%d%d",&n,&m)!=EOF){
memset(heap,-,sizeof(heap));
for(int i=;i<=n;i++){
scanf("%d",&num);
heap[i]=num;
for(int j=i/;j>=;j--){
min_heap_adjust(j);
}
} // debug();
for(int i=;i<=m;i++)
solve(n);
} return ;
}

【数组模拟-小顶堆的插入构造/遍历】PAT-L2-012.-关于堆的判断--数组模拟的更多相关文章

  1. 【小顶堆的插入构造/遍历】PatL2-012. 关于堆的判断

    L2-012. 关于堆的判断 时间限制   将一系列给定数字顺序插入一个初始为空的小顶堆H[].随后判断一系列相关命题是否为真.命题分下列几种: “x is the root”:x是根结点: “x a ...

  2. 面试题:给定一个长度为N的数组,其中每个元素的取值范围都是1到N。判断数组中是否有重复的数字

    题目:给定一个长度为N的数组,其中每个元素的取值范围都是1到N.判断数组中是否有重复的数字.(原数组不必保留) 方法1.对数组进行排序(快速,堆),然后比较相邻的元素是否相同.时间复杂度为O(nlog ...

  3. 堆排序(大顶堆、小顶堆)----C语言

    堆排序 之前的随笔写了栈(顺序栈.链式栈).队列(循环队列.链式队列).链表.二叉树,这次随笔来写堆 1.什么是堆? 堆是一种非线性结构,(本篇随笔主要分析堆的数组实现)可以把堆看作一个数组,也可以被 ...

  4. POJ 2442 - Sequence - [小顶堆][优先队列]

    题目链接:http://poj.org/problem?id=2442 Time Limit: 6000MS Memory Limit: 65536K Description Given m sequ ...

  5. POJ 1456 - Supermarket - [贪心+小顶堆]

    题目链接:http://poj.org/problem?id=1456 Time Limit: 2000MS Memory Limit: 65536K Description A supermarke ...

  6. 大顶堆与小顶堆应用---寻找前k小数

    vector<int> getLeastNumber(vector<int>& arr,int k){ vector<int> vec(k,); if(== ...

  7. HDU 4006The kth great number(K大数 +小顶堆)

    The kth great number Time Limit:1000MS     Memory Limit:65768KB     64bit IO Format:%I64d & %I64 ...

  8. BZOJ 1150 - 数据备份Backup - [小顶堆][CTSC2007]

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1150 Time Limit: 10 Sec Memory Limit: 162 M De ...

  9. 《排序算法》——堆排序(大顶堆,小顶堆,Java)

    十大算法之堆排序: 堆的定义例如以下: n个元素的序列{k0,k1,...,ki,-,k(n-1)}当且仅当满足下关系时,称之为堆. " ki<=k2i,ki<=k2i+1;或k ...

随机推荐

  1. Ubuntu 18.04 使用标准Ubuntu 仓库进行自动化安装NVIDIA驱动

    首先,检测你的NVIDIA显卡型号和推荐的驱动程序的模型.在命令行中输入如下命令: $ ubuntu-drivers devices == /sys/devices/pci0000:/::::00.0 ...

  2. 常见的序列化框架及Protobuf序列化原理

    原文链接:https://www.jianshu.com/p/657fbf347934 https://www.cnblogs.com/javazhiyin/p/11375553.html https ...

  3. 使用Docker-Compose编排发布.Net Core+Redis应用两个镜像到Docker

    对于刚刚完成的Alipay支的Demo, 我想要把它部署到Docker中去, 下面我来演示相关步骤. 创建配置文件 配置文件的重中之重是Dockerfile, 他的内容如下: # 第一部分是编译并发布 ...

  4. [bzoj5483][Usaco2018 Dec]Balance Beam_凸包_概率期望

    bzoj5483 Usaco2018Dec Balance Beam 题目链接:https://lydsy.com/JudgeOnline/problem.php?id=5483 数据范围:略. 题解 ...

  5. 关于epoll,select,poll的理解

    select: 轮询+fd_set 1.采用fd_set存储fd(fd_set通过数组位图实现) 2.每次调用select,都需要把fd集合从用户态拷贝到内核态,fd越多开销越大 3.每次调用sele ...

  6. JAVA线程中的发牌题

    发牌题主要考虑的就是线程的问题,一个buffer缓冲区的问题, 首先,发牌的优先级当然是最高的了,但是取牌不能有优先级,否则会一直有牌先取,因此需要一个信号量order,当order=线程的数字时,取 ...

  7. python学习-41 装饰器 -- 高阶函数

    装饰器:本质就是函数.是为其他函数添加附加功能的. 原则:1.不修改被修饰函数的源代码2.不修改被修饰函数的调用方式 --- 装饰器的知识储备 装饰器=高阶函数+函数嵌套+闭包 高阶函数 1.高阶函数 ...

  8. Eclipse设置每行的最大字符数

    Eclipse默认宽度是 120 个字符.如下图所示(提示:格式化快捷键Ctrl + Shift + F): 设置步骤如下: 菜单栏倒数第二项,选择Window 下拉栏最后一项,选择Preferenc ...

  9. ef core2.2 mysql迁移问题

    前段时间,遇到的是ef core mysql迁移的时候,bool类型会自动yingsheweishort的问题,需要手动更正一下今天测试的时候,遇到了MySQL数据表修改后迁移的问题. 问题详情如下  ...

  10. mysql8.0入坑体验

    正常从官网下载,并且正常安装,直到安装完成.然后用navicate连接,发现报错信息如下所示Client does not support authentication protocol reques ...