Relationell auch ohne SQL Relationale Datenbanken mit ScalaQuery nutzen Stefan Zeiger Commerzbank AG.

Slides:



Advertisements
Similar presentations
5 Copyright © 2005, Oracle. All rights reserved. Accessing the Database with Servlets.
Advertisements

16 Copyright © 2005, Oracle. All rights reserved. Using JDBC to Access the Database.
Patterns for Slick database applications Jan Christopher Vogt, EPFL Slick Team #scalax.
Slick SQL Interaction in Scala
Persistence Jim Briggs 1. 2 Database connectivity: JDBC Java Database Connectivity An API for connecting Java programs (applications, applets and servlets)
CE203 - Application Programming Autumn 2013CE203 Part 51 Part 5.
Java II--Copyright © Tom Hunter. Chapter 18 JDBC.
Exercises of the Tutorial on Advanced Web Programming Authors: Miroslava Mitrovic Dragan Milicev Nino.
NOEA/IT FEN - Databases/SQL1 More about SQL Nested SELECT Other Joins Views Index.
Scaling Scala to the Database Jan Christopher Vogt, EPFL Stefan Zeiger, Typesafe.
DATA MODELS A collection of conceptual tools for describing data, data relationships, data semantics, and consistency constraints. Provide a way to describe.
Murach’s Java SE 6, C21© 2007, Mike Murach & Associates, Inc.Slide 1.
1 JDBC Java Database Connectivity. 2 c.pdf
1 Lecture 05: Database Programming (JDBC). 2 Outline JDBC overview JDBC API Reading: Chapter 10.5 Pointbase Developer Manual.
JDBC Overview Autumn 2001 Lecturer: C. DeJong. Relational Databases widespread use used via SQL (Structured Query Language) freely available powerful.
The Relational Model Codd (1970): based on set theory Relational model: represents the database as a collection of relations (a table of values --> file)
Session-01. Hibernate Framework ? Why we use Hibernate ?
DAT702.  Standard Query Language  Ability to access and manipulate databases ◦ Retrieve data ◦ Insert, delete, update records ◦ Create and set permissions.
UFCE4Y UFCE4Y-20-3 Components and Services Julia Dawson.
Data Persistence and Object-Relational Mapping Slides by James Brucker, used with his permission 1.
Object Persistence and Object-Relational Mapping James Brucker.
Introduction to JPA Java Persistence API Introduction to JPA.
CSE446 S OFTWARE Q UALITY M ANAGEMENT Spring 2014 Yazılım ve Uyguluma Geliştirme Yöneticisi Orhan Başar Evren.
Object and object-relational databases 1. Object databases vs. Object-relational databases Object databases Stores complex objects – Data + functions.
What is Slick and why we use it? Mark van der Tol.
Training - Day 3 OJB. What is OR Mapping? OR Mapping is the mapping of relational database tables to objects (Java Objects in our case) Many OR Mapping.
Advance Computer Programming Java Database Connectivity (JDBC) – In order to connect a Java application to a database, you need to use a JDBC driver. –
1 CSC 440 Database Management Systems JDBC This presentation uses slides and lecture notes available from
UPortal Developers MIT August 2004 Persistence Strategy for uPortal 3 Mike DeSimone the r-smart group
Agenda What is Hibernate Spring Integration Questions Overview
JDBC. What is JDBC JDBC is an acronym for –Java Data Base Connectivity. It allows java/jsp program to connect to any database.
Java Utility Classes CS 21b. Some Java Utility Classes Vector Hashtable StringTokenizer * import java.util.*;
Using Java with PSQL and Oracle Thanks to Drs. Raj and Liu for sharing some of these slides TRUDY: ALSO NEED SAMPLE PROGRAM.
CHAPTER 14 USING RELATIONAL DATABASES TO PROVIDE OBJECT PERSISTENCE (ONLINE) © 2013 Pearson Education, Inc. Publishing as Prentice Hall 1 Modern Database.
12-CRS-0106 REVISED 8 FEB 2013 CSG2H3 Object Oriented Programming.
Introduction to JDBC Michelle Lee, Ye Wu & Jeff Offutt SWE 432 Design and Implementation of Software for the Web.
JDBC. JDBC stands for Java Data Base Connectivity. JDBC is different from ODBC in that – JDBC is written in Java (hence is platform independent, object.
Sadegh Aliakbary. Copyright ©2014 JAVACUP.IRJAVACUP.IR All rights reserved. Redistribution of JAVACUP contents is not prohibited if JAVACUP.
Topic : Hibernate 2: Object Persistence and ORM Kaster Nurmukan.
JPA / HIBERNATE CSCI 6370 Nilayan Bhattacharya Sanket Sable.
Topic : Hibernate 3:Advanced ORM Kaster Nurmukan.
JDBC. Java.sql.package The java.sql package contains various interfaces and classes used by the JDBC API. This collection of interfaces and classes enable.
Presentation On How To Create Connection To A Database.
Li Tak Sing COMPS311F. Database programming JDBC (Java Database Connectivity) Java version of ODBC (Open Database Connectivity) ODBC provides a standard.
Topic : Hibernate 1 Kaster Nurmukan. An ORM tool The problem fixed by ORM Advantage Hibernate Hibernate Basic –Hibernate sessionFactory –Hibernate Session.
Tasks Needed for MissionMapEditor Martin Q. Zhao September 18, 2010.
AND OBJECT-ORIENTED DATABASES OBJECT-RELATIONAL DATABASES.
2006 JavaOne SM Conference | Session BOF-0367 JDBC™Technology-based Metadata Recipes Mahmoud Parsian Server Architect LimeLife, Inc.
ECMM6018 Enterprise Networking For Electronic Commerce Tutorial 6 CGI/Perl and databases.
Topic : Hibernate 1 Kaster Nurmukan. An ORM tool Used in data layer of applications Implements JPA.
SQL and Java The vision for Java is to be the concrete and nails that people use to build this incredible network system that is happening all around us.
Introduction to JDBC Instructor: Mohamed Eltabakh 1.
1 JDBC – Java Database Connectivity CS , Spring 2010.
Introduction to ORM Hibernate Hibernate vs JDBC. May 12, 2011 INTRODUCTION TO ORM ORM is a programming technique for converting data between relational.
JDBC I IS Why do we have databases?
SQL pepper. Why SQL File I/O is a great deal of code Optimal file organization and indexing is critical and a great deal of code and theory implementation.
CS422 Principles of Database Systems Object-Relational Mapping (ORM) Chengyu Sun California State University, Los Angeles.
CS 440 Database Management Systems Stored procedures & OR mapping 1.
Hibernate Java Persistence API. What is Persistence Persistence: The continued or prolonged existence of something. Most Applications Achieve Persistence.
JDBC Statements The JDBC Statement, CallableStatement, and PreparedStatement interfaces define the methods and properties that enables to send SQL or PL/SQL.
CS3220 Web and Internet Programming Database Access with JDBC
Chengyu Sun California State University, Los Angeles
JDBC – Java Database Connectivity
Chengyu Sun California State University, Los Angeles
JDBC.
Design and Implementation of Software for the Web
Hibernate Bayu Priyambadha, S.Kom.
Using a Database with JDBC
CS3220 Web and Internet Programming Database Access with JDBC
CS3220 Web and Internet Programming Database Access with JDBC
Presentation transcript:

Relationell auch ohne SQL Relationale Datenbanken mit ScalaQuery nutzen Stefan Zeiger Commerzbank AG

Herbstcampus 2011 – Relationell auch ohne SQL2 Relationale Datenbanken Größere Anwendungen brauchen oft Datenbanken Relationales Modell verhindert Silobildung

Herbstcampus 2011 – Relationell auch ohne SQL3 Wozu? Wir haben doch JDBC

Herbstcampus 2011 – Relationell auch ohne SQL4 Wozu? Wir haben doch JDBC def usersMatching(pattern: String)(conn: Connection) = { val st = conn.prepareStatement("select id, name from users where name like ?") try { st.setString(1, pattern) val rs = st.executeQuery() try { val b = new ListBuffer[(Int, String)] while(rs.next) b.append((rs.getInt(1), rs.getString(2))) b.toList } finally rs.close() } finally st.close() } Class.forName("org.h2.Driver") val conn = DriverManager.getConnection("jdbc:h2:test1") try { println(usersMatching("%zeiger%")(conn)) } finally conn.close()

Herbstcampus 2011 – Relationell auch ohne SQL5 JDBC Gute Grundlage für Frameworks Zu niedrige Abstraktionsebene für Anwendungen

Herbstcampus 2011 – Relationell auch ohne SQL6 ScalaQuery: Simple Queries val usersMatching = query[String, (Int, String)] ("select id, name from users where name like ?") Database.forURL("jdbc:h2:test1", driver = "org.h2.Driver") withSession { println(usersMatching("%zeiger%").list) }

Herbstcampus 2011 – Relationell auch ohne SQL7 Wozu? Wir haben doch ORMs Object/Relational Mapping Tools Hibernate, Toplink, JPA Lösen des Problems 80% 50% 95%

Herbstcampus 2011 – Relationell auch ohne SQL8 Relationales Modell Relational Model: Relation Attribute Tuple Relation Value Relation Variable COF_NAMESUP_IDPRICE Colombian French_Roast Espresso Colombian_Decaf French_Roast_Decaf TABLE COFFEES Beispiele aus:

Herbstcampus 2011 – Relationell auch ohne SQL9 Impedance Mismatch: Konzepte Object-Oriented: Identity State Behaviour Encapsulation Relational: Identity State Behaviour Encapsulation : Transactional

Herbstcampus 2011 – Relationell auch ohne SQL10 Impedance Mismatch: Retrieval Strategies Colombian French_Roast Espresso Colombian_Decaf French_Roast_Decaf Espresso Price:9.99 Supplier:The High Ground select COF_NAME from COFFEES select c.*, s.SUP_NAME from COFFEES c, SUPPLIERS s where c.COF_NAME = ? and c.SUP_ID = s.SUP_ID

Herbstcampus 2011 – Relationell auch ohne SQL11 Impedance Mismatch: Retrieval Strategies

Herbstcampus 2011 – Relationell auch ohne SQL12 Impedance Mismatch: Retrieval Strategies def getAllCoffees(): Seq[Coffee] = … def printLinks(s: Seq[Coffee]) { for(c <- s) println(c.name ) } def printDetails(c: Coffee) { println(c.name) println("Price: " + c.price) println("Supplier: " + c.supplier.name) } + " " + c.price

Herbstcampus 2011 – Relationell auch ohne SQL13 O/R-Mapper Falsche Abstraktionsebene Nicht transparent

Herbstcampus 2011 – Relationell auch ohne SQL14 Object/Relational Mapping is The Vietnam of Computer Science (Ted Neward)

Herbstcampus 2011 – Relationell auch ohne SQL15 A Better Match: Functional Programming Relation Attribute Tuple Relation Value Relation Variable case class Coffee(name: String, supplierId: Int, price: Double) val coffees = Set( Coffee("Colombian", 101, 7.99),, Coffee("Espresso", 150, 9.99) ) Coffee("French_Roast", 49, 8.99) - mutable state in the DB

Herbstcampus 2011 – Relationell auch ohne SQL16 ScalaQuery Session-Management Typsichere Queries in Scala + Insert, Update, Delete, DDL Direkte SQL-Statements Gemeinsames API zur Ausführung beider Arten von Statements org.scalaquery.session org.scalaquery.ql org.scalaquery.simple org.scalaquery

Herbstcampus 2011 – Relationell auch ohne SQL17 Session Management: Database JDBC kennt zwei Connection-Management- Modelle: DriverManager und DataSource Wie mit DriverManager Connections zu einer URL öffnen: Database.forURL(…) Ein DataSource-Objekt verwenden: Database.forDataSource(…) Ein DataSource-Objekt über einen JNDI- Namen holen: Database.forName(…)

Herbstcampus 2011 – Relationell auch ohne SQL18 Session Management: Session Alle Zugriffe auf die Datenbank erfolgen über ein Session -Objekt Wrapper für java.sql.Connection Oft als implizites Objekt verwendet: Database.threadLocalSession Kein Caching von Connections und PreparedStatements

Herbstcampus 2011 – Relationell auch ohne SQL19 Session Management s: Session => (s) import org.scalaquery.session._ val db = Database.forURL("jdbc:h2:mem:test1", driver = "org.h2.Driver") db withSession { doSomethingWithSession } import org.scalaquery.session.Database.threadLocalSession s withTransaction { } import org.scalaquery.session._ import org.scalaquery.session.Database.threadLocalSession val db = Database.forURL("jdbc:h2:mem:test1", driver = "org.h2.Driver") db withTransaction { doSomethingWithSession }

Herbstcampus 2011 – Relationell auch ohne SQL20 Typsichere Queries: Scala-Collections case class Coffee( name: String, supID: Int, price: Double ) val coffees = List( Coffee("Colombian", 101, 7.99), Coffee("Colombian_Decaf", 101, 8.99), Coffee("French_Roast_Decaf", 49, 9.99) ) val l = for { c <- coffees if c.supID == 101 } yield (c.name, c.price) l.foreach { case (n, p) => println(n + ": " + p) } Scala Collections

Herbstcampus 2011 – Relationell auch ohne SQL21 Typsichere Queries: Query Language val Coffees = new Table[(String, Int, Double)]("COFFEES") { def name = column[String]("COF_NAME", O.PrimaryKey) def supID = column[Int]("SUP_ID") def price = column[Double]("PRICE") def * = name ~ supID ~ price } Coffees.insertAll( ("Colombian", 101, 7.99), ("Colombian_Decaf", 101, 8.99), ("French_Roast_Decaf", 49, 9.99) ) val q = for { c <- Coffees if c.supID === 101 } yield c.name ~ c.price q.foreach { case (n, p) => println(n + ": " + p) } ScalaQuery

Herbstcampus 2011 – Relationell auch ohne SQL22 Tabellendefinitionen val Suppliers = new Table[(Int, String, String, String, String, String)]("SUPPLIERS") { def id = column[Int ]("SUP_ID", O.PrimaryKey) def name = column[String]("SUP_NAME") def street = column[String]("STREET") def city = column[String]("CITY") def state = column[String]("STATE") def zip = column[String]("ZIP") def * = id ~ name ~ street ~ city ~ state ~ zip def nameConstraint = index("SUP_NAME_IDX", name, true) }

Herbstcampus 2011 – Relationell auch ohne SQL23 Tabellendefinitionen val Coffees = new Table[(String, Int, Double, Int, Int)]("COFFEES") { def name = column[String]("COF_NAME") def supID = column[Int ]("SUP_ID") def price = column[Double]("PRICE") def sales = column[Int ]("SALES") def total = column[Int ]("TOTAL") def * = name ~ supID ~ price ~ sales ~ total def supplier = foreignKey("SUP_FK", supID, Suppliers)(_.id) def pk = primaryKey("COF_NAME_PK", name) }

Herbstcampus 2011 – Relationell auch ohne SQL24 Tabellen Erzeugen val db = Database.forURL("jdbc:h2:mem:test1", driver = "org.h2.Driver") val Suppliers = … val Coffees = … db withSession { (Suppliers.ddl ++ Coffees.ddl).create }

Herbstcampus 2011 – Relationell auch ohne SQL25 import org.scalaquery.ql._ import org.scalaquery.ql.TypeMapper._ import org.scalaquery.ql.extended.H2Driver.Implicit._ import org.scalaquery.ql.extended.{ExtendedTable => Table} def column[C : TypeMapper](n: String, options: ColumnOption[C, ProfileType]*) = … def column[C : TypeMapper](n: String, options: ColumnOption[C, ProfileType]*) = … Query Language Imports basic.BasicDriver extended.AccessDriver extended.DerbyDriver extended.H2Driver extended.HsqldbDriver extended.MySQLDriver extended.PostgresDriver extended.SQLiteDriver extended.SQLServerDriver basic.BasicDriver extended.AccessDriver extended.DerbyDriver extended.H2Driver extended.HsqldbDriver extended.MySQLDriver extended.PostgresDriver extended.SQLiteDriver extended.SQLServerDriver

Herbstcampus 2011 – Relationell auch ohne SQL26 Ein DAO-Pattern class DAO(driver: ExtendedProfile, db: Database) { import driver.Implicit._ val Props = new Table[(String, String)]("properties") { def key = column[String]("key", O.PrimaryKey) def value = column[String]("value") def * = key ~ value } def insert(k: String, v: String) = db withSession Props.insert(k, v) def get(k: String) = db withSession ( for(p <- Props if p.key === k) yield p.value ).firstOption }

Herbstcampus 2011 – Relationell auch ohne SQL27 Inner Joins & Abstraktionen for { c <- Coffees if c.price < 9.0 s <- Suppliers if s.id === c.supID } yield c.name ~ s.name for { c <- coffees if c.price < 9.0 s <- suppliers if s.id == c.supID } yield (c.name, s.name) Scala Collections ScalaQuery for { c <- Coffees if c.price < 9.0 s <- c.supplier } yield c.name ~ s.name val Coffees = new Table … { def supplier = foreignKey("SUP_FK", supID, Suppliers)(_.id) } val Coffees = new Table … { def supplier = foreignKey("SUP_FK", supID, Suppliers)(_.id) } val Coffees = new Table … { def supplier = Suppliers.where(_.id === supID) } val Coffees = new Table … { def supplier = Suppliers.where(_.id === supID) } val Coffees = new Table … { def supplier = Suppliers.where(_.id === supID) def cheaperThan(d: Double) = this.where(_.price < d) } val Coffees = new Table … { def supplier = Suppliers.where(_.id === supID) def cheaperThan(d: Double) = this.where(_.price < d) } Suppliers.where(_.id === supID) for { c <- Coffees.cheaperThan(9.0) s <- c.supplier } yield c.name ~ s.name

Herbstcampus 2011 – Relationell auch ohne SQL28 Datentypen Basistypen Byte, Int, Long String Boolean Date, Time, Timestamp Float, Double Blob, Clob, Array[Byte] Option[T] für alle Basistypen T Datenbank- NULL wird auf Default-Wert gemappt 0 "" false :00: null, null, [] None

Herbstcampus 2011 – Relationell auch ohne SQL29 NULL Three-Valued Logic (3VL) in SQL a b NULL wenn a = NULL oder b = NULL Gilt auch für = a = NULL NULL NULL = a NULL a IS NULL TRUE oder FALSE

Herbstcampus 2011 – Relationell auch ohne SQL30 NULL In ScalaQuery über OptionMapper abgebildet Für Basistypen A, B, C: Column[ A ] Column[ B ] Column[ C ] Column[Option[A]] Column[ B ] Column[Option[C]] Column[ A ] Column[Option[B]] Column[Option[C]] Column[Option[A]] Column[Option[B]] Column[Option[C]]

Herbstcampus 2011 – Relationell auch ohne SQL31 Eigene Datentypen Verwenden object Values extends Enumeration { val a, b, c = Value } implicit val valuesTypeMapper = MappedTypeMapper.base[Values.Value, Int](_.id, Values(_)) val MyTable = new Table[Values.Value]("MYTABLE") { def a = column[Values.Value]("A") def * = a } MyTable.ddl.create MyTable.insertAll(Values.a, Values.c) val q = MyTable.map(t => t.a ~ t.a.asColumnOf[Int]) q.foreach(println) (a,0) (c,2) (a,0) (c,2)

Herbstcampus 2011 – Relationell auch ohne SQL32 Aggregieren und Sortieren val q = for { c <- Coffees s <- c.supplier _ <- Query groupBy s.id _ <- Query orderBy c.name.count } yield s.id ~ s.name.min.get ~ c.name.count Aggregierungsmethoden:.min,.max,.avg,.sum,.count

Herbstcampus 2011 – Relationell auch ohne SQL33 Operatoren Für Columns Allgemein:.in(Query),.notIn(Query),.count,.countDistinct,.isNull,.isNotNull,.asColumnOf,.asColumnOfType Vergleiche: === (.is), =!= (.isNot),, >=,.inSet,.inSetBind,.between,.ifNull Numerisch: +, -, *, /, %,.abs,.ceil,.floor,.sign,.toDegrees,.toRadians Boolean: &&, ||,.unary_! String:.length,.like, ++,.startsWith,.endsWith,.toUpperCase,.toLowerCase,.ltrim,.rtrim,.trim

Herbstcampus 2011 – Relationell auch ohne SQL34 Invokers Alle Datenbankzugriffe erfolgen über Invoker Eine implizite Konvertierung von Query nach Invoker erlaubt das direkte Ausführen von Queries

Herbstcampus 2011 – Relationell auch ohne SQL35 Invoker-Methoden: Strict.to[C]() – erzeugt eine Collection C aller Ergebnisse.list – Shortcut für.to[List]().toMap – erzeugt eine Map[K,V] für einen Query[(K,V)].first,.firstOption,.firstFlatten – geben das erste Ergebnis zurück z.B. myQuery.to[List]() myQuery.to[Array]() z.B. myQuery.to[List]() myQuery.to[Array]()

Herbstcampus 2011 – Relationell auch ohne SQL36 Invoker-Methoden: Lazy / Incremental.elements – erzeugt CloseableIterator, der alle Ergebnisse bei Bedarf liest.elementsTo – nur bis zur angegebenen Maximalanzahl.foreach – führt die angegebene Funktion für jedes Ergebnis aus Optional mit Maximalanzahl.foldLeft – berechnet einen Wert aus allen Ergebnissen.execute – führt das Statement aus for(r <- myQuery)...

Herbstcampus 2011 – Relationell auch ohne SQL37 Debugging val q = for { c <- Coffees if c.supID === 101 } yield c.name ~ c.price q.dump("q: ") println(q.selectStatement) q: Query select: Projection2 0: NamedColumn COF_NAME table: AbstractTable.Alias 0: Table COFFEES 1: NamedColumn PRICE table:... where: Is(NamedColumn SUP_ID,ConstColumn[Int] 101) 0: NamedColumn SUP_ID table:... 1: ConstColumn[Int] 101 q: Query select: Projection2 0: NamedColumn COF_NAME table: AbstractTable.Alias 0: Table COFFEES 1: NamedColumn PRICE table:... where: Is(NamedColumn SUP_ID,ConstColumn[Int] 101) 0: NamedColumn SUP_ID table:... 1: ConstColumn[Int] 101 SELECT "t1"."COF_NAME","t1"."PRICE" FROM "COFFEES" "t1 WHERE ("t1"."SUP_ID"=101) SELECT "t1"."COF_NAME","t1"."PRICE" FROM "COFFEES" "t1 WHERE ("t1"."SUP_ID"=101)

Herbstcampus 2011 – Relationell auch ohne SQL38 Explizite Inner Joins for ( Join(c, s) <- Coffees innerJoin Suppliers on (_.supID === _.id) ) yield c.name ~ s.name (Colombian,Acme, Inc.) (Espresso,The High Ground) namesupID Colombian101 Espresso150 Colombian_Decaf42 idname 101Acme, Inc. 49Superior Coffee 150The High Ground Suppliers Coffees

Herbstcampus 2011 – Relationell auch ohne SQL39 Left Outer Joins for ( Join(c, s) <- Coffees leftJoin Suppliers on (_.supID === _.id) ) yield c.name ~ s.name (Colombian,Acme, Inc.) (Espresso,The High Ground) (Colombian_Decaf,) for ( Join(c, s) <- Coffees leftJoin Suppliers on (_.supID === _.id) ) yield c.name.? ~ s.name.? (Some(Colombian),Some(Acme, Inc.)) (Some(Espresso),Some(The High Ground)) (Some(Colombian_Decaf),None) namesupID Colombian101 Espresso150 Colombian_Decaf42 idname 101Acme, Inc. 49Superior Coffee 150The High Ground Suppliers Coffees

Herbstcampus 2011 – Relationell auch ohne SQL40 Right Outer Joins for ( Join(c, s) <- Coffees rightJoin Suppliers on (_.supID === _.id) ) yield c.name.? ~ s.name.? (Some(Colombian),Some(Acme, Inc.)) (None,Some(Superior Coffee)) (Some(Espresso),Some(The High Ground)) namesupID Colombian101 Espresso150 Colombian_Decaf42 idname 101Acme, Inc. 49Superior Coffee 150The High Ground Suppliers Coffees

Herbstcampus 2011 – Relationell auch ohne SQL41 Full Outer Joins for ( Join(c, s) <- Coffees outerJoin Suppliers on (_.supID === _.id) ) yield c.name.? ~ s.name.? (Some(Colombian),Some(Acme, Inc.)) (None,Some(Superior Coffee)) (Some(Espresso),Some(The High Ground)) (Some(Colombian_Decaf),None) namesupID Colombian101 Espresso150 Colombian_Decaf42 idname 101Acme, Inc. 49Superior Coffee 150The High Ground Suppliers Coffees

Herbstcampus 2011 – Relationell auch ohne SQL42 Case for { c <- Coffees } yield (Case when c.price < 8.0 then "cheap" when c.price < 9.0 then "medium" otherwise "expensive") ~ c.name If-then-else für Queries Rückgabetyp wird automatisch zu Option, wenn otherwise fehlt

Herbstcampus 2011 – Relationell auch ohne SQL43 Sub-Queries for { c <- Coffees s <- c.supplier _ <- Query groupBy s.id orderBy s.id } yield s.name.min.get ~ c.price.min.get for { c <- Coffees s <- c.supplier _ <- Query if c.price === lowestPriceForSupplier _ <- Query orderBy s.id } yield s.name ~ c.price val lowestPriceForSupplier = (for { c2 <- Coffees s2 <- c2.supplier if s2.id === s.id } yield c2.price.min).asColumn Auch in yield verwendbar Direkt (ohne.asColumn ) mit.in und.notIn.exists,.count

Herbstcampus 2011 – Relationell auch ohne SQL44 Unions val l1 = coffees.filter(_.supID == 101) val l2 = coffees.filter(_.supID == 150) val l3 = l1 ++ l2 val q1 = Coffees.filter(_.supID === 101) val q2 = Coffees.filter(_.supID === 150) val q3 = q1 union q2 Scala Collections ScalaQuery All

Herbstcampus 2011 – Relationell auch ohne SQL45 Paginierung val l = for { c <- coffees if … } yield … val l2 = l.drop(20).take(10) val q = for { c <- Coffees if … _ <- Query orderBy c.name } yield … val q2 = q.drop(20).take(10) Scala Collections ScalaQuery

Herbstcampus 2011 – Relationell auch ohne SQL46.bind Bind-Variablen def coffeesForSupplier(supID: Int) = for { c <- Coffees if c.supID === supID } yield c.name coffeesForSupplier(42).list Query select: NamedColumn COF_NAME table: AbstractTable.Alias 0: Table COFFEES where: Is(NamedColumn SUP_ID,ConstColumn[Int] 42) 0: NamedColumn SUP_ID table:... 1: ConstColumn[Int] 42 Query select: NamedColumn COF_NAME table: AbstractTable.Alias 0: Table COFFEES where: Is(NamedColumn SUP_ID,ConstColumn[Int] 42) 0: NamedColumn SUP_ID table:... 1: ConstColumn[Int] 42 SELECT "t1"."COF_NAME" FROM "COFFEES" "t1" WHERE ("t1"."SUP_ID"=42) SELECT "t1"."COF_NAME" FROM "COFFEES" "t1" WHERE ("t1"."SUP_ID"=42) =?) Bind

