今天在写AVL删除的时候犯了一个傻逼错误,调了很久,教训惨痛,引以为鉴。

树中允许有重复节点,如果删除的节点有重复,则只删除1个。

AVL删除采取的方法是首先判断待删除节点是否存在,如果存在,那么判断该节点是否有两个儿子,如果有,则找到它左子树中的最大节点,设其值为t,将之删除,并将它的值去覆盖待删除节点。此处删除函数是带返回值的,为删除的节点的值。但是如果树中节点可以重复出现,碰巧待删除节点有多个,那么这些节点的值可能有多个被t覆盖,导致发生错误。

那么怎么避免这一错误呢?

当然我们可以把重复的节点弄成一个节点,用一个cnt来标记它出现的次数。

或者,在将t覆盖带删除节点,保证只覆盖一次。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
using namespace std;
#define MAXN 100055
int root,tot=0,n,m;
char opt[10];
struct node
{int ch[2],val,h,sz;
}tree[MAXN];
void update(int r)
{
tree[r].h=max(tree[tree[r].ch[0]].h,tree[tree[r].ch[1]].h)+1;
tree[r].sz=tree[tree[r].ch[0]].sz+tree[tree[r].ch[1]].sz+1;
}
void zig(int &r) //鍙虫棆
{
int t=tree[r].ch[0];
if(t==0)return;
tree[r].ch[0]=tree[t].ch[1];
tree[t].ch[1]=r;
update(r);
update(t);
r=t;
}
void zag(int &r)
{
int t=tree[r].ch[1];
if(t==0)return;
tree[r].ch[1]=tree[t].ch[0];
tree[t].ch[0]=r;
update(r);
update(t);
r=t;
}
void zigzag(int &r)
{
zig(tree[r].ch[1]);
zag(r);
}
void zagzig(int &r)
{
zag(tree[r].ch[0]);
zig(r);
}
bool find(int r,int x)
{
if(r==0)return 0;
if(tree[r].val==x)
return 1;
else if(tree[r].val<x)
return find(tree[r].ch[1],x);
else
return find(tree[r].ch[0],x);
}
void insert(int &r,int x)
{
if(r==0)
{tree[++tot].val=x;
tree[tot].sz=1;
tree[tot].h=1;
r=tot;
return;
}
if(x<=tree[r].val)
{
insert(tree[r].ch[0],x);
if(tree[tree[r].ch[0]].h==tree[tree[r].ch[1]].h+2)
{
if(x<=tree[tree[r].ch[0]].val)
zig(r);
else
zagzig(r);
}
}
else
{
insert(tree[r].ch[1],x);
if(tree[tree[r].ch[1]].h==tree[tree[r].ch[0]].h+2)
{
if(x>tree[tree[r].ch[1]].val)
zag(r);
else
zigzag(r);
}
}
update(r); }
void maintain(int &r)
{
if(r==0)return;
if(tree[tree[r].ch[0]].h==tree[tree[r].ch[1]].h+2)
{
int t=tree[r].ch[0];
if(t==0)return;
if(tree[tree[t].ch[0]].h==tree[tree[r].ch[1]].h+1)
zig(r);
else zagzig(r);
}
else if(tree[tree[r].ch[1]].h==tree[tree[r].ch[0]].h+2)
{
int t=tree[r].ch[1];
if(t==0)return;
if(tree[tree[t].ch[1]].h==tree[tree[r].ch[0]].h+1)
zag(r);
else zigzag(r);
}
update(r);
}
int del(int &r,int x)
{
int temp;
if(tree[r].val==x||(tree[r].val<x&&tree[r].ch[1]==0)||(tree[r].val>x&&tree[r].ch[0]==0))
{
if(tree[r].ch[1]==0||tree[r].ch[0]==0)
{
temp=tree[r].val;
r=tree[r].ch[1]+tree[r].ch[0];
return temp;
}
else
{
temp=tree[r].val; //注意这里,这样保证待删除节点只删掉了一个。
tree[r].val=del(tree[r].ch[0],x);
}
}
else if(tree[r].val<x)
temp= del(tree[r].ch[1],x);
else temp=del(tree[r].ch[0],x);
maintain(r);
return temp;
}
int query(int r,int rk)//鎵炬帓鍚嶄负绗瑀k鐨勫厓绱?
{
if(r==0)return 0;
int t=tree[r].ch[0];
if(tree[t].sz>=rk)
return query(tree[r].ch[0],rk);
else if(tree[t].sz+1<rk)
return query(tree[r].ch[1],rk-tree[t].sz-1);
else return tree[r].val;
}
int main()
{
int t,u,rnk=0;
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
{
scanf("%d",&t);
insert(root,t);
}
rnk=n;
for(int i=0;i<m;i++)
{
scanf("%s",opt);
if(opt[0]=='I')
{
rnk++;
scanf("%d",&t);
insert(root,t);
}
else if(opt[0]=='D')
{
scanf("%d",&t);
if(find(root,t))
{
del(root,t);
rnk--;
}
}
else if(opt[0]=='M')
{
scanf("%d%d",&t,&u);
if(find(root,t))
{
del(root,t);
insert(root,u);
}
}
else if(opt[0]=='Q')
{
printf("%d\n",query(root,(rnk+1)/2));
}
}
return 0;
}

  

