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. JS 多数组中取最大数组,分组

    代码小结,使用 JS 对数据进行分组,对分组中的数组取最大数组: var combineTemplate = JSON.parse(data.combineTemplate); //根据仪器种类ID分 ...

  2. pycharm设置开发模板/字体大小/背景颜色(3)

    一.pycharm设置字体大小/风格 选择 File –> setting –> Editor –> Font ,可以看到如上界面,可以根据自己的喜好随意调整字体大小,字体风格,文字 ...

  3. GWAS中的名称概念

    基因: 是指决定生物某一遗传性状的染色体DNA片段 基因型: `基因型`又称`遗传型`,是某一生物个体全部基因组合的总称.它反应生物体的遗传构成,即从双亲获得的全部基因的总和.遗传学中具体使用的基因型 ...

  4. QT 源码分析--1

    Ref: http://blog.sina.com.cn/s/blog_6e80f1390100qoc0.html 安装qt之后(我使用的是online自动安装),安装目录下有\5.10.1\Src\ ...

  5. From 虚拟机模板 创建单节点K8S1.14.1的操作步骤

    半年前总结的 还是有记不住的地方... 1. 根据上一篇blog 自己创建了一个虚拟机 里面包含 k8s1.14.1 的k8s集群 这里简单说一下 虚拟机开机之后 如何处理以能够使用k8s 的简单过程 ...

  6. HDOJ-1100 Trees made to order

    一.题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1100 二.题目分析 对二叉树的所有形态顺序编号,编号规则是:节点数越多的编号越大:节点数相等,左子 ...

  7. 如何创建etcd双向通信证书

    # 安装证书生成软件 wget https://pkg.cfssl.org/R1.2/cfssl_linux-amd64 -O /usr/bin/cfssl wget https://pkg.cfss ...

  8. notepad++安装markdown

    notepad++ 安装markdown安装markdown插件一.下载最新的markdown插件, github:https://github.com/nea/MarkdownViewerPlusP ...

  9. 【HC89S003F4开发板】9ASM写定时器1

    HC89S003F4开发板ASM写定时器1 一.实现过程 1.外部寄存器设置 扩展 XSFR 采用和 XRAM 同样的访问方式,使用 MOVX A, @DPTR 和 MOVX @DPTR ,A 来进行 ...

  10. 怎样获取全局对象 window

    1. 使用window.self window.self === window; // true 2. 使用window.window window.window === window; // tru ...