Herbstcampus 2011 – Relationell auch ohne SQL47 Query-Templates val coffeesForSupplier = for { supID <- Parameters[Int] c <- Coffees if c.supID === supID } yield c.name coffeesForSupplier(42).list Query select: NamedColumn COF_NAME table: AbstractTable.Alias 0: Table COFFEES where: Is(NamedColumn SUP_ID,ParameterColumn[Int]) 0: NamedColumn SUP_ID table:... 1: ParameterColumn[Int] Query select: NamedColumn COF_NAME table: AbstractTable.Alias 0: Table COFFEES where: Is(NamedColumn SUP_ID,ParameterColumn[Int]) 0: NamedColumn SUP_ID table:... 1: ParameterColumn[Int] SELECT "t1"."COF_NAME" FROM "COFFEES" "t1" WHERE ("t1"."SUP_ID"=?) SELECT "t1"."COF_NAME" FROM "COFFEES" "t1" WHERE ("t1"."SUP_ID"=?)

Herbstcampus 2011 – Relationell auch ohne SQL48 Coffee <> (Coffee, Coffee.unapply _) Coffee Mapped Entities val Coffees = new Table[ ]("COFFEES") { def name = column[String]("COF_NAME", O.PrimaryKey) def supID = column[Int]("SUP_ID") def price = column[Double]("PRICE") def * = name ~ supID ~ price } Coffees.insertAll( ("Colombian", 101, 7.99), ("French_Roast", 49, 8.99) ) val q = for(c <- Coffees if c.supID === 101) yield c q.foreach(println) case class Coffee(name: String, supID: Int, price: Double) (String, Int, Double) (Colombian,101,7.99) Coffee

