java线程同步操作实例详解

网友投稿 303 2023-01-22


java线程同步操作实例详解

本文实例讲述了java线程同步操作。分享给大家供大家参考,具体如下:

java线程同步

public class Hello {

public static void main(String[] args) {

MyRun myRun0 = new MyRun();

new Thread(myRun0, "Thread0").start();

new Thread(myRun0, "Thread1").start();

new Thread(myRun0, "Thread2").start();

}

}

class MyRun implements Runnable {

private int k = 0;

@Override

public void run() {

for (http://int i = 0; i < 3; i++) {

System.out.println(Thread.currentThread().getName() + "**********" + i);

k++;

if (k <= 3) {

if ("Thread0".equals(Thread.currentThread().getName())) {

try {

Thread.sleep(100);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

System.out.println(Thread.currentThread().getName() + "," + k);

}

}

}

}

输出结果

Thread0**********0

Thread1**********0

Thread2**********0

Thread1,2

Thread2,3

Thread1**********1

Thread2**********1

Thread2**********2

Thread1**********2

Thread0,7

Thread0**********1

Thread0**********2

说明多线程在某些场景是存在问题的,有时候需要线程同步。

同步 synchronized

同步代码块,synchronized(obj){},obj是一个对象,在这里就相当于一把锁,表示一旦有进程抢到了这把锁的钥匙(就是进入了代码块),其他进程将无法进入该锁的代码块(当前代码块其他进程一定是进不来了,其他地方的代码块如果也是用了这把锁,同样进不去),只有代码块执行完,释放锁后,所有进程再重新抢钥匙。

注意,上同一把锁的代码块都会被锁住,这些代码块可能写在不同方法不同位置上。

被同步代码块包住的代码多个线程只能顺次进入。

synchronized (this) {

k++;

if (k <= 3) {

if ("Thread0".equals(Thread.currentThread().getName())) {

try {

Thread.sleep(100);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

System.out.println(Thread.currentThread().getName() + "," + k);

}

}

this表示当前对象,这里考虑的只是运行这个方法,不涉及其它类也不涉及这个类的其它地方需要同步问题,所以用this也是可以的。k增加和输出一个流程内只能有一个线程在访问,所以可以得到想要的输出结果

输出结果

Thread0**********0

Thread1**********0

Thread2**********0

Thread0,1

Thread0**********1

Thread2,2

Thread2**********1

Thread1,3

Thread1********CZndftyJi**1

Thread0**********2

Thread2**********2

Thread1**********2

对方法进行同步,如果存在多线程,每个线程顺次访问该方法

注意,如果一个类里面存在多个同步方法,那么这些同步方法的锁是一个,都是当前对象,所以不同线程想同时访问同一对象的不同方法也是不行的,因为这些方法都上了同一把锁,但是钥匙只有一把,只能一个线程持有。

@Override

public synchronized void run() {

for (int i = 0; i < 3; i++) {

System.out.println(Thread.currentThread().getName() + "**********" + i);

k++;

if (k <= 3) {

if ("Thread0".equals(Thread.currentThread().getName())) {

try {

Thread.sleep(100);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

System.out.println(Thread.currentThread().getName() + "," + k);

}

}

}

输出结果

Thread0**********0

Thread0,1

Thread0**********1

Thread0,2

Thread0**********2

Thread0,3

Thread2**********0

Thread2**********1

Thread2**********2

Thread1**********0

Thread1**********1

Thread1**********2

死锁

public class Hello {

public static void main(String[] args) {

A a = new A();

B b = new B();

new Thread(new MyRun(a,b)).start();

new Thread(new MyRun1(a,b)).start();

}

}

class MyRun implements Runnable{

private A a;

private B b;

public MyRun(A a, B b) {

this.a = a;

this.b = b;

}

@Override

public void run(){

a.say(b);

}

}

class MyRun1 implements Runnable {

private A a;

private B b;

public MyRun1(A a, B b) {

this.a = a;

this.b = b;

}

@Override

public void run() {

b.say(a);

}

}

class A{

public synchronized void say(B b){

System.out.println("A要知道B的信息");

b.info();

}

public synchronized void info(){

System.out.println("这是A");

}

}

class B{

public synchronized void say(A a){

System.out.println("B要知道A的信息");

a.info();

}

public synchronized void info(){

System.out.println("这是B");

}

}

如果两个线程同时进入了两个say方法,就是出现死锁。

关键点在于一个对象的多个同步方法具有相同的锁,都是当前对象。也就是x线程在访问a对象的say方法过程中,y线程是无法访问a对象的info方法的,因为开锁的钥匙已经被x线程抢占了。

上面的程序,如果线程x,y同时进入了两个say方法,a对象同步方法的锁被线程x抢占,b对象同步方法的锁被线程y抢占,此时线程x无法访问b对象的同步方法,线程y无法访问a对象的同步方法。代码中恰好想要访问,所以就出现死锁了。

更多java相关内容感兴趣的读者可查看本站专题:《Java进程与线程操作技巧总结》、《Java数据结构与算法教程》、《Java操作DOM节点技巧总结》、《Java文件与目录操作技巧汇总》和《Java缓存操作技巧汇总》

希望本文所述对大家jaCZndftyJiva程序设计有所帮助。


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:app接口管理工具有哪些(app的api接口)
下一篇:SpringBoot配置文件的加载位置实例详解
相关文章

 发表评论

暂时没有评论,来抢沙发吧~