Java 里面进行多线程通信的主要方式就是共享内存的方式,共享内存主要的关注点有两个:可见性和有序性原子性。Java 内存模型(JMM)解决了可见性和有序性的问题,而锁解决了原子性的问题,理想情况下我们希望做到“同步”和“互斥”。有以下常规实现方法:
一、将数据抽象成一个类,并将数据的操作作为这个类的方法;
- 将数据抽象成一个类,并将对这个数据的操作作为这个类的方法,这么设计可以和容易做到同步,只要在方法上加”synchronized“。
public class MyData
{
private int j = 0;
public synchronized void add()
{
j++;
System.out.println("线程" + Thread.currentThread().getName() + "j 为:" + j);
}
public synchronized void dec()
{
j--;
System.out.println("线程" + Thread.currentThread().getName() + "j 为:" + j);
}
public int getData()
{
return j;
}
}
public class AddRunnable implements Runnable
{
MyData data;
public AddRunnable(MyData data)
{
this.data = data;
}
public void run()
{
data.add();
}
}
public class DecRunnable implements Runnable
{
MyData data;
public DecRunnable(MyData data)
{
this.data = data;
}
public void run()
{
data.dec();
}
}
public static void main(String[] args){
MyData data = new MyData();
Runnable add = new AddRunnable(data);
Runnable dec = new DecRunnable(data);
for(int i = 0; i < 2; i++)
{
new Thread(add).start();
new Thread(dec).start();
}
}
二、Runnable对象作为一个类的内部类;
- 将Runnable 对象作为一个类的内部类,共享数据作为这个类的成员变量,每个线程对共享数据的操作方法也封装在外部类,以便实现对数据的各个操作的同步和互斥,作为内部类的各个Runnable 对象调用外部类的这些方法。
public class MyData
{
private int j = 0;
public synchronized void add()
{
j++;
System.out.println("线程" + Thread.currentThread().getName() + "j 为:" + j);
}
public synchronized void dec()
{
j--;
System.out.println("线程" + Thread.currentThread().getName() + "j 为:" + j);
}
public int getData()
{
return j;
}
}
public class TestThread
{
public static void main(String[] args)
{
final MyData data = new MyData();
for(int i = 0; i < 2; i++)
{
new Thread(new Runnable()
{
public void run()
{
data.add();
}
}).start();
new Thread(new Runnable()
{
public void run()
{
data.dec();
}
}).start();
}
}
}
网友评论