Herbstcampus 2011 – Relationell auch ohne SQL49 Insert, Delete, Update class Coffees(n: String) extends Table[(String, Int, Double)](n) { def name = column[String]("COF_NAME") def supID = column[Int]("SUP_ID") def price = column[Double]("PRICE") def * = name ~ supID ~ price } val Coffees1 = new Coffees("COFFEES_1") val Coffees2 = new Coffees("COFFEES_2") (Coffees1.ddl ++ Coffees2.ddl).create Coffees1.insertAll( ("Colombian", 101, 7.99), ("French_Roast", 49, 8.99), ("Espresso", 150, 9.99) ) println(Coffees1.insertStatement) INSERT INTO "COFFEES1" ("COF_NAME","SUP_ID","PRICE") VALUES (?,?,?)

Herbstcampus 2011 – Relationell auch ohne SQL50 Insert, Delete, Update val q = Coffees1.where(_.supID === 101) Coffees2.insert(q) println(Coffees2.insertStatementFor(q)) q.delete println(q.deleteStatement) INSERT INTO "COFFEES2" ("COF_NAME","SUP_ID","PRICE") SELECT "t1"."COF_NAME","t1"."SUP_ID","t1"."PRICE" FROM "COFFEES1" "t1" WHERE ("t1"."SUP_ID"=101) INSERT INTO "COFFEES2" ("COF_NAME","SUP_ID","PRICE") SELECT "t1"."COF_NAME","t1"."SUP_ID","t1"."PRICE" FROM "COFFEES1" "t1" WHERE ("t1"."SUP_ID"=101) DELETE FROM "COFFEES1" WHERE ("COFFEES1"."SUP_ID"=101)

