Java中synchronized关键字实现线程同步互斥(2)

下面说一下这种同步机制的实现。java为每个类对象分配一个独一无二的对象锁,每个线程要访问这个对象的成员就需要获得这个锁(当然这里是指需要同步的线程)。因为一个对象只有一个对象锁,所以一个线程在拿到锁之后,另一个访问相同对象的线程必须等待前者执行完成后释放掉对象锁,以此实现互斥。也就是说synchronized是针对对象的,不是针对类的。看看上面的图形。ObjectA和ObjectB是同一个类的不同实例,所以ThreadAn和ThreadBn没有关系。只有ThreadA1234同步,ThreadB1234亦是如此。并且对于非synchronized的方法和代码块,并没有影响,没有互斥

那有没有实现类级别上的同步呢?有的。在 Java 中,不光是类实例,每一个类也对应一把锁,这样我们也可将类的静态成员函数声明为 synchronized ,以控制其对类的静态成员变量的访问。  使用同步代码块是synchronized(classname.class){}

最后几点总结:

1)是某个对象实例内,synchronized aMethod(){}可以防止多个线程同时访问这个对象的synchronized方法(如果一个对象有多个synchronized方法,只要一个线程访问了其中的一个synchronized方法,其它线程不能同时访问这个对象中任何一个synchronized方法)。这时,不同的对象实例的synchronized方法是不相干扰的。也就是说,其它线程照样可以同时访问相同类的另一个对象实例中的synchronized方法;
2)是某个类的范围,synchronized static aStaticMethod{}防止多个线程同时访问这个类中的synchronized static 方法。它可以对类的所有对象实例起作用。
2、除了方法前用synchronized关键字,synchronized关键字还可以用于方法中的某个区块中,表示只对这个区块的资源实行互斥访问。用法是: synchronized(this){/*区块*/},它的作用域是当前对象;
3、synchronized关键字是不能继承的,也就是说,基类的方法synchronized f(){} 在继承类中并不自动是synchronized f(){},而是变成了f(){}。继承类需要你显式的指定它的某个方法为synchronized方法;


对synchronized(this)的一些理解

一、当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执      行完这个代码块以后才能执行该代码块。 

二、然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。 

三、尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对object中所有其它synchronized(this)同步代码块的访问将被      阻塞。 

四、第三个例子同样适用其它同步代码块。也就是说,当一个线程访问object的一个synchronized(this)同步代码块时,它就获得了这个object的对象锁。结          果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。 

五、以上规则对其它对象锁同样适用

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/8ea0f16ab85a6c2f14b978a12ccb72b0.html