582104 Ohjelmistojen mallintaminen, olioja relaatiomallinnuksen suhteesta 1
Tietojen pysyvyys liiketoiminnan edellytys Tällä kurssilla on keskitytty oliomenetelmiä hyödyntävään ohjelmistojen mallintamiseen Rinnakkainen kurssi Tietokantojen perusteet käsittelee relaatioteoriaa hyöndyntävään tietosisällön mallintamiseen Oliomallinnus on de facto standardi ohjelmistojen toiminnan mallintamiseksi Relaatiomallinnus on de facto standardi pysyväisluonteisen tiedon mallintamiseksi Toimiva liiketoimintajärjestelmä käyttää pysyväisluonteista dataa 2
Sisältö Tietosisällön mallintaminen relaatio- ja oliomaailmassa Olio- ja relaatiomallin yhteensovittaminen (ORM) Hibernate kehyksen pikaesittely Kirjastoesimerkin sovittaminen Hibernatesovellukseksi 3
Tiedon mallintaminen relaatiokannassa Relaatiokannassakin mallinnetaan järjestelmän tietosisältöä useammalla abstraktiotasolla Kohdealuetta mallinnetaan esimerkiksi ER-kaavioilla (entity-relationship) Taulukaavioin (SQL-taulut) Fyysisen tallennuksen kaavioin Järjestelmän suorituskyky tärkeä tekijä, tarkoituksena löytää tehokkaat tallennusformaatit Tietokannan hallintajärjestelmä käsittelee miljoonia tai miljardeja rivejä 4
Käsitetaso, rakennetaso, tallennustaso 5
Oliomallinnuksen käyttäminen relaatiokannan yhteydessä UML-luokkamallilla pystytään monesti saavuttamaan ER-kaaviota vastaava ilmaisuvoima Useimmat oliokielet eivät kuitenkaan tue tehokasta tietojen säilyttämistä ja hakemista, joten monesti tietojen pysyvyyden varmistamiseksi käytetään SQL-kantaa Olio- ja relaatiomallien yhteensovittamiseksi voidaan käyttää olio-relaatiokuvauksia (object-torelational mapping, ORM) 6
Relaation esittäminen taulumuodossa Relaatiot esitetään SQL:ssä taulumuodossa Käsitteiden väliset yhteydet esitetään viite- ja pääavainten välisten arvojen samuudella 7
Olioiden esittäminen UML:ssä 8
Olioiden esittäminen relaatiokannassa Yksinkertaisessa tapauksessa olioiden tallentaminen relaatiokantaan on suoraviivaista class Asiakas { } String tunnus; String nimi; Set<Laskutustieto> laskutustiedot; class Laskutustieto { } int tilinumero; boolean luottotili; decimal summa; Asiakas omistaja; 9
SQL-taulurakenteen luomislausekkeet create table ASIAKAS ( ); TUNNUS VARCHAR(15) NOT NULL PRIMARY KEY, NIMI VARCHAR(50) NOT NULL, SALASANA CHAR(32) NOT NULL create table LASKUTUSTIETO ( TILINUMERO INT NOT NULL PRIMARY KEY, LUOTTOTILI BOOLEAN, RAHAMAARA DECIMAL(10, 3), OMISTAJA FOREIGN KEY REFERENCES ASIAKAS ); 10
Oliomallin relaatiovastineet moniselitteisiä Edellisessä esimerkissä ei tunnettu asiakkaan osoitetta Vakiotapa osoitteen mallintamiseen on käyttää kompositiota Miten relaatiomalli muuttuu? Ovatko Osoite-luokan kentät Asiakas-taulun sarakkeita? Vai oma erillinen taulunsa, viittaukset viiteavaimin? Olio/relaatiomallin epäyhteensopivuus yleistä 11
Olio- ja relaatioparadigmojen epäyhteensopivuus Olio- ja relaatiomallien yhteensovittamisessa syntyviä ongelmia syntyy esim. seuraavista: Mallinnuksen tarkkuustasosta (granularity) Luokkahierarkioiden tallennustavassa Olioiden identiteettiongelmat Yhteyksien tallennustavasta Yhteyksien navigoituvuustavoissa Joidenkin arvioiden mukaan jopa 30% oliojärjestelmän koodiriveistä voi olla puuduttavaa, toisteista olio- ja relaatiomuunnosten toteuttamista 12
Mallinnuksen tarkkuustaso-ongelma Oliomallinnuksessa voidaan tietosisältöä jakaa luokkiin vapaasti tarpeen mukaan Relaatiomallinnuksessa käytetään jaotellaan taulu sarakkeisiin ja riveihin uusien tietotyyppien esitteleminen monesti vaikeaa, järjestelmäkohtaista create table ASIAKAS ( ); TUNNUS VARCHAR(15) NOT NULL PRIMARY KEY, NIMI VARCHAR(50) NOT NULL, SALASANA CHAR(32) NOT NULL OSOITE_KATUOSOITE VARCHAR(50), OSOITE_KUNTA VARCHAR(15), OSOITE_POSTINUMERO CHAR(5), 13
Luokkahierarkian tallennusongelma Luokkahierarkian laajennosmekanismille ei ole suoraa vastinetta relaatiomallissa Kukin aliluokka määrittelee oman tietosisältönsä, vaikuttaa taulujen rakenteeseen Kullakin oliolla on luokkansa, vaikuttaa taulujen rakenteeseen Taulujen viiteavaimet voivat osoittaa vain yhteen viitattuun tauluun 14
Olioidentiteetin ongelma Asiakkaan identifioivaksi kentäksi määriteltiin käyttäjätunnus Esim. Javassa olioita voidaan tunnistaa viitesamuuden ja identiteettisamuuden kannalta Viitesamuus: ab == bc Identiteettisamuus: ab.equals(bc) Tietokannan identiteettikenttä tuo kolmannen samuuden käsitteen Esimerkiksi tietojen muokkaustilanteessa voi järjestelmässä olla olioita, jotka viite- ja identiteettitasolla ovat eriäviä, mutta tietokannan kannalta samoja 15
Yhteyksien tallennustapaongelma UML:ssä on mahdollista merkitä navigoituvuussääntöjä Vastaavat olion hallussaan pitämää viitettä Olioviite aina yksisuuntainen Molempiin suuntiin navigoitavat yhteydet toteutetaan kahtena olioviitteenä Relaatiokannassa viitteet ovat luonnostaan kaksisuuntaisia Viiteavain viittaa toisen taulun pääavaimeen 16
Olioverkkojen navigoituvuusongelma Olio-ohjelmalle on luontaista käsitellä olioverkkoja ja delegoida palvelun toteuttaminen osaolioille Etukäteen palvelun toteuttamiseen osallistuvien olioiden tunnistaminen vaikeaa Kunkin olion noutaminen tietokannasta siinä vaiheessa, kun olion palveluita tarvitaan, on tehotonta ns. n+1 kyselyn ongelma 17
Olio/relaatiomallin kuvausten käyttäminen Olio- ja relaatiomallien välisten yhteyksien kuvaamisen käytetään monesti erillistä työkalua ORM-kuvaus määrittelee miten oliomalli esitetään relaatiokannassa ORM-työkalu sisältää rajapinnat olioiden tietokannasta lataamiseksi ja tallentamiseksi Javassa standardi Java Persistence API (JPA) 18
ORM-työkalun ongelmakenttä ORM-työkalun käytännöt vastaavat mm. seuraaviin kysymyksiin: Minkälainen rakenne pysyväisluokilla tulee olla Miten O/R-kuvaukset esitetään Miten periytymisrakenteet tallennetaan Olioidentiteettiin liittyvät kysymykset Miten sovelluslogiikka kudotaan tallennuskerroksen kanssa yhteen Mikä on pysyväisluokkien olioiden elinkaari Miten pysyväisolioita noudetaan tietokannasta Miten pysyväisolioverkkoja noudetaan tehokkaasti 19
Hello World-esimerkki Hibernatella Hibernate on ehkäpä käytetyin ORM-toteutus Javaympäristössä Alkuperäinen JPA-määrittely perustuu Hibernaten esittelemiin rajapintoihin Esimerkiksi linkitetyn viestilistan tallentaminen Hibernatella 20
Olioita voidaan käsitellä normaalisti public class HoiMaailma1 { public static void main(string args[]) { } } Message msg = new Message( Terve maailma ); System.out.println(msg.getText()); 21
Tietokantakäytössä sessiot ja transaktiot Tietokantaa käytettäessä tietokannan hallintajärjestelmä erottelee eri käyttäjien muutokset kannan sisältöön Muutokset tietokantaan jaetaan erillisiin transaktioihin Hibernate käyttää sisäistä session käsitettä tietokantayhteyden ohessa { Session session = getsessionfactory().opensession(); Transaction tx = session.begintransaction(); Message msg = new Message( Terve kaverit! ); } session.save(msg); tx.commit(); session.close() 22
Luokkien kuvaus tietokantatauluiksi Luokkien tallennustapa tietokantaan kuvataan XMLkuvauksin, esim: <hibernate-mapping> <class name= Message table= MESSAGES > <id name= id column= MSG_ID > <generator class= increment /> </id> <property name= text column= MSG_TEXT /> <many-to-one name= nextmessage cascade= all column= NEXT_MESSAGE_ID /> </class> </hibernate-mapping> 23
Vastaava tietokantakaavio Relaatiotietokannasta löytyvien taulujen tulisi vastata luokkakuvauksia Mahdollista käyttää luokkakuvauksista automaattisesti tuotettua taulukaaviota Hello-world esimerkissä: 24
Olioiden tietojen lataaminen Olioiden tietokantaan talletettu tila voidaan ladata sessiota käyttäen: { Session session = HibernateUtil.getSessionFactory().openSession(); Transaction newtransaction = session.begintransaction(); List messages = session.createquery("from Message").list(); System.out.println(messages.size() + " message(s) found:"); for (Iterator iter = messages.iterator(); iter.hasnext();) { Message msg = (Message) iter.next(); System.out.println("[" + msg.getid() + "]: " + msg.gettext()); } } newtransaction.commit(); session.close(); 25
Olioiden tietojen päivittäminen Muistissa olevien olioiden tietoja voidaan päivittää session kautta: { } Session session = HibernateUtil.getSessionFactory().openSession(); Transaction newtransaction = session.begintransaction(); Message oldmsg = (Message) session.load(message.class, msgid); oldmsg.settext(newmsg); newtransaction.commit(); newsession.close(); 26
Kirjastojärjestelmän toinen iteraatio Toteutetaan järjestelmän käsittelemä data pysyväksi Toteutus tehdään Hibernate-kehystä hyödyntäen Esitellään ensi viikolla ohjelmointieksperimentissä Ohjelmointieksperimentissä osallistujia pyydetään tutustumaan toisen iteraation lopputuloksena syntyneeseen ohjelmistoon Ja tekemään muutamia ylläpidollisia muutoksia siihen Kullakin osallistujalla omanlaisensa versio, tutkimuksen kohteena eri toteutusrakenteiden vaikutus ohjelmiston ylläpidettävyyteen 27