155 lines
6.5 KiB
Markdown
155 lines
6.5 KiB
Markdown
|
---
|
||
|
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;
|
||
|
```
|