Herbstcampus 2011 – Relationell auch ohne SQL51 Insert, Delete, Update val q2 = q.map(_.supID) q2.update(49) println(q2.updateStatement) UPDATE "COFFEES1" SET "SUP_ID"=? WHERE ("COFFEES1"."SUP_ID"=101)

Herbstcampus 2011 – Relationell auch ohne SQL52 Static Queries import org.scalaquery.simple._ import org.scalaquery.simple.StaticQuery._ def allCoffees = queryNA[String]( "select cof_name from coffees").list def supplierNameForCoffee(name: String) = query[String, String](""" select s.sup_name from suppliers s, coffees c where c.cof_name = ? and c.sup_id = s.sup_id """).firstOption(name) def coffeesInPriceRange(min: Double, max: Double) = query[(Double, Double), (String, Int, Double)](""" select cof_name, sup_id, price from coffees where price >= ? and price <= ? """).list(min, max)

Herbstcampus 2011 – Relationell auch ohne SQL53 Static Queries import org.scalaquery.simple._ import org.scalaquery.simple.StaticQuery._ case class Coffee( name: String, supID: Int, price: Double) implicit val getCoffeeResult = GetResult(r => Coffee(r<<, r<<, r<<)) def coffeesInPriceRange(min: Double, max: Double) = query[(Double, Double), (String, Int, Double)](""" select cof_name, sup_id, price from coffees where price >= ? and price <= ? """).list(min, max) [P : SetParameter, R : GetResult] Coffee

