Bài giảng Lập trình Java - Chương 6: Lập trình đa luồng - Nguyễn Đức Hiển

ppt 31 trang phuongnguyen 4790
Bạn đang xem 20 trang mẫu của tài liệu "Bài giảng Lập trình Java - Chương 6: Lập trình đa luồng - Nguyễn Đức Hiển", để tải tài liệu gốc về máy bạn click vào nút DOWNLOAD ở trên

Tài liệu đính kèm:

  • pptbai_giang_lap_trinh_java_chuong_6_lap_trinh_da_luong_nguyen.ppt

Nội dung text: Bài giảng Lập trình Java - Chương 6: Lập trình đa luồng - Nguyễn Đức Hiển

  1. Java Object-Oriented Programming ❑ Giảng viên : Nguyễn Đức Hiển ❑ Email : ndhien@udn.vn ❑ Website : ❑ Thời lượng ❑ Lý thuyết : 2 tín chỉ (30 tiết) ❑ Thực hành + thảo luận : 1 tín chỉ Nguyễn Đức Hiển – Bài giảng Lập trình Java 1
  2. Chương 6 Lập trình đa luồng (Multi-Thread Programming) Nguyễn Đức Hiển – Bài giảng Lập trình Java 2
  3. Nội dung ❑ Giới thiệu về luồng (thread) ❑ Cách tạo luồng trong Java ❑ Đồng bộ hóa luồng Nguyễn Đức Hiển – Bài giảng Lập trình Java 3
  4. Giới thiệu ❑ Một luồng (thread) là gì? ❑ Một “dòng điều khiển " trong chương trình ❑ Các chương trình thường chỉ có một dòng điều khiển. ❑ Với các luồng, bạn có thể có nhiều dòng điều khiển thực hiện cùng lúc trong chương trình ❑ Ví dụ: Xem xét bộ xử lý từ cơ bản ❑ Bạn soạn thảo văn bản và nhấn nút lưu trữ ❑ Nó có thể mất một lượng thời gian đáng kể để lưu dữ liệu mới trên đĩa, tất cả điều này được thực hiện với một luồng tách biệt dưới nền (background) ❑ Không có các luồng, ứng dụng sẽ bị treo trong khi bạn đang lưu file và không đáp ứng cho đến khi thao tác lưu hoàn thành Nguyễn Đức Hiển – Bài giảng Lập trình Java 4
  5. Luồng Java ❑ Khi chương trình Java thực thi hàm main() tức là tạo ra một luồng (luồng main). Trong luồng main: ❑ Có thể tạo các luồng con. ❑ Chương trình phải đảm bảo main là luồng kết thúc cuối cùng. ❑ Khi luồng main ngừng thực thi, chương trình sẽ kết thúc ❑ Luồng có thể được tạo ra bằng 2 cách: ❑ Tạo lớp dẫn xuất từ lớp Thread ❑ Tạo lớp hiện thực giao tiếp Runnable. Nguyễn Đức Hiển – Bài giảng Lập trình Java 5
  6. Tạo luồng ❑ Trong Java có sẵn lớp Thread. Để tạo một luồng mới ta có thể tạo một lớp thừa kế (extends) lớp Thread và ghi đè phương thức run() ❑ Ví dụ: Nguyễn Đức Hiển – Bài giảng Lập trình Java 6
  7. Chạy luồng ❑ Tạo ra một thể hiện của lớp Thread (hoặc dẫn xuất của nó) và gọi phương thức start() ❑ Khi gọi myThread.start() một luồng mới tạo ra và chạy phương thức run() của myThread. ❑ myThread.start() trả về gần như ngay lập tức. Nguyễn Đức Hiển – Bài giảng Lập trình Java 7
  8. Bài tập ❑ Bài 1. Tạo 2 luồng: luồng 1 hiển thị các số chẳn, luồng 2 hiển thị các số lẻ. ❑ Bài 2. Tạo 2 luồng: luồng 1 hiển thị các số nguyên tố, luồng 2 hiển thị các số hoàn thiện. Nguyễn Đức Hiển – Bài giảng Lập trình Java 8
  9. Giao tiếp Runnable ❑ Ngoài tạo luồng bằng cách thừa kế từ lớp Thread, cũng có một cách khác để tạo luồng trong Java. ❑ Bạn có thể tạo luồng bằng cách tạo lớp mới hiện thực giao tiếp Runnable và định nghĩa phương thức: ❑ public abstract void run() ❑ Điều này đặc biệt hữu ích nếu bạn muốn để tạo ra một đối tượng Thread nhưng muốn sử dụng một lớp cơ sở khác Thread. Nguyễn Đức Hiển – Bài giảng Lập trình Java 9
  10. Ví dụ Nguyễn Đức Hiển – Bài giảng Lập trình Java 10
  11. Giao tiếp Runnable ❑ Để tạo ra một luồng mới từ một đối tượng hiện thực giao tiếp Runnable, bạn phải khởi tạo một đối tượng Thread mới với đối tượng Runnable như đích của nó ❑ Khi gọi start() trên đối tượng luồng sẽ tạo ra một luồng mới và phương thức run() của đối tượng Runnable sẽ được thực hiện. Nguyễn Đức Hiển – Bài giảng Lập trình Java 11
  12. Vòng đời của một luồng Nguyễn Đức Hiển – Bài giảng Lập trình Java 12
  13. Điều phối luồng ❑ JVM chọn luồng để chạy theo “giải thuật quyền ưu tiên cố định” ❑ Mọi luồng có một quyền ưu tiên trong khoảng phạm vi Thread.MIN_PRIORITY và Thread.MAX_PRIORITY. ❑ Theo mặc định một luồng được khởi tạo với cùng quyền ưu tiên với luồng tạo ra nó. ❑ Bạn có thể thay đổi quyền ưu tiên sử dụng phương thức setPriority() của lớp Thread. Nguyễn Đức Hiển – Bài giảng Lập trình Java 13
  14. Điều phối luồng ❑ Các luồng với quyền ưu tiên cao có một cơ hội nhận thời gian sử dụng CPU để hoàn thành trước các luồng với quyền ưu tiên thấp hơn. ❑ JVM sử dụng giải thuật không độc quyền. Vì thế, nếu một luồng quyền ưu tiên thấp đang được chạy, luồng quyền có quyền ưu tiên cao hơn có thể giành quyền sử dụng CPU của nó. ❑ Nếu các luồng có cùng quyền ưu tiên đang chờ đợi để thực hiện, một luồng tùy ý sẽ được lựa chọn. Nguyễn Đức Hiển – Bài giảng Lập trình Java 14
  15. Điều phối luồng ❑ Khi một luồng giành quyền sử dụng CPU, nó sẽ thực hiện cho đến khi một sự kiện sau xuất hiện: ❑ Phương thức run() kết thúc ❑ Một luồng quyền ưu tiên cao hơn ❑ Nó gọi phương thức sleep() hay yield() – nhượng bộ ❑ Khi gọi yield(), luồng đưa cho các luồng khác với cùng quyền ưu tiên cơ hội sử dụng CPU. Nếu không có luồng nào khác cùng quyền ưu tiên tồn tại, luồng tiếp tục thực hiện ❑ Khi gọi sleep(), luồng ngủ trong một số mili-giây xác định, trong thời gian đó bất kỳ luồng nào khác có thể sử dụng CPU. Nguyễn Đức Hiển – Bài giảng Lập trình Java 15
  16. Một số phương thức khác ❑ Phương thức join() ❑ Khi một luồng (A) gọi phương thức join() của một luồng nào đó (B), luồng hiện hành (A) sẽ bị khóa chờ (blocked) cho đến khi luồng đó kết thúc (B). ❑ Ví dụ: Nguyễn Đức Hiển – Bài giảng Lập trình Java 16
  17. Một số phương thức khác ❑ Phương thức interrupt() ❑ Đặt trạng thái luồng ngắt (không ngừng hẳn luồng). ❑ Phương thức interrupted() ❑ Phương thức này trả lại một giá trị boolean cho biết trạng thái ngắt quãng của luồng hiện thời. ❑ Phương thức này cũng đặt lại trạng thái của luồng hiện thời thành không ngắt. ❑ Kết hợp sử dụng hai phương thức này có thể được dùng làm phương pháp yêu cầu một luồng nhượng bộ, ngủ hoặc kết thúc chính nó. Nguyễn Đức Hiển – Bài giảng Lập trình Java 17
  18. Sự đồng bộ hóa ❑ Trường hợp nhiều luồng cùng truy cập trên các tài nguyên đồng thời. ❑ Đọc/ghi trên cùng một file ❑ Sửa đổi cùng một đối tượng/biến ❑ ❑ Trong những trường hợp này, bạn phải cẩn thận phối hợp các thao tác này như thế nào để các tài nguyên kết thúc trong một trạng thái an toàn. ❑ Java có sẵn cơ chế cho sự phối hợp này → đồng bộ hóa luồng. Nguyễn Đức Hiển – Bài giảng Lập trình Java 18
  19. Bài toán Producer/Consumer ❑ Có hai luồng, một sản xuất và một tiêu thụ cả hai truy cập cùng môt đối tượng CubbyHole (chổ ấm áp). ❑ CubbyHole là một đối tượng đơn giản lưu giữ một giá trị đơn như nội dung của nó. ❑ Luồng sản xuất phát sinh ngẫu nhiên các giá trị và cất giữ chúng trong đối tượng CubbyHole ❑ Luồng tiêu thụ lấy các giá trị này khi chúng được sinh ra bởi luồng sản xuất. Nguyễn Đức Hiển – Bài giảng Lập trình Java 19
  20. Lớp CubbyHole Nguyễn Đức Hiển – Bài giảng Lập trình Java 20
  21. Luồng sản xuất (Producer) Nguyễn Đức Hiển – Bài giảng Lập trình Java 21
  22. Luồng tiêu thụ (Customer) Nguyễn Đức Hiển – Bài giảng Lập trình Java 22
  23. Bài toán Producer/Customer Nguyễn Đức Hiển – Bài giảng Lập trình Java 23
  24. Các vấn đề Producer/Customer ❑ Khi luồng sản xuất sinh ra một giá trị, nó cất giữ nó vào CubbyHole và sau đó luồng tiêu thụ chỉ phải lấy nó một và chỉ một lần. ❑ Phụ thuộc vào các luồng được điều phối như thế nào ❑ Chẳng hạn luồng sản xuất có thể sinh ra hai giá trị trước khi tiêu thụ có thể lấy một. ❑ Luồng tiêu thụ có thể lấy cùng giá trị hai lần trước đây sản xuất có được sinh ra giá trị tiếp theo. ❑ Nếu luồng sản xuất và tiêu thụ truy cập CubbyHole cùng lúc, chúng đã có thể sinh ra một trạng thái mâu thuẫn hay thiếu một giá trị được sản xuất. Nguyễn Đức Hiển – Bài giảng Lập trình Java 24
  25. Giải pháp đồng bộ hóa ❑ Xây dựng đối tượng với các phương thức đồng bộ hóa với từ khóa synchronized ❑ Ví dụ: Nguyễn Đức Hiển – Bài giảng Lập trình Java 25
  26. Giải pháp đồng bộ hóa ❑ Khi một luồng gọi thực hiện một phương thức đồng bộ hóa của một đối tượng, nó sẽ khóa đối tượng đó. ❑ Khi đó, các phương thức đồng bộ hóa được gọi bởi luồng khác trên đối tượng đó sẽ không được thực hiện cho đến khi đối tượng được mở khóa. Nguyễn Đức Hiển – Bài giảng Lập trình Java 26
  27. Giải pháp đồng bộ hóa ❑ Các phương thức đồng bộ hóa ngăn chặn luồng sản xuất và luồng tiêu thụ sửa đổi CubbyHole cùng lúc. ❑ Tuy nhiên, chúng ta vẫn còn cần phối hợp các luồng sản xuất và tiêu thụ sao cho chúng không sinh ra hay tiêu thụ không đúng thứ tự. ❑ Các phương thức wait() và notifyAll() được sử dụng để thóa khóa trên một đối tượng và thông báo các luồng đang đợi các chúng có thể có lại điều khiển. Nguyễn Đức Hiển – Bài giảng Lập trình Java 27
  28. Giải pháp đồng bộ hóa Nguyễn Đức Hiển – Bài giảng Lập trình Java 28
  29. Giải pháp đồng bộ hóa Nguyễn Đức Hiển – Bài giảng Lập trình Java 29
  30. Giải pháp đồng bộ hóa Nguyễn Đức Hiển – Bài giảng Lập trình Java 30
  31. Thanks for listenning!!! Nguyễn Đức Hiển – Bài giảng Lập trình Java 31