Giới thiệu về Java Module System

Java Module System được Java giới thiệu từ Java 9 để giải quyết 1 số vấn đề của Java mà nếu các bạn để ý sẽ thấy nó thật cần thiết như thế nào?

  • Lỗi NoClassDefFoundError: lỗi này xảy ra lúc chương trình Java của chúng ta đang chạy, JVM không thể tìm thấy class cần thiết để execute một tác vụ nào đó. Mặc dù hiện tại chúng ta có các tool để build và fetch tất cả dependencies cần thiết để chạy ứng dụng như Maven, Gradle, … nhưng các bạn có đồng ý với Khanh là: sometime chúng ta cũng gặp lỗi này.
  • Bất kỳ một class hay method nào được định nghĩa với public hoặc protected access modifier trong class Java library mà nằm trong classpath của project của ứng dụng, chúng ta đều có thể access tới chúng. Không có cách nào để các Java library này che giấu chúng, chỉ để các class của các package nằm trong library truy cập được mà thôi.
  • Và còn nhiều vấn đề khác nữa …

Java Module System được giới thiệu để giải quyết những vấn đề trên:

  • Nó giúp chúng ta có thể quản lý được những module nào sẽ được sử dụng trong code của chúng ta, những module này sẽ được include trong quá trình chúng ta build module, bằng cách sử dụng module directive requires.
  • Chúng ta cũng có thể quyết định được ai, package nào được sử dụng code của mình, bằng cách sử dụng từ khóa module directive exports.

Trong bài viết này, mình sẽ giới thiệu với các bạn tổng quan về Java Module System với một ví dụ nhỏ về cách tạo mới một Java module các bạn nhé!


Tổng quan về Java Module

Đầu tiên, các bạn hãy cài đặt JDK từ Java 9 trở đi nhé.

Hãy mở command line rồi nhập dòng lệnh:

để xem tất cả những module được cung cấp mặc định bởi Java các bạn nhé!

Kết quả:

Ở đây, Java chia ra 2 loại modules là những standard modules và các non-standard modules. Standard modules là những module hiện thực Java SE  specification với tên module được bắt đầu với java.* còn những non-standard modules là những module nằm trong Java Development Kit được bắt đầu với jdk.*. Các bạn có thể thấy rõ điều này trong phần kết quả sau khi mình execute câu lệnh “java –list-modules” trên.

Mỗi module name sẽ bao gồm tên module và Java version cho chúng ta biết là module đó thuộc về Java version nào. Như các bạn thấy, mình đang sử dụng Java 15 và trong tên module, các bạn có thể thấy, chúng được kết thúc với @15.

Để xem thông tin về một module nào đó, các bạn có thể sử dụng câu lệnh sau:

Ví dụ, để xem thông tin của module java.logging, mình sẽ nhập câu lệnh sau:

Kết quả:

Giới thiệu về Java Module System

Thông tin của một module sẽ bắt đầu với tên module, và tiếp theo là những module directives mà mình có đề cập ở trên như exports, requires, provides, … như các bạn thấy trong hình trên. Tất cả những thông tin này sẽ được định nghĩa trong một tập tin tên là module-info.java các bạn nhé!

Mỗi module directive có tác dụng định nghĩa cách mà module sẽ được build và sử dụng, ví dụ như directive exports giúp chúng ta định nghĩa một package nào đó trong module này được access bởi ai hay directive provides có mục đích chỉ rõ module này đang implement một interface nào đó cho các class khác có thể sử dụng, … Chúng ta sẽ tìm hiểu rõ hơn về từng module directive sau các bạn nhé!

Trong output của ví dụ trên, các bạn có thể thấy module java.logging đang sử dụng một dependency là module java.base. Để xem tất cả các dependencies mà một module đang sử dụng, các bạn cũng có thể sử dụng câu lệnh với công cụ jdeps như sau:

Hoặc có thể viết tắt –module bằng -m như sau:

Ví dụ, để xem dependencies mà module java.logging đang sử dụng, mình sẽ chạy câu lệnh như sau:

Kết quả:

Giới thiệu về Java Module System


Ví dụ

Bây giờ, mình sẽ hướng dẫn cho các bạn cách tạo mới một module cơ bản các bạn nhé!

Đầu tiên, mình sẽ tạo mới một Java project để làm ví dụ.

Mình sẽ sử dụng Spring Tool Suite để làm điều này.

Trong quá trình tạo mới project, các bạn sẽ thấy Spring Tool Suite hỗ trợ chúng ta tạo mới tập tin module-info.java luôn.

Giới thiệu về Java Module System

và:

Giới thiệu về Java Module System

Như mình nói ở trên, đây là tập tin định nghĩa thông tin của một Java Module.

Kết quả:

Giới thiệu về Java Module System

Nội dung của tập tin module-info.java lúc này các bạn có thể thấy như sau:

Bây giờ mình sẽ thêm một class với phương thức sayHello(), return về chữ “Hello World” như sau:

Và export package của class này trong tập tin module-info.java để các module khác có thể sử dụng class Example như sau:

Bây giờ, mình sẽ tạo mới một Java module project khác:

Giới thiệu về Java Module System

Mình sẽ khai báo module huongdanjava.module.example.test sử dụng module huongdanjava.module.example bằng cách click chuột phải vào project huongdanjava.module.example.test chọn Build Path, xong chọn Configure Build Path… Trong cửa sổ Java Build Path, mình sẽ select module huongdanjava.module.example để sử dụng như sau:

Giới thiệu về Java Module System

Module huongdanjava.module.example.test này sẽ khai báo directive requires tới package huongdanjava.module.example trong tập tin module-info.java để sử dụng class Example của module huongdanjava.module.example:

và in ra dòng chữ “Hello World” khi chạy chương trình như sau:

Kết quả khi chạy chương trình này như sau:

Giới thiệu về Java Module System

Nếu bây giờ trong tập tin module-info.java của module huongdanjava.module.example, mình xoá dòng

các bạn sẽ thấy Spring Tool Suite báo lỗi class Application ngay như sau:

Giới thiệu về Java Module System

Đó là bởi vì lúc này, chúng ta không cho phép bất kỳ class nào trong các module khác sử dụng package huongdanjava.module.example của module huongdanjava.module.example nữa!

Chúng ta sẽ tìm hiểu thêm về các module directives trong Java Module trong bài viết sau của Hướng Dẫn Java các bạn nhé.

3 thoughts on “Giới thiệu về Java Module System

  1. Theo mình hiểu lỗi ở đây là do mình đã define module nhưng ko export bất kỳ một class nào thành ra JVM không thể access được class path đó.
    Nếu như mình không dùng module-info hay thậm chí comment cả file thì theo keyword access, code sẽ chạy bình thường.

    1. Theo keyword access là sao bạn nhỉ? Mình chưa hình dung được ý của bạn lắm.

      Bạn nói “JVM không thể access được class path đó”, Khanh thấy không đúng lắm! Ở đây, là chúng ta đang ngăn chặn class khác access vào class mà chúng ta đang muốn không cho bên thứ ba sử dụng mà thôi.

      1. Hi Khanh,
        mình comment phần này là để cho rõ việc compile bị lỗi inaccessible thôi vì mấu chốt chỗ limit module access này là do bản thân việc khai báo trong file module-info.java mà thôi. Nếu như không có file này thì việc access vẫn bình thường (keyword ở đây là public, 1 public class có thể được access từ bất cứ chỗ nào).
        Còn việc java 9 support modular là 1 việc rất tốt, đặc biệt khi mình cần viết library, đúng như Khanh có bảo

Add Comment