Herbstcampus 2011 – Relationell auch ohne SQL54 Weitere Features Mutating Queries JDBC-Metadaten Iteratees Sequences Dynamic Queries MutatingInvoker.mutate org.scalaquery.meta org.scalaquery.iter org.scalaquery.simple

Herbstcampus 2011 – Relationell auch ohne SQL55 Getting Started tree/master/src/test/scala/org/scalaquery/testhttps://github.com/szeiger/scala-query/ tree/master/src/test/scala/org/scalaquery/test

Herbstcampus 2011 – Relationell auch ohne SQL56 Ausblick Slick – A common framework for connecting with databases and distributed collections by Christopher Vogt

Herbstcampus 2011 – Relationell auch ohne SQL57 Ausblick ScalaQuery 0.9 ScalaQuery 0.10 SIQ Virtualized Scala for { c <- Coffees if c.price === 8.99 || c.price === 9.99 s <- c.supplier orderBy s.id } yield s.id ~ s.name ~ c.name ~ c.price ~ … for { c <- Coffees if c.price === 8.99 || c.price === 9.99 s <- c.supplier orderBy s.id } yield ((s.id, s.name), c) for { s <- Suppliers val cs = Coffees.filter(c => c.supID === s.id && (c.price === 8.99 || c.price === 9.99)) } yield ((s.id, s.name), cs) for { s <- Suppliers val cs = Coffees.filter(c => c.supID == s.id && (c.price == 8.99 || c.price == 9.99)) } yield ((s.id, s.name), cs)

Vielen Dank! Stefan Zeiger Commerzbank AG