#include <stdlib.h>
#include <stdio.h>
#include <string.h>
typedef struct no{
int pontuacao, id;
struct no *dir, *esq, *pai;
}No;
No *cria_no(int nova_pontuacao, int ID){
No
*novo
= malloc(sizeof(No
)); novo->pontuacao = nova_pontuacao;
novo->dir = novo->esq = NULL;
novo->id = ID;
novo->pai = NULL;
return novo;
}
No *insere_no(No *raiz, int nova_pontuacao, int ID){
No *pai_aux = NULL;
No *aux = NULL;
No *nova = cria_no(nova_pontuacao, ID);
if (raiz == NULL)
raiz = nova;
else
{
aux = raiz;
while (aux != NULL)
{
pai_aux = aux;
if (nova_pontuacao < aux->pontuacao)
aux = aux->esq;
else
aux = aux->dir;
}
nova->pai = pai_aux;
if (nova_pontuacao > pai_aux->pontuacao)
pai_aux->dir = nova;
else
pai_aux->esq = nova;
}
return raiz;
}
void em_ordem(No *raiz)
{
if (raiz != NULL)
{
em_ordem(raiz->esq);
printf("%d ", raiz
->pontuacao
); em_ordem(raiz->dir);
}
}
void pre_ordem(No *raiz)
{
if (raiz != NULL)
{
printf("%d ", raiz
->pontuacao
); pre_ordem(raiz->esq);
pre_ordem(raiz->dir);
}
}
No* sucessor(No *no_remover)
{
// por sucessor
No *suc = no_remover->dir;
while (suc->esq != NULL)
{
suc = suc->esq;
}
return suc;
}
No* antecessor(No *no_remover)
{
// por sucessor
No *ant = no_remover->esq;
while (ant->dir != NULL)
{
ant = ant->dir;
}
return ant;
}
No *remove_noavl(No *raiz, int chave)
{
No *no_remover = raiz;
No *pai_no_remover = NULL;
No *copia = NULL;
while (no_remover != NULL && no_remover->pontuacao != chave)
{
if (no_remover->pontuacao < chave)
no_remover = no_remover->dir;
else
no_remover = no_remover->esq;
}
while (no_remover != NULL) // encontrou o valor a ser removido
{
if (no_remover->esq == NULL && no_remover->dir == NULL)//eh folha
{
if (raiz == no_remover)
{
no_remover = raiz = NULL;
}
else
{
if (no_remover->pai->esq != NULL && no_remover->pai->esq->pontuacao == chave)
no_remover->pai->esq = NULL;
else
no_remover->pai->dir = NULL;
no_remover = NULL;
}
}
else // tem um filho ou dois filhos
{
if (no_remover->dir != NULL)
copia = sucessor(no_remover);
else
copia = antecessor(no_remover);
//substitui o valor da ser removido pelo valor
//do sucessor ou do antecessor
no_remover->pontuacao = copia->pontuacao;
no_remover->id = copia->id;
//muda o no a ser removido para o sucessor ou antecessor
no_remover = copia;
chave = copia->pontuacao;
}
}
return raiz;
}
No* registrar(No *raiz){
int pontuacao;
int id;
return insere_no(raiz, pontuacao, id);
}
int verifica_existencia(No*raiz, int id, int *pontuacao_velha, int*flag){
if(raiz != NULL && (*flag) != 1){
verifica_existencia(raiz->dir, id, pontuacao_velha, flag);
if(raiz->id == id){
(*flag) = 1;
(*pontuacao_velha) = raiz->pontuacao;
}
verifica_existencia(raiz->esq, id, pontuacao_velha, flag);
}
}
No* UPDATE(No *raiz){
int pontuacao_nova, pontuacao_velha, id, flag = 0;
scanf("%d", &pontuacao_nova
);
if(raiz != NULL){
verifica_existencia(raiz, id, &pontuacao_velha, &flag);
if(flag == 1){
raiz = remove_noavl(raiz, pontuacao_velha);
raiz = insere_no(raiz, pontuacao_nova, id);
}
}
return raiz;
}
int ID_MAX(No *raiz){
No* aux = raiz;
while(aux->esq != NULL)
aux = aux->esq;
return aux->id;
}
void posicao_calc(No *raiz, int id, int *CAT, int *flag){
if(raiz != NULL && (*flag) != 1){
posicao_calc(raiz->dir, id, CAT, flag);
if(raiz->id == id){
(*flag) = 1;
(*CAT)++;
}
else if((*flag) != 1){
(*CAT)++;
}
posicao_calc(raiz->esq, id, CAT, flag);
}
}
void posicao(No *raiz){
int id = 0;
if(raiz != NULL){
int rez_01, rez_02, aux_id, pos;
rez_01 = rez_02 = aux_id = pos = 0;
aux_id = ID_MAX(raiz);
posicao_calc(raiz, id, &rez_01, &pos);
pos = 0;
posicao_calc(raiz, aux_id, &rez_02, &pos);
if(rez_01>rez_02)
else
}
else{
}
}
void top(No *raiz){
if(raiz != NULL){
No* aux = raiz, *pai_aux = NULL;
while(aux != NULL){
pai_aux = aux;
aux = aux->dir;
}
}
else{
}
}
void BOTTOM(No *raiz){
if(raiz != NULL){
No* aux = raiz, *pai_aux = NULL;
while(aux != NULL){
pai_aux = aux;
aux = aux->esq;
}
}
else{
}
}
void RANGE_CALC(No *raiz, int pont_min, int pont_max){
if(raiz != NULL){
RANGE_CALC(raiz->esq, pont_min, pont_max);
if(pont_min <= raiz->pontuacao && raiz->pontuacao <= pont_max){
}
RANGE_CALC(raiz->dir, pont_min, pont_max);
}
}
void RANGE(No *raiz){
int min;
int max;
min = max = 0;
scanf("%d %d", &min
, &max
); if(raiz != NULL){
RANGE_CALC(raiz, min, max);
}
else
}
void COUNT_ABOVE(No *raiz, int ac, int* cont_0x){
if (raiz != NULL)
{
COUNT_ABOVE(raiz->dir, ac, cont_0x);
if(raiz->pontuacao > ac)
(*cont_0x)++;
COUNT_ABOVE(raiz->esq, ac, cont_0x);
}
}
void COUNT_BELOW(No *raiz, int ac, int* cont_0x){
if (raiz != NULL)
{
COUNT_BELOW(raiz->dir, ac, cont_0x);
if(raiz->pontuacao < ac)
(*cont_0x)++;
COUNT_BELOW(raiz->esq, ac, cont_0x);
}
}
int main(){
No *ABB = NULL;
int n, cont = 0;
int ac = 0;
int cont_02 = 0;
char rep[20];
while(cont < n){
if(strcmp(rep
, "REGISTER") == 0) ABB = registrar(ABB);
else if(strcmp(rep
, "UPDATE") == 0) ABB = UPDATE(ABB);
else if(strcmp(rep
, "POSITION") == 0) posicao(ABB);
else if(strcmp(rep
, "RANGE") == 0) RANGE(ABB);
else if(strcmp(rep
, "TOP") == 0) top(ABB);
else if(strcmp(rep
, "BOTTOM") == 0) BOTTOM(ABB);
else if(strcmp(rep
, "COUNT_ABOVE") == 0){ ac = cont_02 = 0;
COUNT_ABOVE(ABB, ac, &cont_02);
}
else if(strcmp(rep
, "COUNT_BELOW") == 0){ ac = cont_02 = 0;
COUNT_BELOW(ABB, ac, &cont_02);
}
cont++;
}
return 0;}
#include <stdlib.h>
#include <stdio.h>
#include <string.h>

