在详细了解了synchronized关键字的用法和本质之后,笔者罗列了三个初学者容易出现的错误,以供读者参考。
1.与monitor关联的对象不能为空
private final Object mutex=null;
public void syncMethod()
{
synchronized (mutex)
{
//
}
}
Mutex为null,很多人还是会犯这么简单的错误,每一个对象和一个monitor关联,对象都为null了,monitor肯定无从谈起。
2.synchronized作用域太大
由于synchronized关键字存在排他性,也就是说所有的线程必须串行地经过synchronized保护的共享区域,如果synchronized作用域越大,则代表着其效率越低,甚至还会丧失并发的优势,示例代码如下:
public static class Task implements Runnable
{
Override
public synchronized void run()
{
//
}
}
上面的代码对整个线程的执行逻辑单元都进行了synchronized同步,从而丧失了并发的能力,synchronized关键字应该尽可能地只作用于共享资源(数据)的读写作用域。
3.不同的monitor企图锁相同的方法
笔者发现很多人都容易犯这个错误,下面我们就来举个例子说明一下:
public static class Task implements Runnable
{
private final Object MUTEX=new Object();
Override
public void run()
{
//...
synchronized (MUTEX)
{
//...
}
//...
}
}
public static void main(String[] args)
{
for(inti=0;i<5;i++);
{
new Thread(Task:: new). start();
}
}
上面的代码构造了五个线程,同时也构造了五个Runnable实例,Runnable作为线程逻辑执行单元传递给Thread,然后你将发现,synchronized根本互斥不了与之对应的作用域,线程之间进行monitor lock的争抢只能发生在与monitor关联的同一个引用上,上面的代码每一个线程争抢的monitor关联引用都是彼此独立的,因此不可能起到互斥的作用。
以上就是使用synchronized时,需要注意的三个问题,你都了解了吗?