Este es el Post #9 del Topic: Guía Práctica de PostgreSQL
El diagrama siguiente es la consecuencia de la creación de la base de datos con el script descrito en PostgreSQL: Sentencias DDL (1), Es un modelo pequeño, pero servirá para las futuras demostraciones con postgres.
Descripción: Una universidad realiza el registro de contactos, que son las personas que podrían ser potenciales postulantes a diferentes carreras. Los postulantes deberán pertenecer a un periodo académico y deberán elegir una modalidad asì como la carrera a la que quieren postular. Por lo general en un año solo existen dos periodos académicos, por ejemplo los periodos académicos del año 2008 fueron: 2008-1 y 2008-2. El contacto debe tener los datos personales del sujeto asì como la fecha de creación. El postulante debe tener registrado si asistió a su exámen de admisión, y si ingresó o no, debe tener también establecido el puntaje que alcanzó en el exámen.
Con esos datos podemos comenzar a elaborar nuestras primeras consultas usando instrucciones DML.
Antes de ir al grano, aclaro que acostumbro crear funciones cuando se trata de retornar datos, como por ejemplo una consulta que utilice la cláusula SELECT, y procedimientos almacenados únicamente cuando se trata de realizar una operación que no retorne ningún dato, por ejemplo, INSERT, UPDATE o DELETE.
En SQLServer se suele utilizar un Procedimiento Almacenado para cualquiera de los 2 casos descritos, aunque en PostGres también se puede retornar datos con procedimientos almacenados, sería ideal respetar la funcionalidad de cada uno. Las funciones desarrolladas en éste post se realizará haciedno uso del lenguaje procedural ‘plpgsql’, que generalmente trae PostGreSQL.
Si aún no tienes preparado la base de datos, aqui les dejo un script. Es la unión de las DDL y DML creadas en posts anteriores. Ésta creará el entorno en el que se realizarán las demostraciones.
Descargar Script: DBAdmision.sql
FUNCIONES:
1) Cree una función que devuelva los siguientes datos: <ver>
Periodo | 101 | 309 | 310 | Total |
2005-1 | 1 | 7 | 7 | 15 |
2005-2 | 0 | 13 | 12 | 25 |
2006-1 | 0 | 17 | 18 | 35 |
2006-2 | 0 | 23 | 22 | 45 |
2007-1 | 0 | 27 | 28 | 55 |
CREATE OR REPLACE FUNCTION f_ejemplo_1() RETURNS SETOF "record" AS $BODY$ DECLARE r RECORD; BEGIN FOR r IN SELECT idperacad AS "Periodo" ,SUM(CASE WHEN IDCarrera = '101' THEN 1 ELSE 0 END ) AS "101" ,SUM(CASE WHEN IDCarrera = '309' THEN 1 ELSE 0 END ) AS "309" ,SUM(CASE WHEN IDCarrera = '310' THEN 1 ELSE 0 END ) AS "310" ,COUNT(*) AS "TOTAL" FROM Persona.Postulante GROUP BY IDPerAcad LOOP RETURN NEXT r; END LOOP; RETURN; END; $BODY$ LANGUAGE 'plpgsql' VOLATILE; --SELECT * FROM f_ejemplo_1() AS ("Periodo" CHARACTER, "101" bigint,"309" bigint,"310" bigint,"TOTAL" bigint);
2) Cree una función que devuelva los siguientes datos: <ver>
Periodo | 101 | 309 | 310 | Total |
2005-1 | 1 | 7 | 7 | 15 |
2005-2 | 0 | 13 | 12 | 25 |
2006-1 | 0 | 17 | 18 | 35 |
2006-2 | 0 | 23 | 22 | 45 |
2007-1 | 0 | 27 | 28 | 55 |
TOTAL | 1 | 87 | 87 | 175 |
CREATE OR REPLACE FUNCTION f_ejemplo_2() RETURNS SETOF "record" AS $BODY$ DECLARE r RECORD; BEGIN FOR r IN SELECT idperacad AS "Periodo" ,SUM(CASE WHEN IDCarrera = '101' THEN 1 ELSE 0 END ) AS "101" ,SUM(CASE WHEN IDCarrera = '309' THEN 1 ELSE 0 END ) AS "309" ,SUM(CASE WHEN IDCarrera = '310' THEN 1 ELSE 0 END ) AS "310" ,COUNT(<em>) AS "TOTAL" FROM Persona.Postulante GROUP BY IDPerAcad UNION SELECT 'TOTAL' AS "Periodo" ,SUM(CASE WHEN IDCarrera = '101' THEN 1 ELSE 0 END ) AS "101" ,SUM(CASE WHEN IDCarrera = '309' THEN 1 ELSE 0 END ) AS "309" ,SUM(CASE WHEN IDCarrera = '310' THEN 1 ELSE 0 END ) AS "310" ,COUNT(</em>) AS total FROM Persona.Postulante LOOP RETURN NEXT r; END LOOP; RETURN; END; $BODY$ LANGUAGE 'plpgsql' VOLATILE; --SELECT * FROM f_ejemplo_2() AS ("Periodo" CHARACTER, "101" bigint,"309" bigint,"310" bigint,"TOTAL" bigint);
3) Cree una función que liste todos los postulantes de un determinado periodo académico. La lista debe reiniciar la numeración por orden alfabético. <ver>
Nº | Apellidos y Nombres |
1 | Alanya Padilla Alina Susan |
2 | Alarcon Castro Gustavo Claudio Andres |
… | … |
1 | Baldeón Balvín Olger |
2 | Baldeón Sanabria Natalia Ivonne |
… | … |
1 | Cachuán Cámac Miguel Fernando |
2 | Cajachagua Chui Jose Arturo |
… | … |
4) Cree una función que liste todos los postulantes, la lista debe estar enumerada y ordenada por apellido paterno, materno y nombres. <Requerimiento> <ver>
Nº | Apellidos y Nombres |
1 | Alanya Padilla Alina Susan |
2 | Alarcon Castro Gustavo Claudio Andres |
3 | Alarco Lama Ricardo Rafael |
… | … |
CREATE OR REPLACE FUNCTION f_ejemplo_4() RETURNS SETOF "record" AS $BODY$ DECLARE r RECORD; BEGIN FOR r IN SELECT rownumber() "Numero",* FROM (SELECT PC.Paterno || ' ' || PC.Materno || ' ' || PC.Nombres "Apellidos y Nombres" FROM Persona.Contacto PC INNER JOIN Persona.Postulante PP ON PP.IDContacto=PC.IDContacto ORDER BY PC.Paterno,PC.Materno,PC.Nombres) AS tb2 LOOP RETURN NEXT r; END LOOP; RETURN; END; $BODY$ LANGUAGE 'plpgsql' VOLATILE; --SELECT * FROM f_ejemplo_4() AS ("Numero" INTEGER, "Apellidos y Nombres" TEXT);
CREATE OR REPLACE FUNCTION rownumber() RETURNS integer AS $$ BEGIN EXECUTE 'CREATE TEMP SEQUENCE "'||current_timestamp||'"'; RETURN nextval('"'||current_timestamp||'"'); EXCEPTION WHEN duplicate_table THEN RETURN nextval('"'||current_timestamp||'"'); END $$ LANGUAGE 'plpgsql'; ;;
5) Cree una función que reciba como parámetro el id de un contacto y que muestra la siguiente información. Se debe de mostrar todos los periodos académicos y una X en todos aquellos en la cual el contacto postuló. <ver>
Periodo | Postuló |
2005-1 | X |
2005-2 | |
2006-1 | |
2006-2 | |
2007-1 | X |
6) Cree una función que muestre un resumen como el siguiente: <ver>
Apellidos y Nombres | 2005-1 | 2005-2 | 2006-1 | 2006-2 | 2007-1 | 2007-2 | TOTAL |
Alanya Padilla Alina Susan | X | 1 | |||||
Alarcon Castro Gustavo Claudio Andres | X | 1 | |||||
Alarco Lama Ricardo Rafael | X | 1 | |||||
Altez Yanez Jasmin Gabriela | X | 1 | |||||
Amable Salva Katerin Lisbet | X | 1 | |||||
… |
7) Cree una función que muestre un resumen como el siguiente: <ver>
Puntaje | 2005-1 | 2005-2 | 2006-1 | 2006-2 | 2007-1 | 2007-2 | TOTAL |
0 | 1 | 3 | 1 | 2 | 2 | 9 | |
12 | 1 | 1 | |||||
13 | 1 | 1 | |||||
14 | 1 | 1 | |||||
21 | 1 | 1 | 2 | ||||
23 | 1 | 3 | 4 | ||||
8.) Cree una función que liste todos aquellos postulantes (Apellidos, Nombres, Puntaje) cuyo puntaje es mayor o igual al puntaje promedio obtenido en todos los exámenes. <ver>
VISTAS:
9) Cree una vista que liste a todos aquellos postulantes que no asistieron al examen. Muestre los nombres, Apellidos, edad y género en formato (Varón/Mujer). <ver>
10) Cree una vista que liste a todos aquellos postulantes que ingresaron cuyo cumpleaños sea mañana (Entiéndase mañama como el dia después de la fecha actual). Muestre nombres y apellidos. <ver>
11) Cree una vista que muestre un resumen como el siguiente: <ver>
Nombre | 2005-1 | 2005-2 | 2006-1 | 2006-2 | 2007-1 | TOTAL |
Administración | 5 | 8 | 12 | 14 | 23 | 62 |
Contabilidad | 6 | 7 | 13 | 17 | 22 | 65 |
Ingeniería Informática | 0 | 0 | 0 | 0 | 0 | 0 |
TOTAL | 11 | 15 | 25 | 31 | 45 | 127 |
CREATE OR REPLACE VIEW v_ejemplo_11 AS ( SELECT AC.Nombre,SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2005-1') THEN 1 ELSE 0 END) "2005-1", SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2005-2') THEN 1 ELSE 0 END) "2005-2", SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2006-1') THEN 1 ELSE 0 END) "2006-1", SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2006-2') THEN 1 ELSE 0 END) "2006-2", SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2007-1') THEN 1 ELSE 0 END) "2007-1", SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2005-1','2005-2','2006-1','2006-2','2007-1') THEN 1 ELSE 0 END) "TOTAL" FROM Persona.Contacto PC INNER JOIN Persona.Postulante PP ON PP.IDContacto=PC.IDContacto INNER JOIN Admision.Carrera AC ON AC.IDCarrera=PP.IDCarrera GROUP BY AC.Nombre <pre><code>UNION SELECT 'TOTAL',SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2005-1') THEN 1 ELSE 0 END) &quot;2005-1&quot;, SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2005-2') THEN 1 ELSE 0 END) &quot;2005-2&quot;, SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2006-1') THEN 1 ELSE 0 END) &quot;2006-1&quot;, SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2006-2') THEN 1 ELSE 0 END) &quot;2006-2&quot;, SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2007-1') THEN 1 ELSE 0 END) &quot;2007-1&quot;, SUM(CASE WHEN PP.Ingreso='1' AND PP.IDPerAcad IN ('2005-1','2005-2','2006-1','2006-2','2007-1') THEN 1 ELSE 0 END) &quot;TOTAL&quot; FROM Persona.Contacto PC INNER JOIN Persona.Postulante PP ON PP.IDContacto=PC.IDContacto INNER JOIN Admision.Carrera AC ON AC.IDCarrera=PP.IDCarrera </code></pre> ); --SELECT * FROM v_ejemplo_11;
PROCEDIMIENTOS ALMACENADOS:
12) Cree un procedimiento almacenado que permita eliminar los postulantes registrados correspondientes a una modalidad. <ver>
CREATE OR REPLACE PROCEDURE sp_ejemplo_12(p_IDModalidad varchar) AS BEGIN DELETE FROM Persona.Postulante WHERE IDModalidad=p_IDModalidad; END --EXEC sp_ejemplo_12('2'); --SELECT * FROM Persona.Postulante;
13)Cree un procedimiento almacenado que permita registrar a un nuevo contacto. <ver>
14)Cree un procedimiento almacenado que permita actualizar los datos de un postulante. <ver>
TOPIC:
FUNCTION, PROCEDURE, VIEW
Thank you for your sharing. I am worried that I lack creative ideas. It is your article that makes me full of hope. Thank you. But, I have a question, can you help me?
alguien me puede ayudar con el ejercicio 5 pls
Thank you, I’ve just been looking for info about this subject for a long time and yours is the best I have discovered so far.
However, what concerning the conclusion? Are you sure concerning the
supply?
amigo, ya no esta la bd, sera que la puedas subir de nuevo? agradezco tu aporte, me ha servido para practicar
si pudieras ayudarme con este ERROR: error de sintaxis en o cerca de «PROCEDURE»
LINE 1: CREATE OR REPLACE PROCEDURE FUNCTION clientes() (n_id_client…
te lo agradeceria.
Buenas tengo un problema con una tarea si alguien me pudiera ayudar
es lo siguiente
tengo estas tablas
–listo
create table producto(
pr_id serial,
pr_nombre varchar (30),
pr_stock integer,
primary key (pr_id));
–listo
create table tablaPrecios(
tp_id serial,
PRODUCTO_pr_id integer,
tp_anterior integer check (tp_anterior>=590),
tp_actual integer check(tp_actual >=590),
primary key (tp_id),
foreign key (PRODUCTO_pr_id) references producto);
–listo
create table cliente(
cl_id serial,
cl_nombre varchar(30),
primary key (cl_id));
–listo
create table boleta(
bl_id serial,
cliente_cl_id integer,
bl_fecha date,
primary key (bl_id),
foreign key (cliente_cl_id) references cliente);
–listo
create table detalle(
PRODUCTO_pr_id integer,
boleta_bl_id integer,
dt_cantidad integer check (dt_cantidad>0),
dt_precio integer check (dt_precio >=590),
primary key (PRODUCTO_pr_id,boleta_bl_id),
foreign key (PRODUCTO_pr_id) references producto,
foreign key (boleta_bl_id) references boleta);
create Table Operacion(
op_id serial,
op_fecha date,
op_responsable varchar(20),
op_tipo char(1) check (op_tipo = ‘A’ or op_tipo = ‘C’),
op_idelemento integer,
primary key (op_id));
y con estas tablas necesito hacer estas funciones
1. Retornar el nombre del producto que ha registrado mayor cambio de precios.
2. Retornar el nombre del producto mas barato que se encuentra registrado.
3. Retornar el nombre del producto mas caro que se encuentra registrado.
4. Retornar el nombre del producto que ha tenido la mayor variacion de precio.
5. Dado un ID de boleta, retornar el total de la boleta.
6. Dado el nombre de un mes y un a~no, retornar la cantidad total de unidades adquiridas
durante ese mes.
7. Dado el nombre de un mes y un a~no, imprimir todas las boletas que fueron emitidas
en ese perodo. Se debera indicar el ID de la boleta, la fecha y el total de la boleta.
8. Dado el nombre de un cliente, imprimir todas las boletas que fueron emitidas en ese
perodo. Se debera indicar el ID de la boleta, la fecha y el total de la boleta.
9. Dado el ID de una boleta, imprimir las lneas de detalle de la boleta, indicando el
numero de unidades, el precio del producto y el valor total (producto entre el numero
de unidades y el precio).
10. Imprimir, para cada mes-a~no, la cantidad de adquisiciones que se han registrado. Se
debe indicar el nombre del mes, el a~no y el numero total de adquisiciones que se
registraron en ese perodo de tiempo.
si alguien me pudiese ayudar
es urgente porfavor ayudenmeeee!!!!
necesito pasar este ramo y de este trabajo es lo unico ke me falta
saludos!!!
yo tambien necesito el manual pdf de store procedures, funciones, y vistas, mi email es:
htmf@hotmail.es
Todo esta bacan necesito algun manual sobre procedimientos almacenados y vistas en formato .pdf si tendrias alguno me lo puedes enviar a mi correo: yeranto_17@hotmail.com
Con que datos se llenaron las tablas??
Con que datos llenaron las tablas
Quien me puede ayudar con una funcion que calcule el monto total de las compras realizadas a un proveedor determinado.
Muy buen aporte man thx, andaba buscando algo asi 😀
buen material para los que se inician gracias
Muy buen material, me ha sido de gran utilidad.
que tal, bueno para mi que estoy iniciandome, me gustaria que lo ejemplos los pusieras en un rar, ya que no todos los pude ver, puesto que con los ejemplos los puedo analizar y preguntar mejor, gracias
el ejemplo 1 y 2 si se ven pero el 3 no lo pude ver y asi sucesivamente
jaja, gracias por recordarme, estos dias me tocaba actualizar los post, y gracias por los procedimientos que resolviste, ….
Ya esta muy trillado tus ejercicios man…
esas las resolvi hace tiempo…actualizate.
CONSULTAS:
–1
select idperAcad
,sum (case when idCarrera =’101′ then 1 else 0 end) as ‘101’
,sum (case when idCarrera =’309′ then 1 else 0 end) as ‘309’
,sum (case when idCarrera =’310′ then 1 else 0 end)as ‘310’
,count (idpostulante) as TOTAL
from persona.postulante
group by idperacad
–2
select IDPerAcad as periodo,[101], [309] ,[310],[101]+[309]+[310]AS TOTAL
from
(select idcarrera,IDPerAcad from persona.postulante ) s
PIVOT
( COUNT(idcarrera)
FOR idcarrera IN ([101], [309] ,[310] ) ) qq
UNION
select ‘total’
,sum([101]), sum([309]),sum([310])
,sum([101]+[309]+[310])as total
from (
select IDPerAcad,idcarrera
from persona.postulante
) as data
PIVOT
(
count(idcarrera)–lo q va ir dentro de los casilleros
FOR idcarrera IN
([101], [309] ,[310] )
) as PVT
–3
select ROW_NUMBER() OVER(partition by substring(persona.contacto.paterno,1,1) ORDER BY persona.contacto.paterno
,persona.contacto.Nombres) [Nro],persona.contacto.Paterno+’ ‘+ persona.contacto.nombres as [Apellidos y Nombres]
from persona.contacto inner join persona.postulante on persona.contacto.idcontacto=persona.postulante.idcontacto
where persona.postulante.idperacad= ‘2007-1′
group by persona.contacto.Paterno,persona.contacto.Nombres
–4
–PREGUNTA 4: Listar a todos aquellos postulantyes cuyo puntaje es mayor o igual
–al puntaje promedio obtenido en los examenes
SELECT Persona.Contacto.Paterno, Persona.Contacto.Materno, Persona.Contacto.Nombres, Persona.Postulante.Puntaje
FROM Persona.Postulante INNER JOIN
Persona.Contacto ON Persona.Postulante.IDContacto = Persona.Contacto.IDContacto
where Persona.Postulante.Puntaje >=(
select avg(cast(puntaje as decimal)) from persona.postulante
where asistioexamen=’1′)
–5
–PREGUNTA 5:Listar a todos aquellos postulantyes cuyo puntaje es mayor o igual
–al puntaje promedio obtenido en el periodo academico
SELECT Persona.Contacto.Paterno, Persona.Contacto.Materno, Persona.Contacto.Nombres, Persona.Postulante.Puntaje
FROM Persona.Postulante INNER JOIN
Persona.Contacto ON Persona.Postulante.IDContacto = Persona.Contacto.IDContacto
where Persona.Postulante.Puntaje >=(
select avg(cast(puntaje as decimal)) from persona.postulante
where asistioexamen=’1′ and idperAcad=’2005-1′ )
–6
–PREGUNTA 6: Vista que liste a todos aquellos postulantes que nunca postularon
create view v_postulante
as
select c.nombres as Nombres,c.paterno +’ ‘+ c.materno as Apellidos, case c.genero
when ‘1’ then ‘Varon’
when ‘0’ then ‘Mujer’
end as Genero
,datediff(year,c.fechanac,getdate()) as Edad from persona.contacto as c
where not exists (select * from persona.postulante as p
where c.idcontacto = p.idcontacto)
go
select * from v_postulante
–7
–PREGUNTA 7:Vista de postulantes que ingresaron cuyo cumpleaños sea mañana
create view v_fechacumpleaños
as
SELECT Persona.Contacto.Nombres, Persona.Contacto.Paterno+’ ‘+ Persona.Contacto.Materno as Apellidos
, Persona.Postulante.Ingreso, Persona.Contacto.FechaNac
FROM Persona.Contacto INNER JOIN
Persona.Postulante ON Persona.Contacto.IDContacto = Persona.Postulante.IDContacto
where ingreso=’1’ and month(Persona.Contacto.FechaNac)=month (getdate()) and
day(Persona.Contacto.FechaNac)=day((getdate()+2))
go
select * from v_fechacumpleaños
–8
create procedure periodoacad
@acad varchar(6)
as
select row_number() over ( order by (
Paterno ))as [N°], Paterno + ‘ ‘+ Materno+’ ‘ + Nombres as [Apellidos y Nombres]
from persona.contacto p inner join persona.postulante pp on p.idcontacto = pp.idcontacto
where pp.idperacad = @acad
order by [Apellidos y Nombres]
exec periodoacad ‘2006-1’
–9
create schema funciones
go
create function funciones.postul(@cod int)
returns table
return
(
select pp.idperacad,
mAX(case when pp.idcontacto=@cod
then ‘x’ else ”end)as postulo
from personA.contacto as pc inner join persona.postulante as pp
on pc.idcontacto=pp.idcontacto
group by pp.idperacad
)
go
select * from funciones.postul(1)
–10
create procedure usp_nombre_periodo
as
SELECT Paterno +’ ‘+ Materno+’ ‘ + Nombres as [Apellidos y Nmbres]
,sum (case persona.postulante.idperacad when ‘2005-1’ then 1 else ‘ ‘ end) as ‘2005-1’
,sum (case persona.postulante.idperacad when ‘2005-2’ then 1 else ‘ ‘ end) as ‘2005-2’
,sum (case persona.postulante.idperacad when ‘2006-1’ then 1 else ‘ ‘ end) as ‘2006-1’
,sum (case persona.postulante.idperacad when ‘2006-2’ then 1 else ‘ ‘ end) as ‘2006-2’
,sum (case persona.postulante.idperacad when ‘2007-1’ then 1 else ‘ ‘ end) as ‘2007-1’
,sum (case persona.postulante.idperacad when ‘2007-2’ then 1 else ‘ ‘ end) as ‘2007-2’
,count (persona.postulante.idcontacto) as TOTAL
FROM Persona.Contacto INNER JOIN
Persona.Postulante ON Persona.Contacto.IDContacto = Persona.Postulante.IDContacto
group by Persona.Contacto.Paterno,Persona.Contacto.materno,Persona.Contacto.nombres
order by [Apellidos y Nmbres]
go
exec usp_nombre_periodo
–11
–11. Cree un procedimiento almacenado que muestre un cuadro resumen como el siguiente:
create procedure usp_puntaje_periodo
as
SELECT Persona.Postulante.Puntaje
,sum (case persona.postulante.idperacad when ‘2005-1’ then 1 else ‘ ‘ end) as ‘2005-1’
,sum (case persona.postulante.idperacad when ‘2005-2’ then 1 else ‘ ‘ end) as ‘2005-2’
,sum (case persona.postulante.idperacad when ‘2006-1’ then 1 else ‘ ‘ end) as ‘2006-1’
,sum (case persona.postulante.idperacad when ‘2006-2’ then 1 else ‘ ‘ end) as ‘2006-2’
,sum (case persona.postulante.idperacad when ‘2007-1’ then 1 else ‘ ‘ end) as ‘2007-1’
,sum (case persona.postulante.idperacad when ‘2007-2’ then 1 else ‘ ‘ end) as ‘2007-2’
,count (persona.postulante.idcontacto) as TOTAL
FROM Persona.Contacto INNER JOIN
Persona.Postulante ON Persona.Contacto.IDContacto = Persona.Postulante.IDContacto
GROUP BY Persona.Postulante.Puntaje
ORDER BY Persona.Postulante.Puntaje
go
exec usp_puntaje_periodo
viejo no admite el procedure al crear