Initial commit, ov1-4

This commit is contained in:
2022-10-04 13:53:01 +02:00
commit 35175b6c69
20 changed files with 1489 additions and 0 deletions

117
ov3/byggOgBo_mysql.sql Normal file
View File

@@ -0,0 +1,117 @@
-- MySQL
-- DROP-setninger i tilfelle vi må kjøre scriptet på nytt.
DROP TABLE leilighet;
DROP TABLE andelseier;
DROP TABLE bygning;
DROP TABLE borettslag;
DROP TABLE poststed;
-- Lager tabellene, legger inn NOT NULL- og UNIQUE-krav der det er naturlig
-- Vær forsiktig med å legge inn slike krav, det er vanskelig å forandre
-- dem i ettertid.
CREATE TABLE poststed(
postnr SMALLINT,
poststed VARCHAR(20) NOT NULL,
CONSTRAINT poststed_pk PRIMARY KEY(postnr));
CREATE TABLE borettslag(
bolag_navn VARCHAR(20),
bolag_adr VARCHAR(40) NOT NULL UNIQUE,
etabl_aar SMALLINT NOT NULL,
postnr SMALLINT NOT NULL,
CONSTRAINT borettslag_pk PRIMARY KEY(bolag_navn));
CREATE TABLE bygning(
bygn_id INTEGER NOT NULL AUTO_INCREMENT,
bygn_adr VARCHAR(40) NOT NULL,
ant_etasjer INTEGER DEFAULT 1,
bolag_navn VARCHAR(20) NOT NULL,
postnr SMALLINT NOT NULL,
CONSTRAINT bygning_pk PRIMARY KEY(bygn_id));
CREATE TABLE leilighet(
leil_nr INTEGER NOT NULL AUTO_INCREMENT,
ant_rom SMALLINT NOT NULL,
ant_kvm REAL NOT NULL,
etasje SMALLINT DEFAULT 1,
bygn_id INTEGER NOT NULL,
and_eier_nr INTEGER NOT NULL UNIQUE,
CONSTRAINT leilighet_pk PRIMARY KEY(leil_nr));
CREATE TABLE andelseier(
and_eier_nr INTEGER NOT NULL AUTO_INCREMENT,
fornavn VARCHAR(30) NOT NULL,
etternavn VARCHAR(30) NOT NULL,
telefon CHAR(15),
ansiennitet SMALLINT,
bolag_navn VARCHAR(20) NOT NULL,
CONSTRAINT andelseier_pk PRIMARY KEY(and_eier_nr));
-- Fremmednøkler
ALTER TABLE borettslag
ADD CONSTRAINT borettslag_fk1 FOREIGN KEY(postnr)
REFERENCES poststed(postnr);
ALTER TABLE bygning
ADD CONSTRAINT bygning_fk1 FOREIGN KEY(postnr)
REFERENCES poststed(postnr);
ALTER TABLE bygning
ADD CONSTRAINT bygning_fk2 FOREIGN KEY(bolag_navn)
REFERENCES borettslag(bolag_navn);
ALTER TABLE leilighet
ADD CONSTRAINT leilighet_fk1 FOREIGN KEY(bygn_id)
REFERENCES bygning(bygn_id);
ALTER TABLE leilighet
ADD CONSTRAINT leilighet_fk2 FOREIGN KEY(and_eier_nr)
REFERENCES andelseier(and_eier_nr);
ALTER TABLE andelseier
ADD CONSTRAINT andelseier_fk2 FOREIGN KEY(bolag_navn)
REFERENCES borettslag(bolag_navn);
-- Legger inn gyldige data
INSERT INTO poststed(postnr, poststed) VALUES(2020, 'Skedsmokorset');
INSERT INTO poststed(postnr, poststed) VALUES(6408, 'Aureosen');
INSERT INTO poststed(postnr, poststed) VALUES(7033, 'Trondheim');
INSERT INTO poststed(postnr, poststed) VALUES(7020, 'Trondheim');
INSERT INTO borettslag(bolag_navn, bolag_adr, etabl_aar, postnr) VALUES('Tertitten', 'Åsveien 100', 1980, 7020);
INSERT INTO borettslag(bolag_navn, bolag_adr, etabl_aar, postnr) VALUES('Sisiken', 'Brurød', 1990, 7033);
INSERT INTO borettslag(bolag_navn, bolag_adr, etabl_aar, postnr) VALUES('Lerken', 'Storgt 5', 2000, 6408);
INSERT INTO andelseier(and_eier_nr, fornavn, etternavn, telefon, ansiennitet, bolag_navn)
VALUES(DEFAULT, 'Even', 'Trulsbo', '56667743', 3, 'Tertitten');
INSERT INTO andelseier(and_eier_nr, fornavn, etternavn, telefon, ansiennitet, bolag_navn)
VALUES(DEFAULT, 'Anna', 'Olsen', '45674588', 10, 'Tertitten');
INSERT INTO andelseier(and_eier_nr, fornavn, etternavn, telefon, ansiennitet, bolag_navn)
VALUES(DEFAULT, 'Ingrid', 'Olsen', '45785388', 8, 'Tertitten');
INSERT INTO andelseier(and_eier_nr, fornavn, etternavn, telefon, ansiennitet, bolag_navn)
VALUES(DEFAULT, 'Arne', 'Torp', '78565388', 7, 'Tertitten');
INSERT INTO andelseier(and_eier_nr, fornavn, etternavn, telefon, ansiennitet, bolag_navn)
VALUES(DEFAULT, 'Arne', 'Martinsen', '78555388', 4, 'Sisiken');
INSERT INTO bygning(bygn_id, bygn_adr, ant_etasjer, bolag_navn, postnr) VALUES(DEFAULT, 'Åsveien 100a', 3, 'Tertitten', 7020);
INSERT INTO bygning(bygn_id, bygn_adr, ant_etasjer, bolag_navn, postnr) VALUES(DEFAULT, 'Åsveien 100b', 3, 'Tertitten', 7020);
INSERT INTO bygning(bygn_id, bygn_adr, ant_etasjer, bolag_navn, postnr) VALUES(DEFAULT, 'Åsveien 100c', 6, 'Tertitten', 7020);
INSERT INTO bygning(bygn_id, bygn_adr, ant_etasjer, bolag_navn, postnr) VALUES(DEFAULT, 'Storgt 10', 2, 'Sisiken', 7020);
-- bruker defaultverdien for antall etasjer
INSERT INTO bygning(bygn_id, bygn_adr, bolag_navn, postnr) VALUES(DEFAULT, 'Åsveien 100', 'Tertitten', 7020);
INSERT INTO leilighet(leil_nr, ant_rom, ant_kvm, etasje, bygn_id, and_eier_nr) VALUES(DEFAULT, 5, 110, 3, 1, 1);
INSERT INTO leilighet(leil_nr, ant_rom, ant_kvm, etasje, bygn_id, and_eier_nr) VALUES(DEFAULT, 5, 110, 3, 1, 2);
INSERT INTO leilighet(leil_nr, ant_rom, ant_kvm, etasje, bygn_id, and_eier_nr) VALUES(DEFAULT, 2, 50, 1, 3, 3);
-- bruker defaultverdien for etasje
INSERT INTO leilighet(leil_nr, ant_rom, ant_kvm, bygn_id, and_eier_nr) VALUES(DEFAULT, 5, 110, 1, 4);

