Java多线程-synchronized同步方法
1、synchronized 方法与锁对象
线程锁的是对象。
1)A线程先持有 object 对象的 Lock 锁, B线程可以以异步的方式调用 object 对象中的非 synchronized 类型的方法
2)A线程先持有 object 对象的 Lock 锁, B线程如果在这时调用 object 对象中的 synchronized 类型的方法,则需要等待,也就是同步。
2、脏读(DirtyRead)
示例:
public class DirtyReadTest {public static void main(String[] args) {
try {
PublicVar publicVar = new PublicVar();
ThreadA thread = new ThreadA(publicVar);
thread.start();
Thread.sleep(200);
publicVar.getValue();
} catch (Exception e) {
e.printStackTrace();
}
}
static class ThreadA extends Thread {
private PublicVar publicVar;
public ThreadA(PublicVar publicVar) {
this.publicVar = publicVar;
}
@Override
public void run() {
super.run();
publicVar.setValue("B", "BB");
}
}
static class PublicVar {
public String username = "A";
public String password = "AA";
synchronized public void setValue(String username, String password) {
try {
this.username = username;
Thread.sleep(1000);
this.password = password;
System.out.println("setValue method thread name = " + Thread.currentThread().getName() + "\tusername = " + username + "\tpassword = " + password);
} catch (Exception e) {
e.printStackTrace();
}
}
// 因为 getValue 方法是非 synchronized 方法,所以造成了脏读
public void getValue() {
System.out.println("getValue method thread name = " + Thread.currentThread().getName() + "\tusername = " + username + "\tpassword = " + password);
}
}
}
结果如图:
3、synchronized 锁重入
"可重入锁"的概念是:
自己可以再次活得自己的内部锁。比如有1条线程获得了某个对象的锁,此时这个对象锁还没有释放,当其再次想要获取这个对象的锁的时候还是可以获取的,如果不可以锁重入的话,就会造成死锁。
可重入锁也支持在父子类继承的环境中:在继承中,子类是完全可以通过“可重入锁”调用父类的同步方法的。
4、出现异常,锁自动释放
线程出现异常,会释放当前线程的锁
5、synchronizd 不具有继承性
示例:
public class SyncExtend {public static void main(String[] args) {
Sub sub = new Sub();
MyThread t1 = new MyThread(sub);
MyThread t2 = new MyThread(sub);
t1.start();
t2.start();
}
static class MyThread extends Thread {
Sub sub;
public MyThread(Sub sub) {
this.sub = sub;
}
@Override
public void run() {
try {
sub.opera();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
static class Main {
public int i = 10;
synchronized public void opera() throws InterruptedException {
try {
System.out.println("Thread now:" + Thread.currentThread().getName());
i--;
Thread.sleep(3000);
System.out.println("main print i = " + i);
} catch (Exception e) {
e.printStackTrace();
}
}
}
static class Sub extends Main {
@Override
public void opera() throws InterruptedException {
System.out.println("in sub,Thread name:" + Thread.currentThread().getName());
Thread.sleep(100);
super.opera();
}
}
}
运行结果:
以上是 Java多线程-synchronized同步方法 的全部内容, 来源链接: utcz.com/z/393491.html