线程的通讯

线程的通讯

线程间通信其实就是多个线程在操作同一个资源,但操作动作不同

生产者消费者

如果有多个生产者和消费者,一定要使用while循环判断标记,然后在使用notifyAll唤醒,否者容易只用notify容易出现只唤醒本方线程情况,导致程序中的所有线程都在等待。

例如:有一个数据存储空间,划分为两个部分,一部分存储人的姓名,一部分存储性别,我们开启一个线程,不停地想其中存储姓名和性别(生产者),开启另一个线程从数据存储空间中取出数据(消费者)。

由于是多线程的,就需要考虑,假如生产者刚向数据存储空间中添加了一个人名,还没有来得及添加性别,cpu就切换到了消费者的线程,消费者就会将这个人的姓名和上一个人的性别进行了输出。

还有一种情况是生产者生产了若干次数据,消费者才开始取数据,或者消费者取出数据后,没有等到消费者放入新的数据,消费者又重复的取出自己已经去过的数据。

public class Demo10 {

public static void main(String[] args) {

Person p = new Person();

Producer pro = new Producer(p);

Consumer con = new Consumer(p);

Thread t1 = new Thread(pro, "生产者");

Thread t2 = new Thread(con, "消费者");

t1.start();

t2.start();

}

}

 

// 使用Person作为数据存储空间

class Person {

String name;

String gender;

}

 

// 生产者

class Producer implements Runnable {

Person p;

 

public Producer() {

 

}

 

public Producer(Person p) {

this.p = p;

}

 

@Override

public void run() {

int i = 0;

while (true) {

if (i % 2 == 0) {

p.name = "jack";

p.gender = "man";

} else {

p.name = "小丽";

p.gender = "";

}

i++;

}

 

}

 

}

 

// 消费者

class Consumer implements Runnable {

Person p;

 

public Consumer() {

 

}

 

public Consumer(Person p) {

this.p = p;

}

 

@Override

public void run() {

 

while (true) {

System.out.println("name:" + p.name + "---gnder:" + p.gender);

}

}

 

}

 

在上述代码中,ProducerConsumer 类的内部都维护了一个Person类型的p成员变量,通过构造函数进行赋值,在man方法中创建了一个Person对象,将其同时传递给ProducerConsumer对象,所以ProducerConsumer访问的是同一个Person对象。并启动了两个线程。

 

输出:

 

显然屏幕输出了小丽 man 这样的结果是出现了线程安全问题。所以需要使用synchronized来解决该问题。

 

package cn.itcast.gz.runnable;

 

public class Demo10 {

public static void main(String[] args) {

Person p = new Person();

Producer pro = new Producer(p);

Consumer con = new Consumer(p);

Thread t1 = new Thread(pro, "生产者");

Thread t2 = new Thread(con, "消费者");

t1.start();

t2.start();

}

}

 

// 使用Person作为数据存储空间

class Person {

String name;

String gender;

}

 

// 生产者

class Producer implements Runnable {

Person p;

 

public Producer() {

 

}

 

public Producer(Person p) {

this.p = p;

}

 

@Override

public void run() {

int i = 0;

while (true) {

synchronized (p) {

if (i % 2 == 0) {

p.name = "jack";

p.gender = "man";

} else {

p.name = "小丽";

p.gender = "";

}

i++;

}

 

}

 

}

 

}

 

// 消费者

class Consumer implements Runnable {

Person p;

 

public Consumer() {

 

}

 

public Consumer(Person p) {

this.p = p;

}

 

@Override

public void run() {

 

while (true) {

synchronized (p) {

System.out.println("name:" + p.name + "---gnder:" + p.gender);

}

 

}

}

 

}

 

编译运行:屏幕没有再输出jack –  或者小丽- man 这种情况了。说明我们解决了线程同步问题,但是仔细观察,生产者生产了若干次数据,消费者才开始取数据,或者消费者取出数据后,没有等到消费者放入新的数据,消费者又重复的取出自己已经去过的数据。这个问题依然存在。

升级:在Person类中添加两个方法,setread方法并设置为synchronized让生产者和消费者调用这两个方法。

public class Demo10 {

public static void main(String[] args) {

Person p = new Person();

Producer pro = new Producer(p);

Consumer con = new Consumer(p);

Thread t1 = new Thread(pro, "生产者");

Thread t2 = new Thread(con, "消费者");

t1.start();

t2.start();

}

}

 

// 使用Person作为数据存储空间

class Person {

String name;

String gender;

 

 

public synchronized void set(String name, String gender) {

this.name = name;

this.gender = gender;

}

 

public synchronized void read() {

System.out.println("name:" + this.name + "----gender:" + this.gender);

}

 

}

 

// 生产者

class Producer implements Runnable {

Person p;

 

public Producer() {

 

}

 

public Producer(Person p) {

this.p = p;

}

 

@Override

public void run() {

int i = 0;

while (true) {

 

if (i % 2 == 0) {

p.set("jack", "man");

} else {

p.set("小丽", "");

}

i++;

 

}

 

}

 

}

 

// 消费者

class Consumer implements Runnable {

Person p;

 

public Consumer() {

 

}

 

public Consumer(Person p) {

this.p = p;

}

 

@Override

public void run() {

 

while (true) {

p.read();

 

}

}

 

}

 

 

需求:我们需要生产者生产一次,消费者就消费一次。然后这样有序的循环。

这就需要使用线程间的通信了。Java通过Object类的waitnotifynotifyAll这几个方法实现线程间的通信。

posted on 2016-10-18 18:15  眼泪笑我愚昧  阅读(135)  评论(0编辑  收藏  举报

导航