Java多线程编程核心技术(第3版)
上QQ阅读APP看书,第一时间看更新

2.2.8 将任意对象作为锁

多个线程调用同一个对象中的不同名称的synchronized同步方法或synchronized(this)同步代码块时,是按顺序执行,也就是同步的。

如果在一个类中同时存在synchronized同步方法和synchronized(this)同步代码块,对其他synchronized同步方法或synchronized(this)同步代码块调用会呈同步的效果,执行特性如下。

1)同一时间只有一个线程可以执行synchronized同步方法中的代码;

2)同一时间只有一个线程可以执行synchronized(this)同步代码块中的代码。

以上两种情况已经在前面的章节中进行了测试。

其实,Java中还支持将任意对象作为锁,来实现同步的功能,这个任意对象大多数是实例变量及方法的参数,格式为synchronized(非this对象)。

synchronized(非this对象x)同步代码块的执行特性如下:在多个线程争抢相同的非this对象x的锁时,同一时间只有一个线程可以执行synchronized(非this对象x)同步代码块中的代码。

创建项目验证此结论。创建测试项目synBlockString,类文件Service.java代码如下:


package service;

public class Service {

private String usernameParam;
private String passwordParam;

private String anyString = new String();

public void setUsernamePassword(String username, String password) {
    try {

        synchronized (anyString) {
            System.out.println("线程名称为:" + Thread.currentThread().getName()
                + "在" + System.currentTimeMillis() + "进入同步块");
            usernameParam = username;
            Thread.sleep(3000); // 模拟处理数据需要的耗时
            passwordParam = password;
            System.out.println("线程名称为:" + Thread.currentThread().getName()
                + "在" + System.currentTimeMillis() + "离开同步块");
        }
    } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

}

自定义线程ThreadA.java代码如下:


package extthread;

import service.Service;

public class ThreadA extends Thread {
private Service service;

public ThreadA(Service service) {
    super();
    this.service = service;
}

@Override
public void run() {
    service.setUsernamePassword("a", "aa");

}

}

自定义线程ThreadB.java代码如下:


package extthread;

import service.Service;

public class ThreadB extends Thread {

private Service service;

public ThreadB(Service service) {
    super();
    this.service = service;
}

@Override
public void run() {
    service.setUsernamePassword("b", "bb");

}

}

运行类Run.java代码如下:


package test;

import service.Service;
import extthread.ThreadA;
import extthread.ThreadB;

public class Run {

public static void main(String[] args) {
    Service service = new Service();

    ThreadA a = new ThreadA(service);
    a.setName("A");
    a.start();

    ThreadB b = new ThreadB(service);
    b.setName("B");
    b.start();

}

}

锁非this对象具有一定的优点,就是如果一个类中有很多synchronized方法,这时虽然能实现同步,但影响运行效率,如果使用同步代码块锁非this对象,则synchronized(非this)代码块中的程序与同步方法是异步的,因为是两把锁,不与其他锁this同步方法争抢this锁,可大大提高运行效率。

图2-29 同步效果

程序运行结果如图2-29所示。