Java Database Connectivity (JDBC) est une composante essentielle de l’écosystème Java, conçue pour offrir une interaction transparente avec une variété de Systèmes de Gestion de Bases de Données Relationnelles (SGBDR). Cette API robuste est conçue pour unifier l’accès aux bases de données à travers une interface commune, exploitant la puissance du langage SQL tout en s’alignant avec les types de données Java. Avec JDBC, les développeurs peuvent établir des connexions fiables avec des bases de données, exécuter des requêtes SQL avec précision et parcourir les résultats de manière intuitive. Bien que JDBC soit suffisamment flexible pour s’adapter aux caractéristiques propres à chaque SGBDR, cette adaptabilité peut parfois se faire au détriment de la portabilité du code, nécessitant des ajustements spécifiques pour chaque système.
Système de gestion de base de données et Dataset
Dans le cadre de ce cours nous utiliserons les systèmes de gestion de base données PostgreSQL
Pour les travaux pratiques n’importe quel SGBDR pourra être utilisé par exemple h2.
H2 est un SGDB écrit en Java qui peut s’executer comme un serveur indépendant ou depuis un programme Java. Il peut aussi être utilisé purement en mémoire ou avec persistence sur disque.
Pour l’utiliser il suffit de télécharger le fichier .jar de h2 par exemple la version h2-2.2.224.jar depuis un entrepôt maven.
Le serveur peut être exécuté simplement avec la commande suivante pour autoriser les connexions tcp et la création automatique d’une base de données lors du premier accès.
Pour accéder à la base de données, il suffit de se connecter à l’URL jdbc:h2:tcp://localhost/Gtfs_RMTT;MODE=PostgreSQL;DATABASE_TO_LOWER=TRUE avec le nom d’utilisateur sa et un mot de passe vide.
java-cp h2*.jar org.h2.tools.RunScript \-url"jdbc:h2:tcp://localhost/Gtfs_RMTT;MODE=PostgreSQL;DATABASE_TO_LOWER=TRUE"\-user sa \-script data.sql
Pour illustrer ce cours, nous allons utiliser une base de données qui représente des données de transport (des bus) au format GTFS (General Transit Feed Specification).
Format de données standardisé créé par Google
Permet le partage des données de transport public
Utilisé par Google Maps et autres applications
Nous utiliserons une extensions de postgresql pour les Systemes d’Information Géographique (SIG) PostGIS qui permet de stocker des données géographiques et de faire des requêtes spatiales.
Conversion des données pour l’intégré dans le système d’information géographique (SIG) PostGIS. PostGIS est une extension du SGBD PostgreSQL qui permet la manipulation d’informations géographiques (spatiales) sous forme de géométries (points, lignes, polygones), conformément aux standards établis par l’Open Geospatial Consortium. En d’autres termes, PostGIS permet le traitement d’objets spatiaux dans PostgreSQL, autorisant le stockage des objets graphiques en base de données pour les systèmes d’informations géographiques (SIG).
%%shelldocker run --rm --quiet --volume shared-data:/shared-data \--entrypoint sh \ ghcr.io/public-transport/gtfs-via-postgres:4-eu -o pipefail -c '/app/cli.js $@--require-dependencies --trips-without-shape-id --lower-case-lang-codes --/shared-data/gtfs/*.txt> /shared-data/gtfs/data.sql'
WARNING: The requested image's platform (linux/amd64) does not match the detected host platform (linux/arm64/v8) and no specific platform was requested
is_valid_lang_code
is_timezone
calendar
processed 382 rows
calendar_dates
processed 6894 rows
service_days
agency
processed 1 rows
routes
processed 97 rows
trips
processed 11787 rows
frequencies
feed_info
processed 1 rows
shapes
processed 195972 rows
stops
processed 1987 rows
stop_times
processed 387239 rows
%%shelldocker network ls|grep notebooknet >/dev/null|| docker network create notebooknetdocker container ls|grep db >/dev/null&& docker stop dbecho "--> Launch DBMS"docker run --quiet --rm --name db -d \--env POSTGRES_USER=${DB_USERNAME:-dba} \--env POSTGRES_PASSWORD=${DB_PASSWORD:-secretsecret} \--env POSTGRES_DB=${DB_NAME:-notebook-db} \--volume dbhost-data:/var/lib/postgresql \--volume shared-data:/shared-data \--network=notebooknet \-p 5432:5432 \ postgis/postgis:17-masterdocker container exec \--env PGPASSWORD=secretsecret \ db \ sh -c 'until pg_isready -U dba -d notebook-db;do echo "$(date) - waiting for database to start"; sleep 1; done'
db
--> Launch DBMS
WARNING: The requested image's platform (linux/amd64) does not match the detected host platform (linux/arm64/v8) and no specific platform was requested
e1aae55d9ea96f2a45d27f6046c70c99c1257698cf78e410ce57d295c9518bbc
/var/run/postgresql:5432 - no response
Tue 10 Dec 2024 07:00:07 AM UTC - waiting for database to start
/var/run/postgresql:5432 - no response
Tue 10 Dec 2024 07:00:08 AM UTC - waiting for database to start
/var/run/postgresql:5432 - no response
Tue 10 Dec 2024 07:00:10 AM UTC - waiting for database to start
/var/run/postgresql:5432 - accepting connections
%%shell# Requête SQL pour trouver les arrêts "Université" avec leurs coordonnées GPSread -r -d '' QUERY << EOMSELECT stop_id,-- Identifiant unique de l'arrêt stop_name,-- Nom de l'arrêtST_X(ST_Transform(stop_loc::geometry,4326)) AS stop_latitude,-- Conversion des coordonnées en latitude(EPSG:4326= standard GPS)ST_Y(ST_Transform(stop_loc::geometry,4326)) AS stop_longitude -- Conversion des coordonnées en longitude(EPSG:4326= standard GPS)FROM stopsWHERE stop_id ILIKE 'GACAM%'LIMIT 3;EOMdocker run --rm \--env PGPASSWORD=secretsecret \--network=notebooknet \ postgres \ psql -h db -U dba gtfs -c $QUERY;
%%shell# Requête SQL pour trouver les arrêts "Université" avec leurs coordonnées GPSread -r -d '' QUERY << EOMSELECT route_id,-- Identifiant unique de la ligne route_short_name,-- Nom court de la ligne route_long_name -- Nom long de la ligneFROM routesWHERE route_long_name ILIKE '%Tln%'LIMIT 3;EOMdocker run --rm \--env PGPASSWORD=secretsecret \--network=notebooknet \ postgres \ psql -h db -U dba gtfs -c $QUERY;
route_id | route_short_name | route_long_name
----------+------------------+------------------------------
U | U | Tech.Mer/Pôle d'Act. Tln Est
0102 | 102 | Gare Rte Tln-Aéroport Hyères
0103 | 103 | Gare Rout. Tln-Moulin 1er
(3 rows)
%%shell# Requête pour trouver les 3 arrêts les plus proches de l'Université de Toulonread -r -d '' QUERY << EOM-- Recherche des arrêts par proximité géographiqueSELECT stop_id,-- Identifiant de l'arrêt stop_name,-- Nom de l'arrêt-- Calcul de la distance entre chaque arrêt et l'UniversitéROUND(ST_Distance( stop_loc::geography,--Position de l'arrêtST_SetSRID(-- Définit le système de coordonnéesST_MakePoint(-- Crée un point géographique6.016499,-- Longitude Université43.135440-- Latitude Université),4326-- En WGS84))) AS distance,-- Conversion des coordonnées en WGS84(format GPS standard)ST_X(-- Extrait la latitudeST_Transform(-- Convertit les coordonnées stop_loc::geometry,--Position de l'arrêt4326-- Code EPSG pour WGS84(standard GPS))) AS LATITUDE,ST_Y(ST_Transform(stop_loc::geometry,4326)) AS LONGITUDE -- Idem pour longitudeFROM stopsORDER BY distance ASC -- Tri du plus proche au plus éloignéLIMIT 5-- Retourne les 5 plus prochesEOMdocker run --rm \--env PGPASSWORD=secretsecret \--network=notebooknet \ postgres \ psql -h db -U dba gtfs -c $QUERY;
%%shell# Store a query in the QUERY shell variableread -r -d '' QUERY << EOMSELECT s.stop_id AS arret,-- Identifiant de l'arrêt s.stop_name AS nom_arret,-- Nom de l'arrêt r.route_short_name AS ligne,-- Numéro de ligne r.route_long_name AS nom_ligne,-- Nom de la ligne t.trip_headsign AS destination,--Destination du voyage st.departure_time AS depart -- Heure de départ FROM stop_times st JOIN stops s ON st.stop_id= s.stop_id-- Liaison avec les arrêts JOIN trips t ON st.trip_id= t.trip_id-- Liaison avec les voyages JOIN routes r ON t.route_id= r.route_id-- Liaison avec les lignesWHERE st.stop_id='VACAMN' AND r.route_short_name='U' AND st.departure_time>(NOW()::time)-- Départ après l'heure actuelleORDER BY st.departure_timeLIMIT 3;EOMdocker run --rm \--env PGPASSWORD=secretsecret \--network=notebooknet \ postgres \ psql -h db -U dba gtfs -c $QUERY;
arret | nom_arret | ligne | nom_ligne | destination | depart
--------+-----------+-------+------------------------------+----------------------+----------
VACAMN | Campus | U | Tech.Mer/Pôle d'Act. Tln Est | Technopole de la Mer | 07:05:00
VACAMN | Campus | U | Tech.Mer/Pôle d'Act. Tln Est | Technopole de la Mer | 07:10:00
VACAMN | Campus | U | Tech.Mer/Pôle d'Act. Tln Est | Technopole de la Mer | 07:10:00
(3 rows)
%%shell# Store a query in the QUERY shell variableread -r -d '' QUERY << EOMSELECT r.route_short_name AS ligne,-- Numéro de ligne st.arrival_time AS arrivee,-- Heure d'arrivée st.departure_time AS depart -- Heure de départ FROM stop_times stJOIN stops s ON st.stop_id= s.stop_id-- Liaison avec les arrêtsJOIN trips t ON st.trip_id= t.trip_id-- Liaison avec les voyagesJOIN routes r ON t.route_id= r.route_id-- Liaison avec les lignesWHERE st.stop_id='GAUNSS'LIMIT 3EOMdocker run --rm \--env PGPASSWORD=secretsecret \--network=notebooknet \ postgres \ psql -h db -U dba gtfs -c $QUERY;
Définissons tout d’abord simplement une classe Java pour représenter l’une des entités de l’exemple, un arrêt de bus. Nous utiliserons un record Java (non mutable).
/*** Représente un arrêt de transport public*/publicrecordStop(String id,// Identifiant uniqueString name,// Nom de l'arrêtDouble latitude,// LatitudeDouble longitude,// LongitudeString locationType // arrêt, station){public Stop {if(id ==null|| name ==null|| latitude ==null|| longitude ==null){thrownewIllegalArgumentException("Required fields cannot be null");}}}
/*** Représente une ligne de transport*/publicrecordRoute(String id,// Identifiant uniqueString shortName,// Numéro/Code courtString longName // Nom complet){public Route {if(id ==null|| shortName ==null){thrownewIllegalArgumentException("Required fields cannot be null");}}}
/*** Représente un voyage sur une ligne*/publicrecordTrip(String id,// Identifiant unique Route route,// ID de la ligneString tripHeadsign,// Destination affichéeString tripShortName // Nom court du voyage){public Trip {if(id ==null|| route ==null|| tripHeadsign ==null){thrownewIllegalArgumentException("Required fields cannot be null");}}}
importjava.time.LocalTime;/*** Représente un horaire de passage à un arrêt*/publicrecordStopTime( Trip trip,// ID du voyage Stop stop,// ID de l'arrêt LocalTime arrivalTime,// Heure d'arrivée LocalTime departureTime,// Heure de départString stopHeadsign // Destination affichée à l'arrêt){public StopTime {// Validation des champs obligatoiresif(trip ==null|| stop ==null|| arrivalTime ==null|| departureTime ==null){thrownewIllegalArgumentException("trip, stop, times are required");}// Validation de la cohérence des heuresif(departureTime.isBefore(arrivalTime)){thrownewIllegalArgumentException("Departure time cannot be before arrival time");}}}
importjava.sql.Connection;importjava.sql.DriverManager;System.out.println("%s connecting on %s.".formatted(DB_USERNAME,DB_URL));finalString query =""" SELECT stop_id, stop_name, ST_X(ST_Transform(stop_loc::geometry, 4326)) AS LATITUDE, ST_Y(ST_Transform(stop_loc::geometry, 4326)) AS LONGITUDE, location_type FROM stops ORDER BY ST_Distance(stop_loc::geometry, ST_SetSRID(ST_MakePoint(5.940207, 43.121145), 4326)) ASC LIMIT 3""";try(Connection connection =DriverManager.getConnection(DB_URL,DB_USERNAME,DB_PASSWORD);Statement stmt = connection.createStatement();){ResultSet rs = stmt.executeQuery(query);long id=0;while(rs.next())System.out.println(newStop(rs.getString("stop_id"), rs.getString("stop_name"), rs.getDouble("latitude"), rs.getDouble("longitude"), rs.getString("location_type")));}catch(SQLException e){ e.printStackTrace();}
dba connecting on jdbc:postgresql://docker/gtfs.
Stop[id=TOCMAS, name=Champ de Mars, latitude=5.940372, longitude=43.121363, locationType=stop]
Stop[id=TOCMAN, name=Champ de Mars, latitude=5.940497, longitude=43.12208, locationType=stop]
Stop[id=TOPIXO, name=Pie X, latitude=5.941181, longitude=43.120398, locationType=stop]
Le Driver JDBC
Le driver JDBC est un composant essentiel qui permet aux applications Java de communiquer avec une base de données. Il sert d’intermédiaire entre l’application et le SGBDR, convertissant les appels Java en commandes compréhensibles par la base de données.
Pour utiliser un driver JDBC dans votre projet, vous devez l’ajouter au classpath de votre application. Les fournisseurs de bases de données mettent généralement à disposition les drivers nécessaires sur leurs sites web officiels. Avec Maven, l’intégration du driver est simplifiée grâce à la gestion des dépendances. Vous pouvez spécifier le driver JDBC comme une dépendance dans votre fichier pom.xml, et Maven se chargera de le télécharger et de l’ajouter au classpath lors de la construction de votre projet.
Ouverture d’une connexion
Pour établir une connexion avec une base de données, il est nécessaire d’initialiser la classe qui implémente le Driver JDBC. Historiquement, cela se faisait en invoquant Class.forName("nom.de.la.classe.Driver"). Cependant, avec les versions modernes de JDBC, cette étape est devenue facultative. à partir de JDBC 4.0+, la correspondance entre l’URL de connexion et le driver approprié est gérée automatiquement.
importjava.sql.Connection;importjava.sql.DriverManager;importjava.sql.SQLException;finalString DB_URL="jdbc:postgresql://docker/gtfs";// Nom d'utilisateur et mot de passe de la base de données récupérés depuis les variables d'environnementfinalString DB_USERNAME=Optional.of(System.getenv("DB_USERNAME")).orElseThrow(()->newIllegalArgumentException("DB_USERNAME is required"));finalString DB_PASSWORD=Optional.of(System.getenv("DB_PASSWORD")).orElseThrow(()->newIllegalArgumentException("DB_PASSWORD is required"));try(Connection connection =DriverManager.getConnection(DB_URL, DB_USERNAME, DB_PASSWORD)){// La connexion est établie, vous pouvez interagir avec la base de données ici}catch(SQLException e){// Gestion des exceptions liées à la base de données e.printStackTrace();}
Exécution de requêtes SQL
Pour exécuter des requêtes SQL en Java, on utilise des instances de la classe Statement. Il existe trois types de Statement :
Statement : Utilisé pour des requêtes SQL simples. Il permet d’exécuter des requêtes dynamiques, mais n’est pas optimisé pour des exécutions répétées.
PreparedStatement : Il s’agit d’une version précompilée du Statement qui est optimisée pour exécuter des requêtes SQL multiples fois avec des paramètres différents. Cela améliore les performances en réduisant le temps de compilation SQL sur la base de données.
CallableStatement : Utilisé pour exécuter des procédures stockées sur la base de données, ce qui permet d’appeler des fonctions complexes stockées directement dans la base de données.
Chaque Statement est créé à partir d’une instance de Connection.
Pour exécuter des requetes SQL on utilise des instances de la classes Statement.
Les requêtes simples
L’approche à adopter pour exécuter des requêtes SQL varie en fonction de leur objectif. Pour une interrogation de données (SELECT), utilisez la méthode executeQuery(). Cette méthode vous permettra de parcourir les résultats à l’aide d’un objet ResultSet. En revanche, pour une modification de données (UPDATE, INSERT, DELETE), privilégiez la méthode executeUpdate(). Celle-ci est conçue pour les opérations de mise à jour ou les commandes de gestion de base de données et renvoie le nombre de lignes affectées par la requête. Enfin, si vous avez affaire à des requêtes de type indéterminé ou des procédures stockées pouvant produire plusieurs jeux de résultats, la méthode execute() est plus flexible et adaptative. Elle vous permettra de gérer les résultats de manière plus versatile.
Ce paragraphe fournit des directives claires sur la méthode à utiliser en fonction du type de requête SQL, en mettant l’accent sur la flexibilité et l’efficacité du processus.
La méthode à appeler est différente suivant la nature de la requêtes SQL que l’on veut exécuter :
Consultation (SELECT)
executeQuery() : Utilisé pour les requêtes de type SELECT
Parcours des résultats avec un ResultSet
Mise à jour (UPDATE, INSERT, DELETE)
executeUpdate() : Utilisé pour les requêtes de mise à jour ou de gestion de la base de données
Renvoie le nombre de lignes modifiées
Type Inconnu ou Résultats Multiples
execute() : Utilisé lorsque le type de requête est inconnu ou pour des procédures stockées pouvant renvoyer plusieurs résultats
Gestion plus flexible des résultats
Parcours des résultats
La manipulation des résultats d’une requête SQL de type SELECT se fait via l’interface ResultSet. La méthode executeQuery() est utilisée pour exécuter une requête SELECT et retourne un objet ResultSet. Cet objet ResultSet contient les données retournées par la requête. Pour accéder aux valeurs des attributs de chaque ligne du résultat, ResultSet offre une série de méthodes getXXX(), où XXX représente le type Java correspondant à la valeur à récupérer. Par exemple, getInt(int numéroDeColonne) ou getString(String nomDeColonne) sont utilisées pour récupérer des valeurs entières ou des chaînes de caractères, respectivement.
// Create a list to store Stop objectsList<Stop> stops =newArrayList<>();finalString query =""" SELECT stop_id, stop_name, ST_X(ST_Transform(stop_loc::geometry, 4326)) AS LATITUDE, ST_Y(ST_Transform(stop_loc::geometry, 4326)) AS LONGITUDE, location_type FROM stops ORDER BY ST_Distance(stop_loc::geometry, ST_SetSRID(ST_MakePoint(5.940207, 43.121145), 4326)) ASC LIMIT 3""";try(Connection connection =DriverManager.getConnection(DB_URL,DB_USERNAME,DB_PASSWORD);Statement stmt = connection.createStatement();){ResultSet rs = stmt.executeQuery(query);long id=0;while(rs.next()) stops.add(newStop(rs.getString("stop_id"), rs.getString("stop_name"), rs.getDouble("latitude"), rs.getDouble("longitude"), rs.getString("location_type")));}catch(SQLException e){ e.printStackTrace();}stops.forEach(System.out::println);
Stop[id=TOCMAS, name=Champ de Mars, latitude=5.940372, longitude=43.121363, locationType=stop]
Stop[id=TOCMAN, name=Champ de Mars, latitude=5.940497, longitude=43.12208, locationType=stop]
Stop[id=TOPIXO, name=Pie X, latitude=5.941181, longitude=43.120398, locationType=stop]
try(Connection connection =DriverManager.getConnection(DB_URL, DB_USERNAME, DB_PASSWORD);Statement statement = connection.createStatement()){// SQL query to insert data into the 'agency' tableString sql =""" INSERT INTO agency(agency_id, agency_name, agency_url, agency_timezone, agency_lang, agency_phone, agency_fare_url, agency_email) VALUES(3, 'Neverland', 'http://nowhere.fr', 'Europe/Paris', 'fr', '555-555-555', 'http://nowhere.fr/fare', 'neverland@nowhere.org')""";// Execute the query and get the number of changesint numberOfChanges = statement.executeUpdate(sql);// Print the resultSystem.out.println("Nombre de changements: "+ numberOfChanges);}catch(SQLException e){ e.printStackTrace();}
Nombre de changements: 1
Les exceptions
Erreur dans le code SQL (SQLException) : Lorsque JDBC rencontre une erreur lors d’une interaction avec une source de données, il génère une instance de SQLException. Cette exception contient des informations telles que la description de l’erreur, le code SQLState (standardisé par ISO/ANSI et Open Group), et un code d’erreur spécifique.
Avertissement lors de l’exécution (SQLWarning) : Les avertissements SQL (SQLWarning) sont des exceptions qui ne provoquent pas l’interruption du programme, mais signalent des problèmes potentiels. Par exemple, un avertissement peut indiquer une conversion de données tronquée (sous-classe de SQLWarning) lors de l’exécution d’une requête. Ces avertissements sont utiles pour la détection précoce de problèmes sans arrêter le flux d’exécution.
String wrongQuery =" SELECT * FROM Employee";try(Connection connection =DriverManager.getConnection(DB_URL,DB_USERNAME,DB_PASSWORD);Statement statement = connection.createStatement();ResultSet resultSet = statement.executeQuery(wrongQuery)){}catch(SQLException e){//Erreur lors de la requête log.severe(e.getMessage());}
CompilationException:
try (Connection connection = DriverManager.getConnection(DB_URL,DB_USERNAME,DB_PASSWORD);
Statement statement = connection.createStatement();
ResultSet resultSet = statement.executeQuery(wrongQuery)) {
} catch (SQLException e) {
//Erreur lors de la requête
log.severe(e.getMessage());
}
using incubating module(s): jdk.incubator.vector
log.severe(e.getMessage());
cannot find symbol
symbol: variable log
ResultSet resultSet = statement.executeQuery(wrongQuery)) {
auto-closeable resource resultSet is never referenced in body of corresponding try statement
Types Java/JDBC et SQL
Types de Données Java/JDBC
Lorsque nous travaillons avec des bases de données relationnelles, il est essentiel de comprendre les différences de types entre les différents SGBD (Systèmes de Gestion de Bases de Données). Heureusement, JDBC (Java Database Connectivity) simplifie cette complexité en définissant ses propres types SQL sous forme de constantes dans la classe java.sql.Types.
Conversion entre SQL et Java :
Lorsque nous récupérons des données depuis la base de données (par exemple, lors de l’exécution d’une requête SELECT), le pilote JDBC effectue automatiquement la conversion des types SQL vers les types Java appropriés.
Lorsque nous passons des paramètres à une requête (par exemple, lors de l’exécution d’une requête INSERT ou UPDATE), le pilote effectue la conversion inverse, de Java vers SQL.
Utilisation explicite des méthodes getXXX() et setXXX() :
Pour récupérer des valeurs depuis le résultat d’une requête, nous utilisons explicitement des méthodes telles que getString(), getInt(), getDouble(), etc.
De même, pour définir des valeurs de paramètres, nous utilisons des méthodes comme setString(), setInt(), setDouble(), etc.
Choix multiples pour certains types :
Pour certains types SQL, plusieurs méthodes Java peuvent être utilisées pour récupérer les valeurs :
CHAR et VARCHAR : Utilisez getString().
LONGVARCHAR : Utilisez getAsciiStream() ou getCharacterStream().
BINARY et VARBINARY : Utilisez getBytes().
REAL : Utilisez getFloat(), et pour DOUBLE et FLOAT, utilisez getDouble().
DECIMAL et NUMERIC : Utilisez getBigDecimal().
Types de date et d’heure : DATE (utilisez getDate()), TIME (utilisez getTime()), et TIMESTAMP (utilisez getTimestamp()).
Transaction
Gestion des Transactions avec JDBC
Par défaut, une connexion à la base de données est ouverte en mode « auto-commit ». Cela signifie qu’après chaque requête SQL qui modifie les données (par exemple, une requête INSERT, UPDATE ou DELETE), un commit est automatiquement effectué. Le commit valide les modifications dans la base de données de manière permanente.
Cependant, pour un contrôle plus fin sur les transactions, nous pouvons désactiver le mode « auto-commit » et gérer manuellement les commits et les rollbacks. Voici comment faire :
Désactiver l’auto-commit :
Utilisez la méthode setAutoCommit(false) sur l’objet Connection pour désactiver l’auto-commit.
Cela signifie que les modifications ne seront pas validées automatiquement après chaque requête.
Valider une transaction (commit) :
Lorsque vous avez terminé une série de requêtes et que vous souhaitez valider les modifications dans la base de données, appelez commit() sur l’objet Connection.
Cela confirme les changements et les rend permanents.
Annuler une transaction (rollback) :
Si une erreur survient ou si vous souhaitez annuler une série de modifications, appelez rollback() sur l’objet Connection.
Cela annule toutes les modifications depuis le dernier commit.
Précompilation des requêtes
Les requêtes SQL construites à partir de chaînes de caractères (par exemple, en concaténant des valeurs), sont compilées à chaque appel. Cette approche peut entraîner une perte de performances, car la compilation est coûteuse en termes de temps et de ressources.
Cependant, JDBC propose une solution plus efficace : les requêtes paramétrées (ou PreparedStatement). Avec les requêtes paramétrées, la requête SQL est compilée une seule fois, lors de sa création. Elle est ensuite réutilisée pour chaque exécution. Cela évite la compilation répétée et améliore considérablement les performances.
La requête est définie avec des espaces réservés pour les paramètres (par exemple, "SELECT * FROM users WHERE username = ?"). Au moment de l’exécution, les espaces réservés sont remplacé par les valeurs réelles.
// Déclaration de la requête SQLString sql ="""SELECT stop_id, stop_name, ST_X(ST_Transform(stop_loc::geometry, 4326)) AS LATITUDE, ST_Y(ST_Transform(stop_loc::geometry, 4326)) AS LONGITUDE, location_type FROM stops WHERE stop_id = ? ORDER BY ST_Distance(stop_loc::geometry, ST_SetSRID(ST_MakePoint(5.940207, 43.121145), 4326)) ASCLIMIT 3""";try(Connection connection =DriverManager.getConnection(DB_URL, DB_USERNAME, DB_PASSWORD);PreparedStatement pstmt = connection.prepareStatement(sql)){// Tableau contenant les identifiants à rechercherString[] ids ={"TOCMAS","TOCMAN","TOPIXO"};for(String id : ids){// Remplacer le paramètre de la requête par l'identifiant actuel pstmt.setString(1, id);// Exécution de la requêtetry(ResultSet rs = pstmt.executeQuery()){while(rs.next()){// Création et affichage d'un objet Stop avec les données récupéréesSystem.out.println(newStop(rs.getString("stop_id"), rs.getString("stop_name"), rs.getDouble("latitude"), rs.getDouble("longitude"), rs.getString("location_type")));}}}}catch(SQLException e){// Gestion des exceptions liées à la connexion ou à la requête e.printStackTrace();}
Stop[id=TOCMAS, name=Champ de Mars, latitude=5.940372, longitude=43.121363, locationType=stop]
Stop[id=TOCMAN, name=Champ de Mars, latitude=5.940497, longitude=43.12208, locationType=stop]
Stop[id=TOPIXO, name=Pie X, latitude=5.941181, longitude=43.120398, locationType=stop]
Les transactions
try(Connection connection =DriverManager.getConnection(DB_URL, DB_USERNAME, DB_PASSWORD)){// Création de la table accountString createAccountTableSql ="CREATE TABLE account ("+"id SERIAL PRIMARY KEY,"+"name VARCHAR(100) NOT NULL,"+"balance DEC(15,2) NOT NULL)";try(Statement statement = connection.createStatement()){ statement.executeUpdate(createAccountTableSql);}// Création des comptes pour Alice et Bob avec une requête préparéeString createAccountSql ="INSERT INTO account(name, balance) VALUES(?, ?);";try(PreparedStatement pstmt = connection.prepareStatement(createAccountSql)){ pstmt.setString(1,"Bob"); pstmt.setInt(2,1000); pstmt.executeUpdate(); pstmt.setString(1,"Alice"); pstmt.setInt(2,1000); pstmt.executeUpdate();}// Préparation de la requête pour mettre à jour le solde des comptesString updateAccountSql ="UPDATE account SET balance = balance + ? WHERE id = ?";try(PreparedStatement pstmtIncreaseAccount = connection.prepareStatement(updateAccountSql)){// Sauvegarde de l'état auto-commitboolean autoCommit = connection.getAutoCommit();try{ connection.setAutoCommit(false);// Désactivation de l'auto-commit pour la transaction// Retirer 500€ du compte de Bob pstmtIncreaseAccount.setInt(1,-500); pstmtIncreaseAccount.setInt(2,1);// ID de Bob pstmtIncreaseAccount.executeUpdate();// Ajouter 500€ au compte d'Alice pstmtIncreaseAccount.setInt(1,500); pstmtIncreaseAccount.setInt(2,2);// ID d'Alice pstmtIncreaseAccount.executeUpdate(); connection.commit();// Valider la transaction}catch(SQLException exc){ connection.rollback();// Annuler la transaction en cas de problèmethrow exc;// Relancer l'exception pour gestion ultérieure}finally{ connection.setAutoCommit(autoCommit);// Rétablir l'état auto-commit}}// Suppression de la table accounttry(Statement statement = connection.createStatement()){ statement.executeUpdate("DROP TABLE account");}}catch(SQLException e){ e.printStackTrace();// Gestion des exceptions SQL}
Procédures stockées
Les procédures stockées sont des sous-programmes stockés dans la base de données qui peuvent être exécutés par un appel direct. Elles permettent d’encapsuler des opérations complexes et de les réutiliser facilement.
Avantages des procédures stockées :
Réutilisabilité : Les procédures stockées peuvent être appelées à partir de différentes applications.
Performance : Les procédures stockées sont précompilées et stockées dans la base de données, ce qui améliore les performances.
Inconvénients potentiels :
Portabilité : Les procédures stockées peuvent être spécifiques à un SGBD, ce qui peut limiter la portabilité du code.
Complexité : Les procédures stockées peuvent être difficiles à déboguer et à maintenir.
Avec JDBC CallableStatement permet d’appeler une procédure stockée directement sur le SGBD.
-- Création d'une procédure stockée pour transférer de l'argent entre comptesCREATEORREPLACEPROCEDURE transfer_funds( from_account INT, to_account INT, amount DECIMAL) AS $$BEGINUPDATEaccountSET balance = balance - amount WHEREid= from_account;UPDATEaccountSET balance = balance + amount WHEREid= to_account;END;$$ LANGUAGE plpgsql;
// Connexion à la base de donnéestry(Connection connection =DriverManager.getConnection(DB_URL, DB_USERNAME, DB_PASSWORD)){// Préparation de l'appel à la procédure stockéeString callProcedureSql ="{CALL transfer_funds(?, ?, ?)}";try(CallableStatement callableStatement = connection.prepareCall(callProcedureSql)){// Définition des paramètres de la procédure callableStatement.setInt(1,1);// ID du compte source callableStatement.setInt(2,2);// ID du compte destination callableStatement.setBigDecimal(3,newBigDecimal("500.00"));// Montant à transférer// Exécution de la procédure stockée callableStatement.execute();}}catch(SQLException e){ e.printStackTrace();// Gestion des exceptions SQL}