AVL的删除写法的一个错误的更多相关文章

  1. 关于Java中的继承和组合的一个错误使用的例子

    [TOC] 关于Java中的继承和组合的一个错误使用的例子 相信绝大多数人都比较熟悉Java中的「继承」和「组合」这两个东西,本篇文章就主要就这两个话题谈论一下.如果我某些地方写的不对,或者比较幼稚, ...

  2. Vim安装jedi-vim提示的一个错误

    (仅为了提醒自己) 第一次的安装方法好像是通过 bundle安装的,好像是通过这个安装的并不是最新的版本,然后删除了通过下面的方法,最重要的是要执行 git submodule update --in ...

  3. 双缓冲绘图和窗口控件的绘制——ATL ActiveX 窗口控件生成向导绘制代码OnDraw的一个错误 .

    双缓冲绘图和窗口控件的绘制 ---ATL ActiveX 窗口控件生成向导绘制代码OnDraw的一个错误 cheungmine 我们通常使用ATL COM组件,生成一个带窗口的ActiveX控件,然后 ...

  4. [20180904]工作中一个错误.txt

    [20180904]工作中一个错误.txt --//昨天看我提交一份修改建议,发现自己写的sql语句存在错误.--//链接:http://blog.itpub.net/267265/viewspace ...

  5. MySQL删除数据库时的错误(errno: 39)

    由于mysql数据库是默认区分大小写的,部署的时候发现多了一些重复的表,于是就把多余的表删掉了.可是,剩下的重复的表再删除时会提示:表不存在. 于是,想把数据库删掉重新创建,可是,得到了 ERROR ...

  6. makefile的一个错误:*** missing separator

    原文转自:http://blog.sina.com.cn/s/blog_87c063060101c9yp.html 1.在写 多目录下makefile的时候,碰到一个错误提示,让我纠结许久,后面还是解 ...

  7. RDP 协议组件 X.224 在协议流中发现一个错误并且中断了客户端连接

    如果你的服务器有如下错误: “RDP 协议组件 X.224 在协议流中发现一个错误并且中断了客户端连接.” 可能的有2种: 1:你试试能否能继续远程登陆,有可能你的远程登陆组件出现问题. 2:有人攻击 ...

  8. git 如何删除远程仓库的错误提交

    前言 最近一个版本发生产环境以后,忘了把分支切回开发分支,直接在release分支上开发新功能提交了....于是就需要去删除远程仓库的错误提交. git命令行实现 1.强制返回上次的版本(~1回退到上 ...

  9. C#反序列化XML异常:在 XML文档(0, 0)中有一个错误“缺少根元素”

    Q: 在反序列化 Xml 字符串为 Xml 对象时,抛出如下异常. 即在 XML文档(0, 0)中有一个错误:缺少根元素. A: 首先看下代码: StringBuilder sb = new Stri ...

随机推荐

  1. Android斗地主棋牌游戏牌桌实现源码下载

    本次给大家分享下Android斗地主棋牌游戏牌桌实现源码下载如下: 为了节约内存资源,每张扑克牌都是剪切形成的,当然这也是当前编程的主流方法. 1.主Activity package com.biso ...

  2. JAVA 布尔型的应用

    定义一个布尔类型的flag, 只是当做一个开关的意思.先定义一个标记,然后根据一些条件给这个flag赋不同的值,最后,再根据这个flag不同的值,做不同的处理.   public static voi ...

  3. SpringMvc静态资源加载出错

    使用mvc:resource配置 web.xml配置是rest风格的/ 服务器启动没问题 访问地址是报404 另外用了default-servlet的方法加载,服务器启动没错,jsp页面加载静态资源要 ...

  4. 探索软件工程道路上的我 IV (Θ∀Θ#)

    开发语言:Java 开发工具:UltraEdit 小伙伴博客:http://www.cnblogs.com/hyating/ github地址:https://github.com/JUNYU217/ ...

  5. [翻译]为什么IIS应用程序池回收时间默认被设置为1740分钟?

    作者:斯科特 福赛斯/Scott Forsyth日期:2013/04/06地址:http://weblogs.asp.net/owscott/why-is-the-iis-default-app-po ...

  6. Python 基礎 - 元組與簡易購物車實做

    tuple(元組) 其實跟列表差不多,也是存一組數,只不過是它一旦建立了,就不能修改了,只能做 切片 跟 查詢,所以只叫 只讀列表 語法: name = ("Rogers", &q ...

  7. 升级OpenSSh到 7.3p1

    1.开启 telnet 服务 Linux yum install -y telnet-server telnet /etc/xinet.d/telnet 中的yes 修改为no service xin ...

  8. hping原理、安装、使用详解介绍

    [原理基础]  Hping是一个命令行下使用的TCP/IP数据包组装/分析工具,其命令模式很像Unix下的ping命令,但是它不是只能发送ICMP回应请求,它还可以支持TCP.UDP.ICMP和RAW ...

  9. 《JavaScript DOM编程艺术》读后总结

    这是我读的第一本关于JS的书,刚开始我为了选择合适的学习JS的书,看了网上许多人的意见,基本上都是推荐先读这本书.书的内容挺简单的,确实很适合初学者看,阅读的过程中不会产生什么障碍.内容虽然简单,但我 ...

  10. 第六章第一个linux个程序:统计单词个数

    第六章第一个linux个程序:统计单词个数 从本章就开始激动人心的时刻——实战,去慢慢揭开linux神秘的面纱.本章的实例是统计一片文章或者一段文字中的单词个数.  第 1 步:建立 Linu x 驱 ...