java---教你怎么判断多线程是否安全
多线程不安全,主要因为cpu分配机制,谁获得了cpu谁就能执行,因此造成了线程的不安全.
那么,如何找出问题呢?
1.明确哪些代码是多线程运行的代码,
2.明确共享数据
3.明确多线程运行代码中哪些语句是操作共享数据.
根据以上三部.下面看例子,
package com.niuli.develop;
public class Test {
public static void main (String [] args) {
Cus c = new Cus();
new Thread(c).start();
new Thread(c).start();
}
}
class Bank{
private int sum = 0;
public void add(int n) {
sum = sum + n;
System.out.println("sum= "+sum);
}
}
class Cus implements Runnable{
Bank b = new Bank();
@Override
public void run() {
for (int i = 0; i < 3; i++) {
b.add(100);
}
}
}
很简单的例子,
首先第一步,找到多线程运行的代码
也就是run方法里面的代码
for (int i = 0; i < 3; i++) {
b.add(100);
}
另外这里用到了Add方法,所以add也算
public void add(int n) {
sum = sum + n;
System.out.println("sum= "+sum);
}
第二步,找到共享数据
Bank b = new Bank();
private int sum = 0;
第三步,明确多线程使用的共享数据.
首先对于数据b就一条语句使用了这个共享变量,因此不会出现线程不安全,但是对于sum,有两条语句使用的,所以就会出现线程安全问题,
解决办法,同步块
Object obj = new Object();
public void add(int n) {
synchronized (obj) {
sum = sum + n;
System.out.println("sum= "+sum);
}
}
或者,同步函数
public synchronized void add(int n) {
sum = sum + n;
System.out.println("sum= "+sum);
}
以上就是线程安全的解决办法,同步机制虽然游侠,但是带来了一定的性能损耗,所以,对操作共享数据的部分执行同步,尽量减少这种损耗
补充:
如果同步函数是静态的,那么使用的锁就必须是所在类的字节码文件对象,也就是 类名.class,解释来说,静态是和类一起加载的,所以在静态加载的时候是不可能有object的对象,所以锁就必须用在静态加在之前的一个对象.
版权声明:本文为博主原创文章,未经博主允许不得转载。