Java & Spring Boot
- Mise en place de l'environnement de développement
- Installer Java
- Installer IntelliJ (Standalone)
- Vérifier que les outils de compilation et d'exécution sont correctement configurés
- Bases de la programmation Java
- Syntaxe Java de base : les variables, les types de données, les opérateurs et les tableaux
- Les concepts de la programmation orientée objet : classes, objets, encapsulation, héritage et polymorphisme
- Collections
- Enumérations
- TP bases de la programmation Java
- Les exceptions
- Gestion des Tests
- Les designs patterns
- Spring Boot
- Compilation et distribution
Mise en place de l'environnement de développement
Installer Java
Une des manières la plus simple d'installer java sur une distribution Debian, serait d'installer une version openjdk, car cette dernière est déjà présente dans les dépôts de base.
Mettre à jour la liste des dépendances
sudo apt-get update && sudo apt-get upgrade
Installer OpenJDK 11 development kit (qui contient à la fois les outils pour compiler, que pour lancer une application java)
sudo apt-get install openjdk-11-jdk
Tester la version de java installé
java -version
Sources
https://www.linode.com/docs/guides/how-to-install-openjdk-on-debian-10/
Installer IntelliJ (Standalone)
IntelliJ est surement l'un des meilleurs IDE du marché pour le développement Java.
On peut installer la version Ultimate, si l'on possède la licence, ou bien la version Community, plus limité, mais gratuite.
Télécharger IntelliJ
https://www.jetbrains.com/idea/download/#section=linux
Installer IntelliJ Ultimate
sudo tar -xzf ideaIU-*.tar.gz -C /opt
Installer IntelliJ Community
sudo tar -xzf ideaIC-*.tar.gz -C /opt
Ensuite, exécuter le fichier idea.sh se trouvant dans le chemin /opt/idea-*/bin/idea.sh
Sources
https://www.jetbrains.com/help/idea/installation-guide.html#standalone
Vérifier que les outils de compilation et d'exécution sont correctement configurés
Pour ce faire, il suffira de créer un projet Java tout simple dans IntelliJ, pour vérifier que vous obtenez bien le petit message Hello World dans la console !
Sources
Premier projet
https://koor.fr/Java/Tutorial/FirstJavaProgram.wp
Bases de la programmation Java
Syntaxe Java de base : les variables, les types de données, les opérateurs et les tableaux
Variables et syntaxe de base
https://fre.myservername.com/java-basics-java-syntax
https://www.jmdoudoux.fr/java/dej/chap-syntaxe.htm
https://gayerie.dev/epsi-b3-java/langage_java/types_primitifs.html
https://gayerie.dev/epsi-b3-java/langage_java/la_classe_string.html
Opérateurs
https://koor.fr/Java/Tutorial/JavaOperators.wp
https://gayerie.dev/epsi-b3-java/langage_java/operateurs.html
Structures de contrôles
https://gayerie.dev/epsi-b3-java/langage_java/structures_de_controle.html#les-structures-de-controle
Tableaux
https://koor.fr/Java/Tutorial/java_tableaux.wp
https://gayerie.dev/epsi-b3-java/langage_java/tableau.html
Les concepts de la programmation orientée objet : classes, objets, encapsulation, héritage et polymorphisme
Classes
https://gayerie.dev/epsi-b3-java/langage_java/premiere_classe.html
https://koor.fr/Java/Tutorial/java_poo_encapsulation.wp
Attributs et méthodes
https://gayerie.dev/epsi-b3-java/langage_java/attributs_et_methodes.html
https://koor.fr/Java/Tutorial/java_poo_encapsulation.wp
Héritage & Implémentation
https://gayerie.dev/epsi-b3-java/langage_java/heritage_composition.html
https://koor.fr/Java/Tutorial/java_poo_heritage.wp
https://gayerie.dev/epsi-b3-java/langage_java/interface.html
https://koor.fr/Java/Tutorial/java_poo_type_abstrait.wp#interface
https://koor.fr/Java/Tutorial/java_poo_interface.wp
Polymorphisme
https://gayerie.dev/epsi-b3-java/langage_java/polymorphisme.html
Collections
https://gayerie.dev/epsi-b3-java/langage_java/les_collections.html
https://gayerie.dev/epsi-b3-java/langage_java/streams.html
Enumérations
https://koor.fr/Java/Tutorial/java_enum.wp
https://gayerie.dev/epsi-b3-java/langage_java/enumeration.html
TP bases de la programmation Java
Dans ce TP, vous allez développer un système de gestion pour une bibliothèque. Nous allons créer plusieurs classes et une interface pour modéliser le fonctionnement de la bibliothèque.
Partie 1 : Création de la classe Publisher
-
Dans un nouveau fichier
Publisher.java, créez une classePublisherqui représente un éditeur de livres. Cette classe doit avoir les attributs privés suivants :name: le nom de l'éditeur (String)address: l'adresse de l'éditeur (String)
-
Ajoutez un constructeur à la classe
Publisherqui prend en entrée un nom et une adresse, et initialise les attributs correspondants. -
Ajoutez des méthodes getters et setters pour les attributs de la classe
Publisher.
Partie 2 : Création de la classe Book
Un livre doit avoir les attributs privés suivants :
title: Le nom du livreauthor: L'auteur du livrenumberOfPages: Le nombre de pages
-
Créez votre classe
Book, y inclure un nouvel attribut privépublisherde typePublisher. -
Modifiez le constructeur de
Bookpour prendre en compte le nouvel attributpublisher. -
Ajoutez une méthode getter et une méthode setter pour l'attribut
publisher.
Partie 3 : Création de l'interface Loanable
- Créez une nouvelle interface
Loanable. Cette interface doit définir les méthodes suivantes :loanTo(String borrower): Cette méthode ne retourne rien et modifie l'état de l'objet pour indiquer qu'il est emprunté par l'emprunteur passé en paramètre.returnBook(): Cette méthode ne retourne rien et modifie l'état de l'objet pour indiquer qu'il n'est plus emprunté.
Partie 4 : Modification de la classe Book pour implémenter Loanable
-
Modifiez la classe
Bookpour qu'elle implémente l'interfaceLoanable. -
Ajoutez un nouvel attribut privé
borrowerde typeStringà la classeBook. Cette variable doit êtrenullquand le livre n'est pas emprunté. -
Implémentez les méthodes
loanToetreturnBookpour modifier l'attributborrower.
Partie 5 : Création de la classe Library
-
Dans un nouveau fichier
Library.java, créez une classeLibraryqui représente une bibliothèque. Cette classe doit avoir un seul attribut privébooks, qui est une liste deBook. -
Ajoutez un constructeur sans arguments à la classe
Libraryqui initialise l'attributbookscomme une nouvelleArrayList. -
Ajoutez une méthode
addBookà la classeLibraryqui prend unBooken argument et l'ajoute à la listebooks. -
Ajoutez une méthode
searchBookà la classeLibraryqui prend un titre de livre en argument et renvoie leBookcorrespondant s'il est dans la liste, etnullsinon. -
Ajoutez une méthode
loanBookà la classeLibraryqui prend un titre de livre et un emprunteur en argument, et emprunte le livre correspondant s'il est dans la liste et n'est pas déjà emprunté. Cette méthode doit lancer une exception si le livre n'est pas disponible.
Partie 6 : Création de la classe LibraryApp
- Enfin, dans un nouveau fichier
LibraryApp.java, créez une classeLibraryAppavec une méthodemain. Dans cette méthode, créez plusieurs objetsBook, ajoutez-les à un objetLibrary, et testez la recherche et l'emprunt de livres.
Arborescence de fichiers
Voici une possible arborescence de fichiers pour ce TP :
src/
├── main/
│ ├── java/
│ │ ├── LibraryApp.java
│ │ ├── Book.java
│ │ ├── Publisher.java
│ │ ├── Loanable.java
│ │ ├── Library.java
Code complet
Voici une implémentation possible de ce TP, en supposant que chaque fichier se trouve dans le package com.example.library :
Publisher.java
package com.example.library;
public class Publisher {
private String name;
private String address;
public Publisher(String name, String address) {
this.name = name;
this.address = address;
}
// Getters and setters
public String getName() {
return this.name;
}
public void setName(String name) {
this.name = name;
}
public String getAddress() {
return this.address;
}
public void setAddress(String address) {
this.address = address;
}
}
Loanable.java
package com.example.library;
public interface Loanable {
void loanTo(String borrower);
void returnBook();
}
Book.java
package com.example.library;
public class Book implements Loanable {
private String title;
private String author;
private int numberOfPages;
private Publisher publisher;
private String borrower;
public Book(String title, String author, int numberOfPages, Publisher publisher) {
this.title = title;
this.author = author;
this.numberOfPages = numberOfPages;
this.publisher = publisher;
this.borrower = null;
}
// Getters and setters
public String getTitle() {
return this.title;
}
public void setTitle(String title) {
this.title = title;
}
public String getAuthor() {
return this.author;
}
public void setAuthor(String author) {
this.author = author;
}
public int getNumberOfPages() {
return this.numberOfPages;
}
public void setNumberOfPages(int numberOfPages) {
this.numberOfPages = numberOfPages;
}
public Publisher getPublisher() {
return this.publisher;
}
public void setPublisher(Publisher publisher) {
this.publisher = publisher;
}
public String getBorrower() {
return this.borrower;
}
@Override
public void loanTo(String borrower) {
if (this.borrower != null) {
throw new IllegalStateException("Book is already borrowed.");
}
this.borrower = borrower;
}
@Override
public void returnBook() {
this.borrower = null;
}
}
Library.java
package com.example.library;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
public class Library {
private List<Book> books;
public Library() {
books = new ArrayList<>();
}
public void addBook(Book book) {
books.add(book);
}
public Book searchBook(String title) {
Optional<Book> foundBook = books.stream()
.filter(book -> Objects.equals(book.getTitle(), title))
.findFirst();
return foundBook.orElse(null);
}
public void loanBook(String title, String borrower) {
Book book = searchBook(title);
if (book == null) {
throw new IllegalArgumentException("Book not found.");
}
book.loanTo(borrower);
}
}
LibraryApp.java
package com.example.library;
public class LibraryApp {
public static void main(String[] args) {
// Create Publisher
Publisher penguin = new Publisher("Penguin", "New York");
// Create Books
Book book1 = new Book("1984", "George Orwell", 328, penguin);
Book book2 = new Book("To Kill a Mockingbird", "Harper Lee", 281, penguin);
// Create Library
Library library = new Library();
// Add books to library
library.addBook(book1);
library.addBook(book2);
// Test searching and loaning books
System.out.println("Searching for '1984': " + library.searchBook("1984"));
library.loanBook("1984", "John Doe");
System.out.println("'1984' borrowed by: " + book1.getBorrower());
}
}
Les exceptions
https://gayerie.dev/epsi-b3-java/langage_java/les_exceptions.html
https://koor.fr/Java/Tutorial/java_exception_throw_try_catch_finally.wp
https://koor.fr/Java/Tutorial/java_exception_classe.wp
https://koor.fr/Java/Tutorial/java_exception_try_with_resources.wp
Gestion des Tests
Pourquoi tester
https://koor.fr/Java/Tutorial/java_junit_unit_test.wp
Les bonnes pratiques
https://zestedesavoir.com/tutoriels/274/les-tests-unitaires-en-java/
https://www.baeldung.com/java-unit-testing-best-practices
Les designs patterns
https://koor.fr/Java/Tutorial/java_design_patterns.wp
Meilleur site pour les design patterns : https://refactoring.guru/design-patterns
Spring Boot
Découverte
https://devstory.net/11267/tutoriel-spring-boot-pour-debutant
https://www.youtube.com/playlist?list=PLtyD11a_24egpX6V-BXtVnBmrbR60I1P6
Gestion de la persistance avec JPA et Hibernate
https://gayerie.dev/docs/spring/spring/spring_data_jpa.html
https://www.baeldung.com/the-persistence-layer-with-spring-data-jpa
https://www.baeldung.com/spring-boot-hibernate
https://www.javadevjournal.com/spring-boot/spring-boot-with-hibernate/
Clean architecture
https://www.baeldung.com/spring-boot-clean-architecture
https://medium.com/swlh/clean-architecture-java-spring-fea51e26e00
https://reflectoring.io/java-components-clean-boundaries/
TP Spring Boot
Dans ce TP, vous allez développer un système de gestion pour une bibliothèque en utilisant Spring Boot et Spring Data JPA. Nous allons créer plusieurs classes, des interfaces pour les dépôts de données et des contrôleurs pour exposer nos services via une API REST.
Partie 1: Initialisation du projet Spring Boot
-
Allez sur Spring Initializr pour générer un nouveau projet Spring Boot. Choisissez "Maven Project", "Java", et la dernière version de Spring Boot.
-
Dans la section "Project Metadata", donnez un nom de groupe et d'artefact approprié, par exemple "com.example" et "library".
-
Dans la section "Dependencies", ajoutez les dépendances suivantes: Spring Web, Spring Data JPA, et Spring Boot DevTools.
-
Cliquez sur "Generate" pour télécharger un fichier zip du projet. Décompressez le fichier et ouvrez le projet dans votre IDE favori.
Partie 2: Création de la classe Publisher
-
Dans le package
model, créez une classePublisherqui représente un éditeur de livres. Cette classe doit avoir les attributs suivants :id: l'ID de l'éditeur (Long)name: le nom de l'éditeur (String)address: l'adresse de l'éditeur (String)
-
Annoter la classe avec
@Entitypour indiquer qu'il s'agit d'une entité JPA. -
Annoter l'attribut
idavec@Idet@GeneratedValuepour indiquer qu'il s'agit de la clé primaire et qu'elle est générée automatiquement.
Partie 3: Création de l'interface PublisherRepository
- Dans le package
repository, créez une interfacePublisherRepositoryqui étendJpaRepository<Publisher, Long>. Cela nous donne gratuitement plusieurs méthodes pour interagir avec la base de données commefindAll(),findById(),save(),delete(), etc.
Partie 4: Création de la classe Book
-
Dans le package
model, créez une classeBookqui représente un livre. Cette classe doit avoir les attributs suivants :id: l'ID du livre (Long)title: le titre du livre (String)author: l'auteur du livre (String)publisher: l'éditeur du livre (Publisher)
-
Annoter la classe avec
@Entitypour indiquer qu'il s'agit d'une entité JPA. -
Annoter l'attribut
idavec@Idet@GeneratedValuepour indiquer qu'il s'agit de la clé primaire et qu'elle est générée automatiquement. -
Annoter l'attribut
publisheravec@ManyToOnepour indiquer la relation entreBooketPublisher.
Partie 5: Création de l'interface BookRepository
- Dans le package
repository, créez une interfaceBookRepositoryqui étendJpaRepository<Book, Long>.
Partie 6: Création des contrôleurs
-
Dans le package
controller, créez une classePublisherControlleravec une méthodegetAllPublishers()qui retourne tous les éditeurs, et une méthodeaddPublisher()qui ajoute un nouvel éditeur. -
Dans le même package, créez une classe
BookControlleravec une méthodegetAllBooks()qui retourne tous les livres, et une méthodeaddBook()qui ajoute un nouveau livre.
Partie 7: Test de l'application
- Exécutez l'application et utilisez un outil comme Postman pour tester votre API REST.
Arborescence de fichiers
Voici une possible arboressence de fichiers pour ce TP :
src/
├── main/
│ ├── java/
│ │ ├── com.example.library
│ │ │ ├── LibraryApplication.java
│ │ │ ├── model/
│ │ │ │ ├── Book.java
│ │ │ │ ├── Publisher.java
│ │ │ ├── repository/
│ │ │ │ ├── BookRepository.java
│ │ │ │ ├── PublisherRepository.java
│ │ │ ├── controller/
│ │ │ │ ├── BookController.java
│ │ │ │ ├── PublisherController.java
│ ├── resources/
│ │ ├── application.properties
Code
Model
Book
package com.example.library.model;
import javax.persistence.*;
@Entity
public class Book {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String title;
private String author;
@ManyToOne
private Publisher publisher;
public Book() {
}
public Book(String title, String author, Publisher publisher) {
this.title = title;
this.author = author;
this.publisher = publisher;
}
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getAuthor() {
return author;
}
public void setAuthor(String author) {
this.author = author;
}
public Publisher getPublisher() {
return publisher;
}
public void setPublisher(Publisher publisher) {
this.publisher = publisher;
}
}
Publisher
package com.example.library.model;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
@Entity
public class Publisher {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String name;
private String address;
public Publisher() {
}
public Publisher(String name, String address) {
this.name = name;
this.address = address;
}
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
}
Repository
BookRepository
package com.example.library.repository;
import com.example.library.model.Book;
import org.springframework.data.jpa.repository.JpaRepository;
public interface BookRepository extends JpaRepository<Book, Long> {
}
PublisherRepository
package com.example.library.repository;
import com.example.library.model.Publisher;
import org.springframework.data.jpa.repository.JpaRepository;
public interface PublisherRepository extends JpaRepository<Publisher, Long> {
}
Controller
BookController
package com.example.library.controller;
import com.example.library.model.Book;
import com.example.library.repository.BookRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
import java.util.List;
@RestController
public class BookController {
@Autowired
private BookRepository bookRepository;
@GetMapping("/books")
public List<Book> getAllBooks() {
return bookRepository.findAll();
}
@PostMapping("/books")
public Book addBook(@RequestBody Book book) {
return bookRepository.save(book);
}
}
PublisherController
package com.example.library.controller;
import com.example.library.model.Publisher;
import com.example.library.repository.PublisherRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
import java.util.List;
@RestController
public class PublisherController {
@Autowired
private PublisherRepository publisherRepository;
@GetMapping("/publishers")
public List<Publisher> getAllPublishers() {
return publisherRepository.findAll();
}
@PostMapping("/publishers")
public Publisher addPublisher(@RequestBody Publisher publisher) {
return publisherRepository.save(publisher);
}
}
Compilation et distribution
Spring Boot
https://docs.spring.io/spring-boot/docs/current/maven-plugin/reference/htmlsingle/
https://www.baeldung.com/spring-boot-run-maven-vs-executable-jar
https://www.yawintutor.com/how-to-create-executable-jar-file-in-spring-boot-using-maven/
Maven
https://maven.apache.org/guides/introduction/introduction-to-the-lifecycle.html
https://www.jmdoudoux.fr/java/dej/chap-maven.htm