1.方法聲明時使用,放在範圍操做符(public等)後,其返回類型聲明(void等)以前。即一次只能有一個線程進入該方法,其餘線程要想在此時調用該方法,只能排隊等候,當前線程(就是在synchronized方法內部的線程)執行完該方法後,別的線程才能進入。
例如:
public synchronized void synMethod() {
//方法體
}
2.對某一代碼塊使用,synchronized後跟括號,括號裏是變量,這樣,一次只有一個線程進入該代碼塊。例如:
public int synMethod(Object a1){
synchronized(Object) {
//一次只能有一個線程進入
}
}
3.synchronized後面括號裏是一對象,此時,線程得到的是對象鎖。例如:
public class MyThread implements Runnable {
public static void main(String args[]) {
MyThread mt = new MyThread();
Thread t1 = new Thread(mt, "t1");
Thread t2 = new Thread(mt, "t2");
Thread t3 = new Thread(mt, "t3");
Thread t4 = new Thread(mt, "t4");
Thread t5 = new Thread(mt, "t5");
Thread t6 = new Thread(mt, "t6");
t1.start();
t2.start();
t3.start();
t4.start();
t5.start();
t6.start();
}
public void run() {
synchronized (this) {
System.out.println(Thread.currentThread().getName());
}
}
}
對於3,若是線程進入,則獲得對象鎖,那麼別的線程在該類全部對象上的任何操做都不能進行。在對象級使用鎖一般是一種比較粗糙的方法。爲何要將整個對象都上鎖,而不容許其餘線程短暫地使用對象中其餘同步方法來訪問共享資源?若是一個對象擁有多個資源,就不須要只爲了讓一個線程使用其中一部分資源,就將全部線程都鎖在外面。因爲每一個對象都有鎖,能夠以下所示使用虛擬對象來上鎖:
class FineGrainLock {
MyMemberClass x, y;
Object xlock = new Object(), ylock = new Object();
public void foo() {
synchronized(xlock) {
//access x here
}
//do something here - but don‘t use shared resources
synchronized(ylock) {
//access y here
}
}
public void bar() {
synchronized(this) {
//access both x and y here
}
//do something here - but don‘t use shared resources
}
}
4.synchronized後面括號裏是類。例如:
class ArrayWithLockOrder{
private static long num_locks = 0;
private long lock_order;
private int[] arr;
public ArrayWithLockOrder(int[] a)
{
arr = a;
synchronized(ArrayWithLockOrder.class) {//-----------------------------------------這裏
num_locks++; // 鎖數加 1。
lock_order = num_locks; // 爲此對象實例設置惟一的 lock_order。
}
}
public long lockOrder()
{
return lock_order;
}
public int[] array()
{
return arr;
}
}
class SomeClass implements Runnable
{
public int sumArrays(ArrayWithLockOrder a1,
ArrayWithLockOrder a2)
{
int value = 0;
ArrayWithLockOrder first = a1; // 保留數組引用的一個
ArrayWithLockOrder last = a2; // 本地副本。
int size = a1.array().length;
if (size == a2.array().length)
{
if (a1.lockOrder() > a2.lockOrder()) // 肯定並設置對象的鎖定
{ // 順序。
first = a2;
last = a1;
}
synchronized(first) { // 按正確的順序鎖定對象。
synchronized(last) {
int[] arr1 = a1.array();
int[] arr2 = a2.array();
for (int i=0; i value += arr1[i] + arr2[i];
}
}
}
return value;
}
public void run() {
//...
}
}
對於4,若是線程進入,則線程在該類中全部操做不能進行,包括靜態變量和靜態方法,實際上,對於含有靜態方法和靜態變量的代碼塊的同步,咱們一般用4來加鎖。
以上4種之間的關係:
鎖是和對象相關聯的,每一個對象有一把鎖,爲了執行synchronized語句,線程必須可以得到synchronized語句中表達式指定的對象的鎖,一個對象只有一把鎖,被一個線程得到以後它就再也不擁有這把鎖,線程在執行完synchronized語句後,將得到鎖交還給對象。
在方法前面加上synchronized修飾符便可以將一個方法聲明爲同步化方法。同步化方法在執行以前得到一個鎖。若是這是一個類方法,那麼得到的鎖是和聲明方法的類相關的Class類對象的鎖。若是這是一個實例方法,那麼此鎖是this對象的鎖。synchronzied塊後面跟類的具體詳細例子:
public class DB2_JDBCFactory {
private static DB2_JDBCFactory instance = null;
public static final ThreadLocal threadLocal = new ThreadLocal();
private DB2_JDBCFactory() {
}
public static DB2_JDBCFactory getInstance() {
if(instance == null) {
synchronized(DB2_JDBCFactory.class) { //synchronized後面跟一個類
instance = new DB2_JDBCFactory();
}
}
return instance;
}
public Connection getConnection_JNDI_localhost(){
Connection c = (Connection) threadLocal.get();
try {
if (c == null || c.isClosed()) {
InitialContext ctx = new InitialContext();
DataSource ds = (DataSource)ctx.lookup("java:comp/env/jdbc/localhost");
c = ds.getConnection();
threadLocal.set(c);
}
} catch (Exception ex) {
System.err.println("getConnection_JNDI Initial failed. " + ex);
return null;
}
return c;
}
} 外面的對象訪問這個類的 須要經過調用它的getInstance()
|