BIN
ov3/ing_db_ov3_sql1.pdf Normal file

Binary file not shown.

154
ov3/losning.md Normal file
View File

@@ -0,0 +1,154 @@
---
geometry: margin=30mm
author: Felix Albrigtsen
...
# Øving 3: SQL del 1 (obligatorisk)
#### Skrevet av Felix Albrigtsen, felixalb@stud.ntnu.no
```
Innleveringsfrist: se Blackboard
Tidligste godkjenning: datoer blir annonsert
Løsningsforslag legges ut i etterkant.
Alle obligatoriske øvinger må være godkjente for å få karakter i emnet.
```
# Oppgave 1 SQL
I denne oppgaven skal vi bruke en borettslag-databasen. Bruk følgende sql-script:
byggOgBo_mysql.sql (tekstfil som finnes i BB). Scriptet inneholder data som oppgavene
nedenfor spør etter.
Sett opp SELECT-setninger som besvarer spørsmålene nedenfor. Kun én setning pr oppgave.
Databaseskjema:
```
MariaDB [idatt2103_ov3]> SELECT * FROM andelseier LIMIT 1;
+-------------+---------+-----------+----------+-------------+------------+
| and_eier_nr | fornavn | etternavn | telefon | ansiennitet | bolag_navn |
+-------------+---------+-----------+----------+-------------+------------+
| 1 | Even | Trulsbo | 56667743 | 3 | Tertitten |
+-------------+---------+-----------+----------+-------------+------------+
MariaDB [idatt2103_ov3]> SELECT * FROM borettslag LIMIT 1;
+------------+-----------+-----------+--------+
| bolag_navn | bolag_adr | etabl_aar | postnr |
+------------+-----------+-----------+--------+
| Lerken | Storgt 5 | 2000 | 6408 |
+------------+-----------+-----------+--------+
MariaDB [idatt2103_ov3]> SELECT * FROM bygning LIMIT 1;
+---------+---------------+-------------+------------+--------+
| bygn_id | bygn_adr | ant_etasjer | bolag_navn | postnr |
+---------+---------------+-------------+------------+--------+
| 1 | Åsveien 100a | 3 | Tertitten | 7020 |
+---------+---------------+-------------+------------+--------+
MariaDB [idatt2103_ov3]> SELECT * FROM leilighet LIMIT 1;
+---------+---------+---------+--------+---------+-------------+
| leil_nr | ant_rom | ant_kvm | etasje | bygn_id | and_eier_nr |
+---------+---------+---------+--------+---------+-------------+
| 1 | 5 | 110 | 3 | 1 | 1 |
+---------+---------+---------+--------+---------+-------------+
MariaDB [idatt2103_ov3]> SELECT * FROM poststed LIMIT 1;
+--------+---------------+
| postnr | poststed |
+--------+---------------+
| 2020 | Skedsmokorset |
+--------+---------------+
```
# Kommentar:
Flere steder i oppgavene har jeg måtte gjøre antagelser om spørsmålene, så alle løsningene er ikke deifnitive. For eksempel "Sortert etter antallet" i oppgave 10 kan være enten stigende eller avtagende. I tillegg har jeg testet på eksempeldataen gitt i blackboard, men prøvd å gjøre setningene kompatible med alle verdier der det er mulig. Dette resulterer i noen ekstra sjekker i besvarelsen, som ikke har noen effekt på eksempeldataen, men for eksempel ville påvirket leiligheter uten noen andelseier.
# Besvarelse:
## 1. Finn alle borettslag etablert i årene 1975-1985.
```sql
SELECT * FROM borettslag WHERE etabl_aar >= 1975 AND etabl_aar <= 1985;
```
## 2. Skriv ut en liste over andelseiere. Listen skal ha linjer som ser slik ut (tekster i kursiv er data fra databasen): "__fornavn etternavn__, ansiennitet: __ansiennitet år__". Listen skal være sortert på ansiennitet, de med lengst ansiennitet øverst.
```sql
SELECT CONCAT(fornavn, ' ', etternavn, ', ansiennitet: ', ansiennitet) AS andelseier
FROM andelseier ORDER BY ansiennitet DESC;
```
## 3. I hvilket år ble det eldste borettslaget etablert?
```sql
SELECT etabl_aar FROM borettslag ORDER BY etabl_aar ASC LIMIT 1;
```
## 4. Finn adressene til alle bygninger som inneholder leiligheter med minst tre rom.
```sql
SELECT bygn_adr FROM bygning b RIGHT JOIN (SELECT * FROM leilighet WHERE ant_rom >= 3) l
ON b.bygn_id = l.bygn_id GROUP BY bygn_adr;
```
## 5. Finn antall bygninger i borettslaget "Tertitten".
```sql
SELECT COUNT(*) FROM bygning WHERE bolag_navn = "Tertitten";
```
## 6. Lag en liste som viser antall bygninger i hvert enkelt borettslag. Listen skal være sortert på borettslagsnavn. Husk at det kan finnes borettslag uten bygninger - de skal også med.
```sql
SELECT COUNT(b.bygn_id) as "Antall Bygninger", bl.bolag_navn FROM borettslag bl LEFT
JOIN bygning b ON b.bolag_navn = bl.bolag_navn GROUP BY bolag_navn ORDER BY bl.bolag_navn;
```
## 7. Finn antall leiligheter i borettslaget "Tertitten".
```sql
SELECT COUNT(*) FROM leilighet l JOIN bygning b ON l.bygn_id = b.bygn_id
WHERE b.bolag_navn = "Tertitten";
```
## 8. Hvor høyt kan du bo i borettslaget "Tertitten"?
```sql
SELECT ant_etasjer FROM bygning WHERE bolag_navn = "Tertitten" ORDER BY
ant_etasjer DESC LIMIT 1;
```
## 9. Finn navn og nummer til andelseiere som ikke har leilighet.
```sql
SELECT fornavn, etternavn, telefon FROM andelseier a LEFT JOIN leilighet l
ON a.and_eier_nr = l.and_eier_nr WHERE leil_nr IS NULL;
```
## 10. Finn antall andelseiere pr borettslag, sortert etter antallet. Husk at det kan finnes borettslag uten andelseiere - de skal også med.
```sql
SELECT b.bolag_navn, COUNT(a.and_eier_nr) as antall_andelseiere FROM andelseier a
RIGHT JOIN borettslag b ON b.bolag_navn = a.bolag_navn GROUP BY b.bolag_navn
ORDER BY antall_andelseiere DESC;
```
## 11. Skriv ut en liste over alle andelseiere. For de som har leilighet, skal leilighetsnummeret skrives ut.
```sql
SELECT a.fornavn, a.etternavn, l.leil_nr FROM andelseier a LEFT JOIN leilighet l
ON a.and_eier_nr = l.and_eier_nr;
```
## 12. Hvilke borettslag har leiligheter med eksakt 4 rom?
```sql
SELECT bolag_navn FROM leilighet l JOIN bygning b ON l.bygn_id = b.bygn_id
WHERE l.ant_rom = 4 GROUP BY bolag_navn;
```
**NB:** tomt resultat, ingen leiligheter i eksempeldataen har 4 rom.
## 13. Skriv ut en liste over antall andelseiere pr postnr og poststed, begrenset til de som bor i leiligheter tilknyttet et borettslag. Husk at postnummeret til disse er postnummeret til bygningen de bor i, og ikke postnummeret til borettslaget. Du trenger ikke ta med poststeder med 0 andelseiere. (Ekstraoppgave: Hva hvis vi vil ha med poststeder med 0 andelseiere?)
```sql
SELECT COUNT(l.and_eier_nr) AS 'Antall andelseiere', b.poststed, b.postnr FROM
leilighet l LEFT JOIN (SELECT b.bygn_id,p.poststed,p.postnr FROM bygning b LEFT JOIN
poststed p ON b.postnr = p.postnr) b ON l.bygn_id = b.bygn_id WHERE
l.and_eier_nr IS NOT NULL GROUP BY b.poststed;
```
kortere løsning, men bruker natural join(ikke anbefalt)
```sql
SELECT l.and_eier_nr, p.poststed, p.postnr FROM leilighet l NATURAL JOIN bygning b
NATURAL JOIN poststed p WHERE l.and_eier_nr IS NOT NULL;
```

BIN
ov3/oving3.pdf Normal file

Binary file not shown.