同步程式碼塊

基本語句

synchronized (任意物件) {

操作共享程式碼

}

程式碼示例


public class SellTicket implements Runnable {
private int tickets = 100;
private Object object = new Object(); @Override
public void run() {
while (true) {
synchronized (object) {
if (tickets > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在出售第" + tickets + "張票");
tickets--;
}
}
}
} public static void main(String[] args) { SellTicket sellTicket = new SellTicket(); Thread thread1 = new Thread(sellTicket, "視窗1");
Thread thread2 = new Thread(sellTicket, "視窗2");
Thread thread3 = new Thread(sellTicket, "視窗3");
thread1.start();
thread2.start();
thread3.start();
}
}

優缺點:

  • 解決了多執行緒的資料安全問題
  • 多執行緒時,每個執行緒都會判斷同步上的鎖,耗費資源,降低了程式的執行效率

同步方法

同步方法:將synchronized關鍵字加到方法上

  • 格式: 修飾符 synchronized 返回值型別 方法名(){ }
  • 同步方法的鎖物件是this

同步靜態方法,就是把synchronized關鍵字加到靜態方法上

  • 格式: 修飾符 static synchronized 返回值型別 方法名(){ }
  • 同步靜態方法的鎖物件是 類名.class

程式碼示例

public class SellTicket implements Runnable {
// private int tickets = 100;
private static int tickets = 100;
private Object object = new Object();
private int x = 0; @Override
public void run() {
while (true) {
if (x % 2 == 0) {
// synchronized (object) {
// synchronized (this) {
synchronized (SellTicket.class) {
if (tickets > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在出售第" + tickets + "張票");
tickets--;
}
}
} else {
// synchronized (object) {
// if (tickets > 0) {
// try {
// Thread.sleep(100);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// System.out.println(Thread.currentThread().getName() + "正在出售第" + tickets + "張票");
// tickets--;
// }
// }
sellTicket();
}
x++;
}
} // private void sellTicket(){
// synchronized (object) {
// if (tickets > 0) {
// try {
// Thread.sleep(100);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// System.out.println(Thread.currentThread().getName() + "正在出售第" + tickets + "張票");
// tickets--;
// }
// }
// } // private synchronized void sellTicket(){
// if (tickets > 0) {
// try {
// Thread.sleep(100);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// System.out.println(Thread.currentThread().getName() + "正在出售第" + tickets + "張票");
// tickets--;
// }
private static synchronized void sellTicket(){
if (tickets > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在出售第" + tickets + "張票");
tickets--;
} } public static void main(String[] args) { SellTicket sellTicket = new SellTicket(); Thread thread1 = new Thread(sellTicket, "視窗1");
Thread thread2 = new Thread(sellTicket, "視窗2");
Thread thread3 = new Thread(sellTicket, "視窗3");
thread1.start();
thread2.start();
thread3.start();
}
}

lock鎖

lock實現提供比使用synchronized方法和語句可獲得更廣泛的操作

  • void lock()獲得鎖
  • void unlock()釋放

lock是介面不能直接例項化,採用實現類例項化ReentrantLock

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class SellTicket implements Runnable {
private int tickets = 100;
private Object object = new Object();
private Lock lock = new ReentrantLock(); @Override
public void run() {
while (true) {
try {
lock.lock();
if (tickets > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在出售第" + tickets + "張票");
tickets--;
}
} finally {
lock.unlock(); }
} } public static void main(String[] args) { SellTicket sellTicket = new SellTicket(); Thread thread1 = new Thread(sellTicket, "視窗1");
Thread thread2 = new Thread(sellTicket, "視窗2");
Thread thread3 = new Thread(sellTicket, "視窗3");
thread1.start();
thread2.start();
thread3.start();
}
}