Spring Course 2024v4
Spring Course 2024v4
Spring
I. Introduction
Spring is a Java framework for building web applications.
- Spring Boot: Takes an opinionated view of building Spring applications and gets you up and
running as quickly as possible.
- Spring Framework: Provides core support for dependency injection, transaction management,
web apps, data access, messaging, and more.
- Spring Data: Provides a consistent approach to data access – relational, non-relational, map-
reduce, and beyond.
- Spring Cloud: Provides a set of tools for common patterns in distributed systems. Useful for
building and deploying microservices.
- Spring Cloud Data Flow: Provides an orchestration service for composable data microservice
applications on modern runtimes.
- Spring Security: Protects your application with comprehensive and extensible authentication
and authorization support.
- Spring Authorization Server: Provides a secure, light-weight, and customizable foundation for
building OpenID Connect 1.0 Identity Providers and OAuth2 Authorization Server products.
- Spring for GraphQL: Spring for GraphQL provides support for Spring applications built on
GraphQL Java.
- Spring Session: Provides an API and implementations for managing a user’s session information.
- Spring Integration: Supports the well-known Enterprise Integration Patterns through lightweight
messaging and declarative adapters.
- Spring Web Services: Facilitates the development of contract-first SOAP web services.
See: https://spring.io/projects
The term "Spring" means different things in different contexts. It can be used to refer to the
Spring Framework project itself. Most often, when people say "Spring", they mean the entire
family of projects. (link: https://docs.spring.io/spring-framework/reference/overview.html)
1
II. Spring Framework
The current version of Spring Framework is 6.1.3. As of Spring Framework 6.0, Spring requires Java 17+.
The Spring Framework uses Beans. It supports the Dependency Injection and Common Annotations.
Features:
Core technologies: dependency injection, events, resources, i18n, validation, data binding, type
conversion, SpEL, AOP.
Integration: remoting, JMS, JCA, JMX, email, tasks, scheduling, cache and observability.
1. Annotations
@Entity
The at sign character (@) indicates to the compiler that what follows is an annotation. In the following
example, the annotation's name is Override:
@Override
void mySuperMethod() {
}
2
The annotation can include elements, which can be named or unnamed, and there are values for those
elements:
or
@SuppressWarnings(value = "unchecked")
void myMethod() {
}
If there is just one element named value, then the name can be omitted, as in:
@SuppressWarnings("unchecked")
void myMethod() {
}
If the annotation has no elements, then the parentheses can be omitted, as shown in the previous
@Override example.
The annotation type can be one of the types that are already defined. In the previous examples,
Override and SuppressWarnings are predefined Java annotations. It is also possible to define your own
annotation type. The Author and Ebook annotations in the previous example are custom annotations.
Annotations can be applied to declarations: declarations of classes, fields, methods, and other program
elements.
3
2. Beans
@RestController
public class HomeController {
@GetMapping
public String index() {
return "hello world";
}
}
Example:
@SpringBootApplication
public class WebapiApplication {
public static void main(String[] args) {
SpringApplication.run(WebapiApplication.class, args);
}
}
3. Dependency injection
Dependency injection (DI) is a process where objects define their dependencies (that is, the other
objects they need). The container then injects those dependencies when it creates the bean.
4
A class can define its dependencies in different ways:
- constructor arguments
- arguments to a method, or
- fields annotated with @Autowired
The following example shows a class that can be dependency-injected with @Autowired annotation:
Spring is mainly used to create REST Web Services (RESTful Web API). It lets you create special
@RestController beans to handle incoming HTTP requests. Methods in your controller are mapped to
HTTP by using @RequestMapping annotations.
The following code shows a typical @RestController that serves JSON data:
5
@RestController
@RequestMapping("/users")
public class MyRestController {
private final UserRepository userRepository;
@GetMapping
public List<User> getUsers() {
return this.userRepository.findAll();
}
@GetMapping("/{userId}")
public User getUser(@PathVariable Long userId) {
return this.userRepository.findById(userId).get();
}
@PostMapping
public void createUser(@RequestBody User user) {
// create user
}
@PutMapping("/{userId}")
public User updateUser(@PathVariable Long userId, @RequestBody User user)
{
// update user
}
@DeleteMapping("/{userId}")
public void deleteUser(@PathVariable Long userId) {
this.userRepository.deleteById(userId);
}
}
5. Spring MVC
The Spring Web MVC framework (often referred to as “Spring MVC”) is a rich “model view controller”
web framework.
Spring MVC lets you create special @Controller or @RestController beans to handle incoming HTTP
requests.
6
Spring MVC allows us to serve dynamic HTML content. It supports a variety of templating technologies,
including:
- Thymeleaf
- JSP
- FreeMarker
- Groovy
- Mustache
If possible, JSPs should be avoided. There are several known limitations when using them with
embedded servlet containers.
When you use one of these templating engines with the default configuration, your templates are
picked up automatically from src/main/resources/templates.
Spring Boot makes it easy to create Spring based Applications that you can "just run".
Features
Getting Started
7
1. Create new project
The easiest way to create a new project is using Spring initializr: https://start.spring.io/
We choose Maven as the project building tool, and Java as the programming language
- Group : name of the application owner (or his reversed domain name)
- Artifact: the application name (Artifact = Name)
Generate the Zip file, extract it and open it using your IDE
8
2. Project Structure
Required tools:
- IDE
o
IntelliJ IDE
o
Eclipse IDE for Enterprise Java and Web Developers:
https://www.eclipse.org/downloads/packages/
o VS Code + Java Extension Pack + Spring Boot Extension Pack
- Postman
9
TP 1
10
Chapter 2. Spring Web
I. Controllers
Link : https://docs.spring.io/spring-framework/reference/web/webmvc/mvc-controller.html
1. Declaration
@RestController
public class HelloController {
@GetMapping("/hello")
public String handle() {
return "Hello World!";
}
}
In the preceding example, the method is accessible using the HTTP GET method at the URL “/hello” and
returns a String.
2. API Endpoint
An API endpoint is a URL that acts as the point of contact between a client and a server:
- Clients send requests to API endpoints in order to access the API’s functionality and data.
- A typical REST API has many endpoints that correspond to its available resources.
11
For example, a REST API for blogging application might have the following URLs endpoints:
3. Mapping Requests
You can use the @RequestMapping annotation to map requests to controller methods. You can use it at
the class level to express shared mappings or at the method level to express a specific endpoint
mapping.
The following example creates an endpoint accessible using the HTTP GET method at the URL “/
@GetMapping
@PostMapping
@PutMapping
@DeleteMapping
@PatchMapping
The shortcuts are mainly used with controller methods while @RequestMapping is usually needed at
the class level to express shared mappings.
12
The following example has class and method level mappings:
@RestController
@RequestMapping("/books")
public class BooksController {
@GetMapping
public String getBooks() {
return "getBooks";
}
@GetMapping("list")
public String getBooks2() {
return "getBooks2";
}
The method “getBooks2” is accessible at the URL “/books/list” (URL “/books/list/” is not defined)
The method “getBooks3” is accessible at the URL “/books/get”, “/books/getAll” and “/books/getAll/”
Methods annotated with @RequestMapping (and its shortcuts) can be mapped using URL patterns.
Example: "/projects/{project}/versions" - match a URL segment (project) and capture it as a variable.
@GetMapping("/owners/{ownerId}/pets/{petId}")
public String findPet(@PathVariable Long ownerId, @PathVariable Long petId) {
// ...
}
4. Handler Methods
Controller methods annotated with @RequestMapping (or its shortcuts) are called handler methods.
13
The method arguments are called the method signature.
Handler methods have a flexible signature and can choose from a range of supported arguments and
return values.
a. Method Arguments
The complete list of supported controller method arguments is available at this link.
Example 1:
@RestController
@RequestMapping("/books")
public class BooksController {
@GetMapping
public String getBooks(
@RequestParam String author,
@RequestParam(defaultValue = "all") String category) {
return "Get Books of " + author + " - category : " + category;
}
}
14
The response of a GET request to http://localhost/books?author=Ali&category=Science is: Get Books
of Ali - category : Science
The response of a GET request to http://localhost/books?author=Ali is: Get Books of Ali - category : all
Example 2:
@RestController
@RequestMapping("/books")
public class BooksController {
@PutMapping("{id}")
public String updateBook(
@PathVariable int id,
@RequestBody Book book) {
return "id=" + id + " - title: " + book.getTitle();
}
}
15
b. Return Values
The complete list of supported controller method return values is available at this link. The next table
describes some of them:
Example 1:
@GetMapping("/titles")
public List<String> getTitles() {
List<String> titles = new ArrayList<String>();
titles.add("title1");
titles.add("title2");
titles.add("title3");
return titles;
}
Response:
[
"title1",
"title2",
"title3"
]
16
Example 2:
@GetMapping("/books/best")
public Book getBestBook() {
Book book = new Book();
book.setTitle("Spring Boot Guide");
book.setAuthor("Ali Dridi");
book.setIsbn("45874584211");
book.setPrice(78.9);
return book;
}
Response:
{
"title": "Spring Boot Guide",
"author": "Ali Dridi",
"isbn": "45874584211",
"price": 78.9
}
Example 3:
@GetMapping("{id}")
public ResponseEntity<Object> getBook(@PathVariable int id) {
Book book = new Book();
book.setTitle("Spring Boot Guide");
book.setAuthor("Ali Dridi");
book.setIsbn("45874584211");
book.setPrice(78.9);
Response to http://localhost:8080/books/0 :
17
c. Validation
Link: https://docs.spring.io/spring-framework/reference/web/webmvc/mvc-controller/ann-
validation.html
Method parameters such as @RequestBody and @RequestPart do perform validation if annotated with
Jakarta’s @Valid or Spring’s @Validated annotation, and raise MethodArgumentNotValidException in
case of validation errors. If you want to handle the errors in the controller method instead, you can
declare an Errors or BindingResult method parameter immediately after the validated parameter.
- Add the “Validation” dependency to the project. It can be added while creating a new
project using Spring initializr, or you can add the following tag to “pom.xml”:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-validation</artifactId>
</dependency>
- Create a new Model and annotate its fields using the appropriate annotations, example:
@Min(0)
private double price;
- Add the @Valid annotation to method parameters such as @RequestBody. This will raise
MethodArgumentNotValidException in case of validation errors.
18
- To handle the errors in the controller method instead, you can declare a BindingResult
method parameter immediately after the validated parameter, example:
@PostMapping
public ResponseEntity<Object> createBook(
@Valid @RequestBody Book book,
BindingResult result
) {
if (result.hasErrors()) {
var errorsList = result.getAllErrors();
var errorsMap = new HashMap<String, String>();
return ResponseEntity.badRequest().body(errorsMap);
}
return ResponseEntity.ok(book);
}
19
TP 2
1) Add the “Validation” dependency to the “webapi” project by adding the following xml tag to
“pom.xml”:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-validation</artifactId>
</dependency>
2) Create the “models” package and add 2 models in this package: Product and ProductDto.
ProductDto has the following fields: String name, String brand, String category, double price,
String description
All these fields are private and required. Generate the getters and setters using the IDE. The
Product model has the same fields in addition to the “int id” and “Date createdAt”.
20
Chapter 3. Database
I. Java JDBC
1) Introduction
The Java API for Database connectivity is called JDBC (Java DataBase Connectivity). This API consists of
two packages:
21
2) Classes
a) DriverManager
This is the basic service for managing a set of JDBC drivers. This class will attempt to load the driver
classes.
Methods:
Example:
try {
// connect to MYSQL server
Connection connection1 = DriverManager.getConnection(MYSQL_SERVER_URL,
USERNAME, PASSWORD);
22
b) Connection
This class corresponds to a connection (session) with a specific database. SQL statements are executed
and results are returned within the context of a connection.
A Connection object's database is able to provide information describing its tables, its supported SQL
grammar, its stored procedures, the capabilities of this connection, and so on. This information is
obtained with the getMetaData method.
Methods:
- Statement createStatement(): Creates a Statement object for sending SQL statements to the
database.
- PreparedStatement prepareStatement(String sql): Creates a PreparedStatement object for
sending parameterized SQL statements to the database.
- void close(): Releases this Connection object's database and JDBC resources immediately
instead of waiting for them to be automatically released.
c) Statement
This is the object used for executing a static SQL statement and returning the results it produces.
Methods:
- ResultSet executeQuery(String sql): Executes the given SQL statement, which returns a single
ResultSet object.
- int executeUpdate(String sql): Executes the given SQL statement, which may be an INSERT,
UPDATE, or DELETE statement or an SQL statement that returns nothing.
- boolean execute(String sql): Executes the given SQL statement, which may return multiple
results.
- void close(): Releases this Statement object's database and JDBC resources immediately instead
of waiting for this to happen when it is automatically closed.
d) PreparedStatement
A SQL statement is precompiled and stored in a PreparedStatement object. This object can then be used
to efficiently execute this statement multiple times.
23
Methods:
- void setInt(int parameterIndex, int x): Sets the designated parameter to the given Java int
value.
- void setString(int parameterIndex, String x): Sets the designated parameter to the given Java
String value.
- void setTimestamp(int parameterIndex, Timestamp x): Sets the designated parameter to the
given java.sql.Timestamp value.
- ResultSet executeQuery(): Executes the SQL query in this PreparedStatement object and returns
the ResultSet object generated by the query.
- Int executeUpdate(): Executes the SQL statement in this PreparedStatement object, which must
be an SQL Data Manipulation Language (DML) statement, such as INSERT, UPDATE or DELETE; or
an SQL statement that returns nothing.
- boolean execute(): Executes the SQL statement in this PreparedStatement object, which may be
any kind of SQL statement.
- void close(): Releases this Statement object's database and JDBC resources immediately instead
of waiting for this to happen when it is automatically closed.
e) ResultSet
A table of data representing a database result set, which is usually generated by executing a statement
that queries the database.
A ResultSet object maintains a cursor pointing to its current row of data. Initially the cursor is positioned
before the first row. The next method moves the cursor to the next row, and because it returns false
when there are no more rows in the ResultSet object, it can be used in a while loop to iterate through
the result set.
A default ResultSet object is not updatable and has a cursor that moves forward only. Thus, you can
iterate through it only once and only from the first row to the last row. It is possible to produce
ResultSet objects that are scrollable and/or updatable.
Methods:
- boolean next(): Moves the cursor forward one row from its current position.
- int getInt(int columnIndex): Retrieves the value of the designated column in the current row of
this ResultSet object as an int in the Java programming language.
- int getInt(String columnLabel): Retrieves the value of the designated column in the current row
of this ResultSet object as an int in the Java programming language.
24
- String getString(int columnIndex): Retrieves the value of the designated column in the current
row of this ResultSet object as a String in the Java programming language.
- String getString(String columnLabel): Retrieves the value of the designated column in the
current row of this ResultSet object as a String in the Java programming language.
- Timestamp getTimestamp(int columnIndex): Retrieves the value of the designated column in
the current row of this ResultSet object as a java.sql.Timestamp object in the Java programming
language.
- Timestamp getTimestamp(String columnLabel): Retrieves the value of the designated column
in the current row of this ResultSet object as a java.sql.Timestamp object in the Java
programming language.
3) Examples
a) Read
try {
// connect to MYSQL server
Connection connection = DriverManager.getConnection(MYSQL_SERVER_URL,
USERNAME, PASSWORD);
statement.close();
connection.close();
25
Read a post by id using a static SQL statement:
try {
int postId = 2;
// connect to MYSQL server
Connection connection = DriverManager.getConnection(MYSQL_SERVER_URL,
USERNAME, PASSWORD);
statement.close();
connection.close();
try {
int postId = 2;
// connect to MYSQL server
Connection connection = DriverManager.getConnection(MYSQL_SERVER_URL,
USERNAME, PASSWORD);
26
statement.close();
connection.close();
b) Create
try {
Date date = new Date();
Timestamp timestamp = new Timestamp(date.getTime());
preparedStatement.close();
connection.close();
}
catch(Exception ex) {
ex.printStackTrace();
}
27
c) Update
preparedStatement.close();
connection.close();
}
catch(Exception ex) {
ex.printStackTrace();
}
d) Delete
try {
int postId = 2;
// connect to the database
Connection connection = DriverManager.getConnection(DB_URL,
USERNAME, PASSWORD);
preparedStatement.setInt(1, postId);
28
if (deletedRows > 0) {
// Post deleted successfully
}
preparedStatement.close();
connection.close();
}
catch(Exception ex) {
ex.printStackTrace();
}
4. Connect to MySQL
To connect to MySQL database using Java JDBC, it is necessary to add the “MySQL connector for Java”
driver, which is a JAR file. You can add the following XML tag to the “pom.xml” file of the Maven project:
<dependency>
<groupId>com.mysql</groupId>
<artifactId>mysql-connector-j</artifactId>
<scope>runtime</scope>
</dependency>
The Repository Pattern is a design pattern used in software development that provides a way to manage
data access logic in a centralized location:
- It separates the logic that retrieves the data and maps it to the entity model from the business
logic that operates on the model.
- The primary objective of the Repository Pattern is to simplify the data access code and achieve a
cleaner architecture for the application.
- It reduces the code duplication and improves the code maintainability.
- The Entity Model: a class that stores the data transferred between the business layer (the
controller) and the data access layer (the repository).
- The Repository: a class that implements the operations that can be performed on the entity
model. These operations typically include Create, Read, Update, and Delete (CRUD) operations.
29
Spring @Repository annotation is used to indicate that the class is an application component (i.e. bean).
So Spring Repository classes are auto detected by spring framework through classpath scanning.
Repository example:
@Repository
public class PostsRepository
{
public List<Post> getPosts()
{
// return the list of posts
}
30