Examen Mayo 2018
Pregunta 1
(Prueba escrita teoría, 0.5 puntos)Describir el proceso de inserción, en un árbol AVL, de la siguiente secuencia de nodos: 10, 12, 14, 9, 11, 13, 19, 16, 17. Se deberán mostrar todos los pasos intermedios, indicando y describiendo las acciones que se deban ir tomando.
Pregunta 2
(Prueba escrita teórica, 1 punto)
a) (0.5 puntos)
Especifique algebraicamente la operación Frontera de árboles binarios teniendo en cuenta las operaciones constructoras generadoras:
CrearArbolBinVacio:
→TArbolBinario
ConstruirArbolBin:
→TArbolBinario
XTElemento
XTArbolBinario
→TArbolBinario
Así como las operaciones del TAD Lista:
CrearListaVacia:
→TLista
Construir:
TElemento
XTLista
→TLista
Concatenar:
TLista
XTLista
→TLista
frontera(crearArbolBVacio) = crearListaVacia
frontera(construirArbolBVacio(i, r, d)) =
b) (0.5 puntos)
Dado un grafo dirigido en memoria dinámica y dos vértices, si se quiere implementar una operación para devolver la longitud de un camino posible entre ambos vértices, cuántos tipos abstractos de datos necesitaría en unidades diferentes. Justifique las respuestas e indique brevemente cómo se haría
FUNCTION logitudCaminoPosisble(var g:TGrafo; nSrc, nDest:^TNodo):integer; begin
recorridoAnchura(l, g, nSrc, nDest); // Recorremos calculamos el recorrido en anchura y lo recivimos el camino en `l`
recortarLista(l; g); // Recorta los ciclos en el camino
longitud(l); // Hallamos la longitud del camino
end;
recortarLista(var l:TLista; g:TGrafo)
Este subprograma recorre una lista de nodos de un grafo y la recorre desde el principio hasta el final. En cada iteración recorre la lista desde el final el nodo actual o hasta encontrar un nodo que sea adyacente con el nodo actual (pertenece(nodoActual, nodoIterInv)
), si se encuentra tal nodo, se van a eliminar de la lista todos los nodos que se encuentren entrenodoActual
ynodoIterInv
Estructuras utilizadas
- Grafo → Estructura a recorrer
- Listas → Lista para devolver el recorrido y posteriormente calcular su longitud
- Conjunto → Necesario para llevar la cuenta de los nodos recorridos
Pregunta 3
No la hemos dado
Pregunta 4
(Prueba escrita práctica, 1,5 puntos)
a) (0.75 puntos)
Implemente en una unidad la definición de tipos de un árbol binario en memoria dinámica que guarde elementos de tipoTElemento
definido en la unidaduElemento
. A continuación implemente la unidaduElemento
con la definición del tipoTElemento
como un array con 2 enteros. Implemente los subprogramas adecuados en sus respectivas unidades y las llamadas necesarias en el programa principal para mostrar por pantalla todos los elementos que contenga un árbol (cada elemento en una línea) que previamente se ha inicializado (no se pide inicializar el árbol)
uses uElemento
type
TArbolB = ^TNodoArbol;
TNodoArbolB = record
raiz: TElemento;
hijoI, hijoD: ^TArbol;
end;
uses uElemento
type
TElemento = array[1..2] of integer;
PROCEDURE imprimirArbol(a:TArbolB);
var i,d:TArbolB;
e:TElemento;
begin
if not esArbolVacio(a) then begin
getRaiz(e, a);
imprimirElemento(e)
getHijoI(i, a);
imprimirArbol(a^.hijoI);
getHijoD(d, a);
imprimirArbol(a^.hijoI);
end;
end;
PROCEDURE getRaiz(var e:TElemento; a:TArbolB); begin
copiarElemento(e, a^.raiz);
end;
PROCEDURE getHijoI(var i:TArbolB; a:TArbolB); begin
i := a^.hijoI;
end;
PROCEDURE imprimirElemento(e:TElemento); begin
writeln(e[1], ', ', e[2]);
end;
PROCEDURE copiarElemento(var eDest:TElemento; eSrc:TElemento); begin
eDest := eSrc;
end;
b) (0.75 puntos)
Haciendo uso del TAD ÁrbolBinario y el TAD Lista vistos en clase, implemente el subprogramaPadresNoAbuelos
que dado un árbol binario devuelva una lista con los elementos que pertenecen a los nodos del árbol que sean padres, pero no abuelos. Asumimos que se dispone de la implementación de la función EsHoja, que recibe un árbol y devuelve un booleano. Se entiende que el tipo base del arbol binario y de la lista es el mismoTElemento
PROCEDURE padresNoAbuelos(var l:TLista; a:TArbolB); begin
if (not esArbolVacio(a)) and (not esHoja(a)) then begin // Es padre
getI(i, a);
getD(d, a);
if (esHoja(i)) and (esHoja(d)) then begin // NO es abuelo
getRaiz(e, a);
construir(l, e);
end;
padresNoAbuelos(l, i);
padresNoAbuelos(l, d);
end;
end;
Pregunta 5
(Prueba escrita práctica, 4,5 puntos)Una universidad quiere limpiar su imagen pública recientemente afectada por un caso de fraude. Su equipo de gobierno persigue que los egresados de la universidad tengan un nivel medio superior al ya de por sí excelente. Para ello, ha creído conveniente crear un sistema (a coste casi cero) de lecciones online de refuerzo obligatorias para los alumnos más rezagados, y optativas para el resto.Cada curso académico, un sistema informatizado calculará el Nivel de Esfuerzo Personal (NEP) de cada uno de sus alumnos, y señalará anónimamente quiénes de ellos tendrán que asistir obligatoriamente a las clases de refuerzo. Para ello, se aportará información, tanto por el propio estudiante como por los profesores, que se utilizará para medir el rendimiento de cada alumno más allá de las calificaciones que obtiene, ya que, como todo el mundo sabe, las calificaciones pueden venir sesgadas por la facilidad de algunas asignaturas, del plan de estudios, de los docentes contratados u otros factores de dudosa objetividad.Para la versión 1.0 del cálculo del NEP, el vicerrector competente ha diseñado una combinación de entradas que incluye información sobre:
- Las 4 asignaturas más importantes de cada curso (2 por cuatrimestre) para cada titulación (elegidas por los docentes y vulgarmente llamadas asignaturas estrella)
- El SDNM se calcula como la suma de las diferencias entre las calificaciones obtenidas por el alumno con respecto a cada una de las medias de las asignaturas estrella en su curso académico. Por ejemplo, para un estudiante que termina 1º de GII, y si las asignaturas estrella son MDA, IP, ED y CAL con notas medias para su clase de y , y calificaciones personales de y , su SDNM se calcularía como
- El entorno del estudiante, formado por los 5 compañeros que cada estudiante decide elegir como Entorno Personal,
- Para el cómputo del SDNM_EP se procedería del mismo modo con las 4 asignaturas estrella de cada uno de los 5 estudiantes del entorno personal y se calcularía la media de los 5 resultados.
- Factor que evalúa el progreso del estudiante durante el curso si durante el primer cuatrimestre le fue mal.
- Para el cómputo de la EVO se calcularía la SDNM de las asignaturas estrella del primer cuatrimestre (SDNM_1) y si devuelve positivo, su EVO es directamente 0, mientras que si devuelve un valor negativo (dando a entender que le fue peor que a la media de la clase), se calcula la SDNM de las asignaturas estrella del segundo cuatrimestre (SDNM_2) y se compara con la mitad de la SDNM_1, si es igual o mayor, el EVO sería +1, y en caso contrario -1.
Con los valores del ejemplo anterior, tendríamos que el estudiante de referencia habría obtenido un SDNM_1= , negativo, por lo que habría que calcular su SDNM_2 = , siendo por lo que su EVO sería+1
.Finalmente, la expresión del NEP es:
a) (0.5 puntos)
Diseñe la estructura de datosTNEP_Universidad
para cargar toda la información del NEP de sus alumnos en memoria principal. La estructura es un listado de Centros de tipoTCentro
(los centros son las Escuelas/Facultades). Cada Centro, además delnombre
del mismo (string), contiene información de las titulaciones (TTitulacion
) que pertenecen al centroEn cada titulación se guarda un entero con elcódigo
de la titulación y el listado de los estudiantes de la titulación.La información concreta de cada estudiante (TPerfilEstudiante
) es:código
de estudiante (compuesto por 2 enteros,TCodigoEstudiante
), un entero para describir el último curso de asignaturas estrella completado por el estudiante (ultimoCurso
), otro entero para saber en cuántos grupos se ha elegido al alumno como parte de un Entorno Personal (numSeleccionEP
), un campo para guardar suNEP
actual (valor de tipo real), un almacen con los 5 códigos de estudiante de su Entorno Personal, y un almacen con la información de las 4 asignaturas estrella (las 2 primeras correspondientes a las de primer cuatrimestre y las 2 segundas a las de segundo cuatrimestre).La información de cada asignatura estrella que guarda qun alumno será elcódigo
de asignatura (un valor entero) y la calificaciónfinal
(valor real) obtenida en dicha asignatura
uses uCentro;
type
TNEP_Universidad = ^TNodoListaC;
TNodoListaC = record
info:TCentro;
sig:^TNodoListaC;
end:
uses uListaE;
type
TCentro = record
nombre:string;
titulaciones:TListaT;
cod:integer;
end:
uses uTitulacion;
type
TListaT = ^TNodoListaT;
TNodoListaT = record
info:TTitulacion;
sig:^TNodoListaT;
end:
type
TTitulacion = record
cod:integer;
estudiantes:TListaE;
end;
uses uEstudiante;
type
TListaE = ^TNodoListaE;
TNodoListaE = record
info:TEstudiante;
sig:^TNodoListaE;
end:
type
TCodigoEstudiante = array[1..2] of integer;
TPerfilEstudiante = record
cod:TCodigoEstudiante;
ultimoCurso:integer;
numSeleccionadoEP:integer;
nep:real;
ep: array[1..5] of TCodigoEstudiante;
asignaturas: array[1..2] of integer;
end;
b) (1 punto)
Crear los procedimientos adecuados para insertar en la estructura principal un alumno de nuevo ingreso al que en el proceso de matrícula le han asignado código de estudiante y código de titulación que se pasarán a los subprogramas adecuados. Algunos de los campos de la información de su perfil quedarán vacíos hasta que no complete el primer curso, haya seleccionado a su Entorno Personal y haya cursado las asignaturas estrella
procedure nuevoAlumno(var facultad:TCentro; codE:TCodigoEstudiante; codT:integer);
var auxc:TListaA;
auxa:TPerfilEstudiante;
auxle:TListaE;
t:TTitulacion;
begin
auxc := facultad.titulaciones;
// Buscar titulacion en base a un codigo (se asume que existe)
getTitulacion(t, auxc);
while (not esTitulacionIgual(codT, t)) do begin
getTitulacion(t, auxc);
auxc := auxc^.sig;
end;
crearAlumno(auxa, codE); // Crear elemento alumno
getListaE(auxle, t); // Obtener campo del registro
construir(auxle, auxa); // Añadir elemento a la lista
setListaE(auxle, t); // Actualizar campo del registro
end;
c) (1 punto)
Para asegurar que no se eligen siempre a los mismos alumnos en los Entornos Personales, no está permitido que un alumno pertenezca a más de 10 grupos. Crear el subprograma Verificacion para comprobar que, dada una variable con toda la información (de tipoTNEP_Universidad
), no hay estudiantes elegidos en más de 10 grupos. Para ello se tendrá en cuenta que cada vez que es elegido un estudiante en un Entorno Personal, se incrementa su correspondiente camponumSeleccionEP
de número de Entornos Personales en los que aparece. Cada vez que se detecte que un alumno aparece en más de 10 grupos se enviará una advertencia al gestor del vicerrectorado de alumnos mediante la llamada al procedimiento de prototipoPROCEDURE AvisoViceAlumnos(cod: TCodigoEstudiante; num: integer); {PRE: “cod” es el código de estudiante, y “num” es el número de apariciones en Entornos Personales que tiene}
procedure comprobarEntornosPersonales(listaCentros:TNEP_Universidad);
var lt:TListaT;
le:TListaE;
auxe:TEstudiante;
begin
// Recorrer cada facultad/centro
while not esListaCVacia(listaCentros) do begin
// Recorre cada titulacion
lt := listaCentros^.titulaciones;
while not esListaTVacia(listaCentros) do begin
// Recorre cada estudiante
le = lt^.estudiantes;
while not esListaEVacia(listaCentros) do begin
primero(auxe, le)
if retNumEP(auxe) > 10 then
avisoViceAlumnos(retCodEstudiante(auxe), retNumEP(auxe))
le^.sig;
end;
lt := lt^.sig;
end;
listaCentros := listaCentros^.sig;
end;
end;
d) (2 puntos)
Dada la estructura principal de tipoTNEP_Universidad
y un código de estudiante (TCodigoEstudiante
), obtener el NEP que le corresponde para el curso académico 2017/18 y asignárselo a su campo específico.La información de las asignaturas de los respectivos planes de estudio se mantiene en un único fichero que custodia el Vicerrectorado de Ordenación Docente. El vicerrectorado ofrece una función para extraer la nota media de una asignatura concreta calculada para todo el alumnado que la cursó en un curso académico y tiene el siguiente prototipoFUNCTION NotaMediaAsign(codigoAsignatura, curso: integer):real; {PRE: codigoAsignatura es un entero que identifica la asignatura, y curso es un entero; por ejemplo 1516 para identificar el curso 2015/16. POST: la función devuelve la nota media de la asignatura para el curso dado}
Además, para la localización de alumnos en el listado de estudiantes de la titulación se ofrece el uso del procedimientoLocalizaEstudiante
con prototipoPROCEDURE LocalizaEstudiante(cod: TCodigoEstudiante; lEst:TListaEstudiantes; VAR pos: TListaEstudiantes); {PRE: “cod” es el código de estudiante y “lEst” la lista de estudiantes de una titulación POST: “pos” es un puntero que apunta al nodo que guarda el perfil del alumno con código de estudiante “cod”. Devuelve NIL en caso de no encontrar ese código de estudiante en la lista}
function obtenerNEP(listaCentros:TNEP_Universidad; cod:TCodigoEstudiante):integer;
var pe:^TEstudiante;
e:TEstudiante;
begin
pe := NIL;
// Recorrer cada facultad/centro (se asume que el alumno existe)
while pe = NIL do begin
// Recorre cada titulacion (se asume que el alumno existe)
lt := listaCentros^.titulaciones;
while not pe = NIL do begin
getListaEstudiantes(listaE, lt);
localizarEstudiante(cod, listaE, pe);
lt := lt^.sig;
end;
listaCentros := listaCentros^.sig;
end;
primero(e, lt)
obtenerNEP := calcSDNM(e) + calcSDNM_EP(e) + calcEVO(e) // Los estos subprogramas devolvería el valor correspondiente de cada calculo
end;