colas dobles

Iniciado por Beginner Web, 15 Octubre 2018, 03:56 AM

0 Miembros y 1 Visitante están viendo este tema.

Beginner Web

Hola, queria preguntarles que operaciones se de colas se modifican cuando usamos bicolas, tengo hecho esto pero no entiendo el comportamiento con arreglos con listas es mucho mas facil ayuda u.u  :(

Código (cpp) [Seleccionar]
#include <iostream>
#include <stdlib.h>

using namespace std;

const int MAX=6;
typedef int contenedor[MAX];
typedef struct tcola{
contenedor datos;
int final, frente;
};

void init_queue(tcola &q);
void push_queue(tcola &q, int nuevo, bool ultimo);
bool full_queue(tcola q);
bool empty_queue(tcola q);
int pop_queue(tcola &q);
int top_queue(tcola q);
int bottom_queue(tcola q);
int next(int indice);
int prev(int indice);

int main()
{
int dato;
tcola q;
bool opcion;
init_queue(q);
while(full_queue(q)==false){
cout<<"Ingrese dato: ";
cin>>dato;
cout<<"1. Agregar por final"<<endl;
cout<<"0. Agregar por frente"<<endl;
cin>>opcion;
push_queue(q,dato,opcion);
}
q.frente=MAX-1;
q.final=MAX-2;
while(empty_queue(q)==false){
cout<<pop_queue(q);
}
cout<<endl;
system("pause");
}

void init_queue(tcola &q)
{
q.final=MAX-1;
q.frente=MAX-1;
}

void push_queue(tcola &q, int nuevo, bool ultimo)
{
if(full_queue(q)==true)
cout<<"COLA LLENA"<<endl;
else{
if(ultimo==true){
q.final=next(q.final);
q.datos[q.final]=nuevo;
}
else{
q.frente=prev(q.frente);
q.datos[q.frente]=nuevo;
}
}
}

bool full_queue(tcola q)
{
return next(q.final)==q.frente;
}

bool empty_queue(tcola q)
{
return q.frente==q.final;
}

int pop_queue(tcola &q)
{
int aux;
if(empty_queue(q)==true)
aux=-1;
else{
q.frente=next(q.frente);
aux=q.datos[q.frente];
}
return aux;
}

int top_queue(tcola q)
{
int aux;
if(empty_queue(q)==true)
aux=-1;
else{
aux=q.datos[next(q.frente)];
}
return aux;
}

int bottom_queue(tcola q)
{
int aux;
if(empty_queue(q)==true)
aux=-1;
else{
aux=q.datos[q.final];
}
return aux;
}

int next(int indice)
{
if(indice==MAX-1)
indice=0;
else
indice++;
return indice;
}

int prev(int indice)
{
if(indice==0){
indice=MAX-1;
}
else{
indice--;
}
return indice;
}
7w7

Serapis

#1
Depende de la forma en que estén unidas las dos colas.
He llegado a programar diferentes variedades como colas que extraían (si no estaba vacía) alternativamente, de una y luego de otra... también por prioridad extrayendo de la cola que tenga un valor que cumpla una cierta condición (por ejemplo el menor)...

En general al margen de otras propiedades que se pueden añadir, la única operación extra suele ser extraer de la cola y encolarla de nuevo por una de las dos. requiere un parámetro para señalar a cual de ellas se refiere, basta un buleano para decidirlo, pués son solo 2.

PopAndPush( cola)



Sobre las colas realizadas con arrays:
La dificultad de los arrays, es que al extraer un elemento, los demás no bajan 'por su peso'.... se soluciona moviendo el índice de lo que es el tope y la cima según convenga...

Sea un elemento lo definido entre 2 rayas verticales: "||"
Y sea una cola vacía de 10 elementos, el fondo y la cima ahora mismo son 0,
  -------------
------------->|--->
  -------------

después de meter 4 elementos (cuenta 4 espacios entre las 5 rayitas), el fondo es 0, la cima 4
  -------------
--------->|||||--->
  -------------

ahora sacamos un elemento... el fondo es 1 y la cima sigue siendo 4
  -------------
--------->||||--->
  -------------

Metemos 4 elementos más (hay 7), el fondo sigue siendo 1, la cima suma 4 más (ahora es 8).
  -------------
----->||||||||--->
  -------------
Ahora extraemos 4 (quedan 3 otra vez), el fondo suma 4 (ahora es 5), la cima no cambia (sigue siendo 8)...
      -------------
----->||||--->
      -------------
Extraemos 1 elemento más, el fondo ahora suma 1 (pasa a valer 5), la cima no cambia (sigue siendo 8 y quedan aún 2 elementos en la cola)
      -------------
----->|||--->
      -------------
Ahora metemos 4 elementos (pasa a haber 6), el fondo no cambia cuando se mete (sigue siendo 5), la cima suma 4, (pasa a ser 12) pero como excede el tamaño del array ... verás como se solventa.
      -------------
---->||||--->||||
      -------------
solo caben 1 ahí, los otros 3 se colocan al otro lado del array... es decir en el índice 0, 1 y 2.
La cima ahora está donde pongo la '¡' y el fondo donde pongo '!'
      -------------
---->|||!---->¡|||
      -------------

Así ves que siempre se suma... en la cima cuando se añaden, en la cola cuando se sacan.
Y entonces resulta circular, pero transparente desde fuera. como nos daba 12 y queda fuera de rango, hay que modularlo...

cima = (((cima + añadidos) modulo size-1) -1)
Se puede resumir, puesto que si "size-1" y luego "-1" se resume en "modulo size"... Esto porque el array se basa en indice 0.

Luego una función resulta ideal para llevar la cuenta de cima y fondo, lo que al final facilita enormemente la tarea...

entero Funcion Recalcular(previo, cambios)
  devolver ((previo + cambios) modulo size)
fin funcion

// se añaden 4 elementos
cima = Recalcular(cima, 4)
size  += 4

// se sacan 2 elementos
fondo = Recalcular(fondo, 2)
size -= 2


Natualmente como se añade uno a uno, y se sacan uno a uno... el parámetro cambios solo tendrá valores: '+1' y '-1' y allí dentro mismo puede actualizarse también size.


entero Funcion Recalcular(previo, cambio)
  devolver ((previo + cambio) modulo size)
  size += cambio  // suma 1 o resta uno según el signo de 'cambio'.
fin funcion

// después de añadir 1 elemento
cima = Recalcular(cima, 1)

// después de sacar 1 elemento
fondo = Recalcular(fondo, -1)






Beginner Web

#2
Muchas gracias no lei todo pero creo que me sirvio me quedo asi.

Bicola entrada restringida:
Código (cpp) [Seleccionar]
int pop_queue(tcola &q, bool ultimo)
{
int aux;
if(empty_queue(q)==true)
aux=-1;
else{
if(ultimo==false){
aux=q.datos[q.final];//Aqui tenia sueño y puse
q.final=prev(q.final);//esta linea antes que la de arriba
}
else{
q.frente=next(q.frente);
aux=q.datos[q.frente];
}
q.contador--;
}
return aux;
}


Bicola salida restringida:
Código (cpp) [Seleccionar]
void push_queue(tcola &q, int nuevo, bool ultimo)
{
if(full_queue(q)==true)
cout<<"COLA LLENA"<<endl;
else{
if(ultimo==true){
q.final=next(q.final);
q.datos[q.final]=nuevo;
}
else{
q.datos[q.frente]=nuevo;//Aqui
q.frente=prev(q.frente);//lo mismo
}
}
}
7w7