typedef struct no{
    int pontuacao, id;
    struct no *dir, *esq, *pai;
}No;

No *cria_no(int nova_pontuacao, int ID){
    No *novo = malloc(sizeof(No));
    novo->pontuacao = nova_pontuacao;
    novo->dir = novo->esq = NULL;
    novo->id = ID;
    novo->pai = NULL;
    return novo;
}

No *insere_no(No *raiz, int nova_pontuacao, int ID){
    No *pai_aux = NULL;
    No *aux = NULL;
    No *nova = cria_no(nova_pontuacao, ID);
    if (raiz == NULL)
        raiz = nova;
    else
    {
        aux = raiz;
        while (aux != NULL)
        {
            pai_aux = aux;
            if (nova_pontuacao < aux->pontuacao)
                aux = aux->esq;
            else
                aux = aux->dir;
        }
        nova->pai = pai_aux;
        if (nova_pontuacao > pai_aux->pontuacao)
            pai_aux->dir = nova;
        else 
            pai_aux->esq = nova;
    }
    return raiz;
}

void em_ordem(No *raiz)
{
    if (raiz != NULL)
    {
        em_ordem(raiz->esq);
        printf("%d ", raiz->pontuacao);
        em_ordem(raiz->dir);
    }
}

void pre_ordem(No *raiz)
{
    if (raiz != NULL)
    {
        printf("%d ", raiz->pontuacao);
        pre_ordem(raiz->esq);
        pre_ordem(raiz->dir);
    }
}

