Java 9 introduces a new level of abstraction above packages, formally known as the Java Platform Module System (JPMS), or “Modules” for short.
Java 9 modules are one of the biggest changes in the structure of Java.
What is Module?
A Module is a group of closely related packages and resources along with a new module descriptor file (module-info.java).
In other words, it's a “package of Java Packages” abstraction that allows us to make our code even more reusable.
Problems of Current Java System?
"Why we need Java SE 9 Module System” means the problems of the Current Java SE8 or earlier System.
- As JDK is too big, it is a bit tough to scale down to small devices. Java SE 8 has introduced 3 types of compact profiles to solve this problem: compact1, compact2, and compact3. But it does not solve this problem.
- JAR files like rt.jar etc are too big to use in small devices and applications.
- There is no Strong Encapsulation in the current Java System because the “public” access modifier is too open. Everyone can access it.
- As JDK, JRE is too big, it is hard to Test and Maintain applications.
- As the public is too open, They are not to avoid accessing of some Internal Non-Critical APIs like the sun.*, *.internal.*, etc. Security is also a big issue.
Advantages of Java SE 9 Module System
Java SE 9 Module System is going to provide the following benefits
- As Java SE 9 is going to divide JDK, JRE, JARs, etc, into smaller modules, we can use whatever modules we want. So it is very easy to scale down the Java Application to Small devices.
- Ease of Testing and Maintainability.
- As the public is not just public, it supports very Strong Encapsulation.
- We cannot access Internal Non-Critical APIs anymore.
- Modules can hide unwanted and internal details very safely, and we can get better Security.
- The application is too small because we can use only whatever modules we want.
- It's easy to support Single Responsibility Principle (SRP).
Compare JDK 8 and JDK 9
We know what JDK software contains. After installing JDK 8 software, we can see a couple of directories like bin, jre, lib etc in Java Home folder. However, Oracle Corp has changed this folder structure a bit differently as shown below.
Mother of Java 9 Module System
As of now, Java 9 Module System has 95 modules in Early Access JDK. Oracle Corp has separated JDK jars and Java SE Specifications into two sets of Modules.
- All JDK Modules start with “jdk.*”
- All Java SE Specifications Modules start with “java.*”
Java 9 Module System has a “java.base” Module. It’s known as Base Module. It’s an Independent module and does NOT dependent on any other modules. By default, all other Modules are dependent on this module. That’s why “java.base” Module is also known as The Mother of Java 9 Modules. It’s the default module for all JDK Modules and User-Defined Modules just like Object class is a base class of all the other classes.
Module Descriptor
When we create a module, we include a descriptor file that defines several aspects of our new module:
- Name – the name of our module
- Dependencies – a list of other modules that this module depends on
- Public Packages – a list of all packages we want accessible from outside the module
- Services Offered – we can provide service implementations that can be consumed by other modules
- Services Consumed – allows the current module to be a consumer of a service
- Reflection Permissions – explicitly allows other classes to use reflection to access the private members of a package
There are four types of modules in the new module system:
- System Modules – These are the modules listed when we run the list-modules command above. They include the Java SE and JDK modules.
- Application Modules – These modules are what we usually want to build when we decide to use Modules. They are named and defined in the compiled module-info.class file included in the assembled JAR.
- Automatic Modules – We can include unofficial modules by adding existing JAR files to the module path. The name of the module will be derived from the name of the JAR. Automatic modules will have full read access to every other module loaded by the path.
- Unnamed Module – When a class or JAR is loaded onto the classpath, but not the module path, it's automatically added to the unnamed module. It's a catch-all module to maintain backward compatibility with previously-written Java code.
We can see what these modules are by typing into the command line:
To set up a module, we need to put a special file at the root of our packages named module-info.java.
This file is known as the module descriptor and contains all of the data needed to build and use our new module.
And all public types exported from a dependency are accessible by our module when we use this directive.
Requires Static
Sometimes we write code that references another module, but that users of our library will never want to use.
For instance, we might write a utility function that pretty-prints our internal state when another logging module is present. But, not every consumer of our library will want this functionality, and they don't want to include an extra logging library.
In these cases, we want to use an optional dependency. By using the requires static directive, we create a compile-time-only dependency:
We commonly work with libraries to make our lives easier.
But, we need to make sure that any module that brings in our code will also bring in these extra ‘transitive' dependencies or they won't work.
Luckily, we can use the requires transitive directive to force any downstream consumers also to read our required dependencies:
Exports
By default, a module doesn't expose any of its API to other modules. This strong encapsulation was one of the key motivators for creating the module system in the first place.
Our code is significantly more secure, but now we need to explicitly open our API up to the world if we want it to be usable.
We use the exports directive to expose all public members of the named package:
Exports... To
We can use exports…to to open up our public classes to the world.
But, what if we don't want the entire world to access our API?
We can restrict which modules have access to our APIs using the exports…to the directive.
Similar to the exports directive, we declare a package as exported. But, we also list which modules we are allowing to import this package as a requires. Let's see what this looks like:
Uses
A service is an implementation of a specific interface or abstract class that can be consumed by other classes.
We designate the services our module consumes with the uses directive.
Note that the class name we use is either the interface or abstract class of the service, not the implementation class:
Provides With
A module can also be a service provider that other modules can consume.
The first part of the directive is the provides keyword. Here is where we put the interface or abstract class name.
Next, we have the with directive where we provide the implementation class name that either implements the interface or extends the abstract class.
Here's what it looks like put together:
Open
We mentioned earlier that encapsulation was a driving motivator for the design of this module system.
Before Java 9, it was possible to use reflection to examine every type and member in a package, even the private ones. Nothing was truly encapsulated, which can open up all kinds of problems for developers of the libraries.
Because Java 9 enforces strong encapsulation, we now have to explicitly grant permission for other modules to reflect on our classes.
If we want to continue to allow full reflection as older versions of Java did, we can simply open the entire module up:
If we need to allow reflection of private types, but we don't want all of our code exposed, we can use the opens directive to expose specific packages.
But remember, this will open the package up to the entire world, so make sure that is what you want:
Okay, so reflection is great sometimes, but we still want as much security as we can get from encapsulation. We can selectively open our packages to a pre-approved list of modules, in this case, using the opens…to directive:
No comments:
Post a Comment