Thứ Năm, 14 tháng 3, 2013

Tìm hiểu Thread và Synchronized trong JAVA

Bây giờ chúng ta sẽ tìm hiểu đồng bộ hóa trong Java với từ khóa synchronized.

Synchronized methods

Giả sử chồng và vợ có 2 cái thẻ ATM. Hai người cùng rút một số tiền nhất định vào cùng một thời điểm, sau khi rút tiền máy ATM hiển thị kết quả ngược lại so với mong muốn.



Như vậy số tiền kia bay đi đâu rồi! Ai sẽ chịu trách nhiệm đây? (Chắc là ngân hàng) :D

Minh họa cho sự cố này:



Hình trên tôi cố tình chạy hai tiến trình vào cùng một thời điểm, cùng với việc trừ 1000$ salary trong tài khoản. Tôi sử dụng thêm phương thức getName() để biết được chính xác tên của Thread hiện tại. Và được kết quả:



Tài khoản salary của gia đình là $2000, mà vợ rút $1000 thì phải còn $1000 nhưng còn lại $0. Nguyên nhân do Thread của vợ đang rút $1000 thì Thread của chồng cũng rút $1000 vào cùng thời điểm đó dẫn đến bất đồng bộ.

Và để giả quyết vấn đề trên ta chỉ cần thêm từ khóa synchronized vào trước kiểu giá trị trả về của phương thức là đủ. Minh họa hình dưới:



Mục đích của chính của synchronized là khi đã có một tiến trình A sử dụng tài nguyên này rồi thì những tiến trình khác phải chờ đến khi tiến trình A kết thúc thì mới được phép sử dụng. Kết quả:



Như vậy tiến trình của người chồng phải chờ tiến trình người vợ kết thúc mới được phép sử dụng tài nguyên, đồng nghĩa với việc thời gian xử lý sẽ tăng lên gấp đôi (do phải chờ tiến trình trước kết thúc) nhưng điều này đảm bảo rằng dữ liệu luôn luôn được toàn vẹn.
Synchronized block

Thay vì sử dụng synchronized một phương thức như ở trên thì chúng ta cũng có thể synchronized một đoạn nhỏ trong phương thức của một đối tượng bằng cách sử dụng synchronized block, minh họa hình dưới:



Ở hình trên tôi đã đánh dấu khối được synchronized, các bạn để ý bên cạnh từ khóa synchronized sẽ có cấu trúc "<TênClass.Class>" như trong trường hợp này sẽ là mThread.Class hoặc đơn giản hơn bạn thay thế bằng từ khóa this cũng được, mục đích của từ khóa đó dùng để chỉ rõ đối tượng đối tượng hiện tại sẽ được synchronized và tôi vừa thêm một dòng thông báo "đăng xuất [X]" để xem chuyện gì sẽ xảy ra, kết quả:



Như vậy tiến trình người vợ sau khi thực hiện xong khối lệnh được synchronized thì sẽ nhường quyền cho tiến trình người chồng thực hiện mà không hề ép tiến trình người chồng chờ đợi thêm.

Như vậy với cách trên chúng ta cũng có thể synchronized một đối tượng khác mà thread hiện tại đang nắm giữ với cú pháp như sau:

Object obj = new Object();
  ...
synchronized (obj) {  // in a method
  ... // nội dung
}

Đến đây xem như kết thúc phần đồng bộ hóa trong Java.

0 nhận xét:

Đăng nhận xét