No* sucessor(No *no_remover)
{

    // por sucessor
    No *suc = no_remover->dir;

    while (suc->esq != NULL)
    {
        suc = suc->esq;
    }

    return suc;
}

No* antecessor(No *no_remover)
{

    // por sucessor
    No *ant = no_remover->esq;

    while (ant->dir != NULL)
    {
        ant = ant->dir;
    }

    return ant;
}

No *remove_noavl(No *raiz, int chave)
{

    No *no_remover = raiz;
    No *pai_no_remover = NULL;
    No *copia = NULL;

    while (no_remover != NULL && no_remover->pontuacao != chave)
    {
        if (no_remover->pontuacao < chave)
            no_remover = no_remover->dir;
        else
            no_remover = no_remover->esq;
    }

    while (no_remover != NULL) // encontrou o valor a ser removido
    {
        if (no_remover->esq == NULL && no_remover->dir == NULL)//eh folha
        {
            if (raiz == no_remover)
            {
                free(raiz);
                no_remover = raiz = NULL;
            }
            else
            {
                if (no_remover->pai->esq != NULL && no_remover->pai->esq->pontuacao == chave)
                    no_remover->pai->esq = NULL;
                else
                    no_remover->pai->dir = NULL;

                free(no_remover);
                no_remover = NULL;
            }
        }
        else // tem um filho ou dois filhos
        {

            if (no_remover->dir != NULL)
                copia = sucessor(no_remover);
            else
                copia = antecessor(no_remover);

                //substitui o valor da ser removido pelo valor
                //do sucessor ou do antecessor
            no_remover->pontuacao = copia->pontuacao;
            no_remover->id = copia->id;
            //muda o no a ser removido para o sucessor ou antecessor
            no_remover = copia;
            chave = copia->pontuacao;
        }
        
    }
    return raiz;
}

No* registrar(No *raiz){
    int pontuacao;
    int id;
    scanf("%d", &id);
    scanf("%d", &pontuacao);
    return insere_no(raiz, pontuacao, id);
}

int verifica_existencia(No*raiz, int id, int *pontuacao_velha, int*flag){
    if(raiz != NULL && (*flag) != 1){
        verifica_existencia(raiz->dir, id, pontuacao_velha, flag);
        if(raiz->id == id){
            (*flag) = 1;
            (*pontuacao_velha) = raiz->pontuacao;
        }
        verifica_existencia(raiz->esq, id, pontuacao_velha, flag);
    }
}

No* UPDATE(No *raiz){
    int pontuacao_nova, pontuacao_velha, id, flag = 0;

    scanf("%d", &id);
    scanf("%d", &pontuacao_nova);
    
    if(raiz != NULL){
        verifica_existencia(raiz, id, &pontuacao_velha, &flag);

        if(flag == 1){
            raiz = remove_noavl(raiz, pontuacao_velha);
            raiz = insere_no(raiz, pontuacao_nova, id);
        }
    }
    return raiz;
}

int ID_MAX(No *raiz){
    No* aux = raiz;
    while(aux->esq != NULL)
        aux = aux->esq;
    return aux->id;
}

