Ejercicio modelo entidad relación
Una empresa de fabricación de tintas nos ha pedido una base de datos.
La empresa necesita guardar información de sus clientes, concretamente el NIF y la razón social. Los clientes hacen pedidos, de los que queremos saber la fecha y la dirección de envío (calle, cp y población).
Los pedidos pueden ser de varios productos. De los productos queremos guardar la referencia, el precio y el stock que tenemos en almacén.
Por otro lado los productos pueden ser productos básicos o fabricados, los fabricados se hacen a partir de otros productos (que pueden ser base o fabricados). Necesitamos guardar la composición de los productos fabricados que incluye los productos, la cantidad de cada uno y el orden en el que hay que añadirlos.
Datos de prueba:
Clientes:
Sugus, nif b1234354
Grefusa, nif b75756756
S&D, nif b66612333
Productos:
Disolvente, 1 €, 2000
Emulsionante, 3 €, 1500
Colorante rojo, 4 €, 200
Colorante azul, 4 €, 300
Colorante amarillo, 4 €, 250
Tinta naranja, 10 €, 100
Tinta verde, 10 €, 100
La tinta naranja se compone en este orden de
Disolvente 5
Colorante rojo 7
Colorante amarillo 10
Emulsionante 10
La tinta verde se compone en este orden de
Disolvente 5
Colorante azul 7
Colorante amarillo 10
Emulsionante 10
Sugus hizo un pedido de 10 unidades de tinta verde el 2 de febrero para enviar a la calle Agla,5, 08001 Barcelona.
Sugus hizo un pedido de 10 unidades de tinta naranja el 5 de febrero para enviar a la calle Agla,5, 08001 Barcelona.
S&D hizo un pedido de 10 unidades de disolvente y 5 de colorante rojo el 10 de febrero para enviar al carrer perill, 15, 08012 Barcelona.
Se pide:
Insertar el cliente ‘Copistería Martínez’, ‘B21312312’
Modificar el precio de todos los productos para incrementar un 10%
Añadir 10 a la cantidad de la composición de todos los productos
¿Qué productos han comprado los clientes que tienen una ‘u’ en el nombre?
Añade el producto ‘aerosol rojo’, 10 €, 20
Elimina los productos cuyo nombre contenga aerosol
¿Qué clientes no tienen pedidos?
Muestra el total de ventas por cliente, incluyendo los que no tienen pedidos
Muestra el total de unidades vendidas por producto, incluyendo los que no tienen ninguna unidad vendidas
Lista de productos finales
Lista de productos que no son finales
Clientes que nunca han comprado un producto que no sea finales
CREATE TABLE `tintas`.`cliente` ( `idcliente` INT NOT NULL AUTO_INCREMENT, `nif` VARCHAR(45) NOT NULL, `razon_social` VARCHAR(45) NOT NULL, PRIMARY KEY (`idcliente`) ); CREATE TABLE `tintas`.`pedido` ( `idpedido` INT NOT NULL AUTO_INCREMENT, `idcliente` INT NOT NULL, `fecha` DATE NOT NULL, `direccion` VARCHAR(45) NOT NULL, `cp` VARCHAR(45) NOT NULL, `poblacion` VARCHAR(45) NOT NULL, PRIMARY KEY (`idpedido`) ); CREATE TABLE `tintas`.`producto` ( `idproducto` INT NOT NULL AUTO_INCREMENT, `referencia` VARCHAR(45) NOT NULL, `precio` DECIMAL(10,2) NOT NULL, `stock` INT NOT NULL, PRIMARY KEY (`idproducto`) ); CREATE TABLE `tintas`.`lineaspedido` ( `idlineaspedido` INT NOT NULL AUTO_INCREMENT, `idpedido` INT NOT NULL, `idproducto` INT NOT NULL, `cantidad` INT NOT NULL, `precio` DECIMAL(10,2) NOT NULL, PRIMARY KEY (`idlineaspedido`) ); CREATE TABLE `composicion` ( `idcomposicion` int(11) NOT NULL AUTO_INCREMENT, `idproducto_final` int(11) NOT NULL, `idproducto_comp` int(11) NOT NULL, `cantidad` int(11) NOT NULL, `orden` smallint(5) unsigned NOT NULL, PRIMARY KEY (`idcomposicion`) ) ALTER TABLE `tintas`.`pedido` ADD INDEX `fk_cliente_idx` (`idcliente` ASC) ; ALTER TABLE `tintas`.`pedido` ADD CONSTRAINT `fk_cliente` FOREIGN KEY (`idcliente`) REFERENCES `tintas`.`cliente` (`idcliente`) ON DELETE RESTRICT ON UPDATE RESTRICT; ALTER TABLE `tintas`.`lineaspedido` ADD INDEX `fk_pedido_idx` (`idpedido` ASC) , ADD INDEX `fk_producto_idx` (`idproducto` ASC) ; ALTER TABLE `tintas`.`lineaspedido` ADD CONSTRAINT `fk_pedido` FOREIGN KEY (`idpedido`) REFERENCES `tintas`.`pedido` (`idpedido`) ON DELETE RESTRICT ON UPDATE RESTRICT, ADD CONSTRAINT `fk_producto` FOREIGN KEY (`idproducto`) REFERENCES `tintas`.`producto` (`idproducto`) ON DELETE RESTRICT ON UPDATE RESTRICT; ALTER TABLE `tintas`.`composicion` ADD INDEX `fk_productof_idx` (`idproducto_final` ASC) , ADD INDEX `fk_productoc_idx` (`idproducto_comp` ASC) ; ALTER TABLE `tintas`.`composicion` ADD CONSTRAINT `fk_productof` FOREIGN KEY (`idproducto_final`) REFERENCES `tintas`.`producto` (`idproducto`) ON DELETE RESTRICT ON UPDATE NO ACTION, ADD CONSTRAINT `fk_productoc` FOREIGN KEY (`idproducto_comp`) REFERENCES `tintas`.`producto` (`idproducto`) ON DELETE RESTRICT ON UPDATE NO ACTION; INSERT INTO `tintas`.`cliente` (`nif`, `razon_social`) VALUES ('b1234354', 'sugus'); INSERT INTO `tintas`.`cliente` (`nif`, `razon_social`) VALUES ('b75756756', 'grefusa'); INSERT INTO `tintas`.`cliente` (`nif`, `razon_social`) VALUES ('b66612333', 's&D'); INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Disolvente', '1', '2000'); INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Emulsionante', '3', '1500'); INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Colorante rojo', '4', '200'); INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Colorante azul', '4', '300'); INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Colorante amarillo', '4', '250'); INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Tinta naranja', '10', '100'); INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Tinta verde', '10', '100'); INSERT INTO `tintas`.`pedido` (`fecha`, `direccion`, `cp`, `poblacion`, `idcliente`) VALUES ('2017-2-2', 'Agla, 5', '08001', 'Barcelona', '1'); INSERT INTO `tintas`.`pedido` (`fecha`, `direccion`, `cp`, `poblacion`, `idcliente`) VALUES ('2017-2-5', 'Agla, 5', '08001', 'Barcelona', '1'); INSERT INTO `tintas`.`pedido` (`fecha`, `direccion`, `cp`, `poblacion`, `idcliente`) VALUES ('2017-2-10', 'Perill, 15', '08012', 'Barcelona',3); INSERT INTO `tintas`.`lineaspedido` (`idpedido`, `idproducto`, `cantidad`, `precio`) VALUES ('1', '7', '10', '10'); INSERT INTO `tintas`.`lineaspedido` (`idpedido`, `idproducto`, `cantidad`, `precio`) VALUES ('2', '6', '10', '10'); INSERT INTO `tintas`.`lineaspedido` (`idpedido`, `idproducto`, `cantidad`, `precio`) VALUES ('3', '1', '10', '3'); INSERT INTO `tintas`.`lineaspedido` (`idpedido`, `idproducto`, `cantidad`, `precio`) VALUES ('3', '3', '5', '4'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('6', '1', '5', '1'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('6', '3', '7', '2'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('6', '5', '10', '3'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('6', '2', '10', '4'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('7', '1', '5', '1'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('7', '4', '7', '2'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('7', '5', '10', '3'); INSERT INTO `tintas`.`composicion` (`idproducto_final`, `idproducto_comp`, `cantidad`, `orden`) VALUES ('7', '2', '10', '4'); select * from cliente join pedido using(idcliente) join lineaspedido using (idpedido) join producto using (idproducto) select * from producto join composicion on producto.idproducto=composicion.idproducto_final join producto p2 on composicion.idproducto_comp=p2.idproducto Insertar el cliente 'Copistería Martínez', 'B21312312' INSERT INTO `tintas`.`cliente` (`nif`, `razon_social`) VALUES ('B21312312', 'Copistería Martínez'); Modificar el precio de todos los productos para incrementar un 10% update producto set precio=precio*1.1 Añadir 10 a la cantidad de la composición de todos los productos update tintas.composicion set cantidad=cantidad+10 Añade el producto 'aerosol rojo', 10 €, 20 INSERT INTO `tintas`.`producto` (`referencia`, `precio`, `stock`) VALUES ('Aerosol rojo', '10', '20'); Elimina los productos cuyo nombre contenga aerosol delete FROM tintas.producto where referencia like '%aerosol%'; ¿Qué productos han comprado los clientes que tienen una 'u' en el nombre? select distinct razon_social, referencia from cliente join pedido using(idcliente) join lineaspedido using (idpedido) join producto using (idproducto) where razon_social like '%u%' ¿Qué clientes no tienen pedidos? SELECT * FROM tintas.cliente left join pedido using (idcliente) where idpedido is null; Muestra el total de ventas por cliente, incluyendo los que no tienen pedidos select razon_social, ifnull(sum(cantidad*precio),0) total from cliente left join pedido using(idcliente) left join lineaspedido using (idpedido) group by razon_social Muestra el total de unidades vendidas por producto, incluyendo los que no tienen ninguna unidad vendidas select referencia, ifnull(sum(cantidad),0) total from producto left join lineaspedido using(idproducto) group by referencia Lista de productos finales select referencia from producto left join composicion on producto.idproducto=composicion.idproducto_comp where idcomposicion is null Lista de productos que no son finales select distinct referencia from producto join composicion on producto.idproducto=composicion.idproducto_comp Clientes que nunca han comprado un producto que no sea finales select * from cliente where idcliente not in ( select distinct idcliente from cliente left join pedido using(idcliente) left join lineaspedido using (idpedido) where idproducto in (select distinct idproducto from producto join composicion on producto.idproducto=composicion.idproducto_comp ) )
Más funciones de mysql
Determinar si una cadena es o no vocal:
CREATE FUNCTION `esvocal`(cadena char(1)) RETURNS tinyint(1) BEGIN RETURN lcase(cadena) regexp '[aeiou]'; END
Contar las vocales de una cadena:
CREATE FUNCTION `vocales`(cadena varchar(100)) RETURNS int(11) BEGIN declare total,cont int default 0; while cont<length(cadena) do set cont=cont+1; if esvocal(substring(cadena,cont,1)) then set total=total+1; end if; end while; RETURN total; END
Cambiar las vocales a mayúsculas y el resto a minúsculas:
CREATE FUNCTION `texto_molon`(cadena varchar(100)) RETURNS varchar(100) BEGIN declare cont int default 1; declare res varchar(100) default ''; declare c varchar(1); while cont<length(cadena) do set c=substring(cadena,cont,1); if esvocal(c) then set res=concat(res,ucase(c)); else set res=concat(res,lcase(c)); end if; set cont=cont+1; end while; RETURN res; END
Ejemplos de funciones en mysql
Prueba de variables:
CREATE FUNCTION `borrame` () RETURNS INTEGER BEGIN declare var int; declare nombre varchar(50); declare num int; set nombre='juan'; set var=5; set var=var+3; select count(actor_id) into num from actor; RETURN num+var+length(nombre); END
Usar un parámetro y un into variable:
CREATE FUNCTION `actores_como`(cadena varchar(50)) RETURNS int(11) BEGIN declare total int; select count(actor_id) into total from actor where first_name like concat('%',cadena,'%') or last_name like concat('%',cadena,'%') ; RETURN total; END
Uso de if:
CREATE FUNCTION `tipo_sueldo`(sueldo INT) RETURNS varchar(10) BEGIN DECLARE a VARCHAR(10); IF sueldo>1500 then RETURN 'ALTO'; ELSE RETURN 'BAJO'; END IF; END
Uso de case:
CREATE FUNCTION `tipo_sueldo_ampliado`(sueldo int) RETURNS varchar(30) BEGIN declare tipo varchar(30); case when sueldo<1500 then set tipo='bajo'; when sueldo<2500 then set tipo='medio'; else set tipo='alto'; end case; RETURN tipo; END
Uso de while:
CREATE FUNCTION `factorial`(num int) RETURNS int(11) BEGIN declare total,cont int default 1; while cont<=num do set total=total*cont; set cont=cont+1; end while; RETURN total; END
Funciones definidas por el usuario, procedimientos y disparadores
Ejemplos sentencias sql
Actores con más de 10 películas
select first_name, last_name, count(film_id) peliculas from actor join film_actor using (actor_id) group by first_name,last_name having peliculas>30
Países con más de 10 ciudades
select country, count(city) cities from country join city using (country_id) group by country having cities>10
Los diez países que más gastan
select country, sum(amount) total, avg(amount) media from country join city using (country_id) join address using (city_id) join customer using (address_id) join payment using (customer_id) group by country order by total desc limit 0,10
El actor con más películas
select first_name, last_name, count(film_id) peliculas from actor join film_actor using(actor_id) group by first_name,last_name order by peliculas desc limit 1
Un castillo japonés hecho con CSS
¿Qué lenguaje de programación aprender?
Ejercicios SQL
Actores con ‘ll’ en el apellido o en el nombre
Actores con id>=100 y menor igual que 200
Una de las dos anteriores o las dos
Insertar actor ‘Juan’ ‘Pérez’
Insertar actor ‘Rosa’ ‘Pi’
Actualizar el nombre del actor con id 10 a ‘PEPE’
Actualizar el nombre de los actores con el id > 200 a ‘Actor’ + id
Borrar los actores con id > 200
select * from sakila.actor where first_name like '%ll%' or last_name like '%ll%'; select * from sakila.actor where actor_id between 100 and 200; select * from sakila.actor where first_name like '%ll%' or last_name like '%ll%' or actor_id between 100 and 200; insert into sakila.actor (first_name,last_name) values ('Juan','Pérez'); insert into sakila.actor (first_name,last_name) values ('Rosa','Pi'); update sakila.actor set first_name='PEPE' where actor_id=10; update sakila.actor set first_name=concat('Actor',actor_id) where actor_id>200; delete from sakila.actor where actor_id>200;