void posicao_calc(No *raiz, int id, int *CAT, int *flag){
    if(raiz != NULL && (*flag) != 1){
        posicao_calc(raiz->dir, id, CAT, flag);
        if(raiz->id == id){
            (*flag) = 1;
            (*CAT)++;
        }
        else if((*flag) != 1){
            (*CAT)++;
        }
        posicao_calc(raiz->esq, id, CAT, flag);
    }
}

void posicao(No *raiz){
    int id = 0;
    scanf("%d", &id);
    
    if(raiz != NULL){
        int rez_01, rez_02, aux_id, pos;
        rez_01 = rez_02 = aux_id = pos = 0;

        aux_id = ID_MAX(raiz);

        posicao_calc(raiz, id, &rez_01, &pos);

        pos = 0;

        posicao_calc(raiz, aux_id, &rez_02, &pos);

        if(rez_01>rez_02)
            printf("NOT_FOUND\n");
        else
            printf("%d\n", rez_01);
    }
    else{
        printf("NOT_FOUND\n");
    }
    
}

void top(No *raiz){
    if(raiz != NULL){
        No* aux = raiz, *pai_aux = NULL;
        while(aux != NULL){
            pai_aux = aux;
            aux = aux->dir;
        }
        printf("%d\n", pai_aux->id);
    }
    else{
        printf("EMPTY\n");
    }
}

void BOTTOM(No *raiz){
    if(raiz != NULL){
        No* aux = raiz, *pai_aux = NULL;
        while(aux != NULL){
            pai_aux = aux;
            aux = aux->esq;
        }
        printf("%d\n", pai_aux->id);
    }
    else{
        printf("EMPTY\n");
    }
}

void RANGE_CALC(No *raiz, int pont_min, int pont_max){
    if(raiz != NULL){
        RANGE_CALC(raiz->esq, pont_min, pont_max);
        if(pont_min <= raiz->pontuacao && raiz->pontuacao <= pont_max){
            printf("%d ", raiz->id);
        }
        RANGE_CALC(raiz->dir, pont_min, pont_max);
    }
}

void RANGE(No *raiz){
    int min;
    int max;
    min = max = 0;

    scanf("%d %d", &min, &max);
    if(raiz != NULL){
        RANGE_CALC(raiz, min, max);
        printf("\n");
    }
    else
        printf("EMPTY\n");
}

void COUNT_ABOVE(No *raiz, int ac, int* cont_0x){
    if (raiz != NULL)
    {
        COUNT_ABOVE(raiz->dir, ac, cont_0x);
        if(raiz->pontuacao > ac)
            (*cont_0x)++;
        COUNT_ABOVE(raiz->esq, ac, cont_0x);
    }
}

void COUNT_BELOW(No *raiz, int ac, int* cont_0x){
    if (raiz != NULL)
    {
        COUNT_BELOW(raiz->dir, ac, cont_0x);
        if(raiz->pontuacao < ac)
            (*cont_0x)++;
        COUNT_BELOW(raiz->esq, ac, cont_0x);
    }
}

int main(){

No *ABB = NULL;
int n, cont = 0;
int ac = 0;
int cont_02 = 0;
char rep[20];

scanf("%d", &n);

while(cont < n){
    strcpy(rep," ");
    fflush(stdin);
    scanf("%s", &rep);

    if(strcmp(rep, "REGISTER") == 0)
        ABB = registrar(ABB); 
    else if(strcmp(rep, "UPDATE") == 0)
        ABB = UPDATE(ABB);    
    else if(strcmp(rep, "POSITION") == 0)
        posicao(ABB);
    else if(strcmp(rep, "RANGE") == 0)
        RANGE(ABB);
    else if(strcmp(rep, "TOP") == 0)
        top(ABB);
    else if(strcmp(rep, "BOTTOM") == 0)
        BOTTOM(ABB);
    else if(strcmp(rep, "COUNT_ABOVE") == 0){
        ac = cont_02 = 0;
        scanf("%d", &ac);
        COUNT_ABOVE(ABB, ac, &cont_02);
        printf("%d\n", cont_02);
    }
    else if(strcmp(rep, "COUNT_BELOW") == 0){
        ac = cont_02 = 0;
        scanf("%d", &ac);
        COUNT_BELOW(ABB, ac, &cont_02);
        printf("%d\n", cont_02);
    }
    cont++;
}

return 0;}