Java

复习JavaSE 10.多线程

2019-03-08  本文已影响0人  第二套广播体操

并发:指两个或者多个事件在同一时间段同时发生 例如cpu在多个线程中来回切换
并行:指 两个或者多个事件在同一时刻发生 单独一个cpu处理单独的线程 一同进行
进程和线程的关系:
一个进程中可以有多个线程执行
锁只能监视由synchronized关键字内部 或者由lock接口实现类包括的代码部分 已实现同步或者通讯
相同锁之间才能实现通信


Java中程序属于抢占式调度方法运行程序

创建多线程方法 1:创建Thread子类 重写run方法 通过创建对象并调用start方法

public class CreateThread01 extends Thread {

    @Override
    public void run() {
        super.run();
        System.out.println(Thread.currentThread().getName());
        for (int n = 0; n <20; n++) {
            System.out.println("n====="+n);
        }
    }
}
class CreateThread01Test {
    public static void main(String[] args) {
        System.out.println(Thread.currentThread().getName());
        new CreateThread01().start();
        for (int i = 0; i <20; i++) {
            System.out.println("i===="+i);
        }
    }
}

2:实现runnable接口 并创建Thread对象 将runnable作为参数传入Thread对象中
Runnable接口的好处:
1:避免了继承的局限性
一个类只能继承一个类实现runnable接口可以实现其他的类和接口
2:增强了程序的扩展性 降低了程序的耦合度
把设置线程的任务和开启新线程进行分离(解耦)
重写run设置线程的任务
创建Thread 开启新线程

public class RunnableImpl01 implements Runnable {
    @Override
    public void run() {
        for (int i = 0; i <10; i++) {
            System.out.println(Thread.currentThread().getName()+"  "+i);
        }
    }
}
class CreateThroad02{
    public static void main(String[] args) {
        RunnableImpl01 impl01 = new RunnableImpl01();
        new Thread(impl01).start();
        for (int i = 0; i <10; i++) {
            System.out.println(Thread.currentThread().getName()+" "+i);
        }
    }
}

内部类形式创建线程

public class InnerThread {
    public static void main(String[] args) {
        new Thread(){
            @Override
            public void run() {
                for (int i = 0; i <20; i++) {
                    System.out.println(Thread.currentThread().getName()+" "+i);
                }
            }
        }.start();
        new Thread(new Runnable() {
            @Override
            public void run() {

                for (int i = 0; i <20; i++) {
                    System.out.println(Thread.currentThread().getName()+" "+i);
                }
            }
        }).start();
        for (int i = 0; i <20; i++) {
            System.out.println(Thread.currentThread().getName()+"  "+i);
        }
    }
}

多线程共享数据可能产生安全问题

所以需要同步处理
同步代码块:
通过代码块的锁对象 可以使用任意对象
但必须保证多个现成的锁对象是同一个
锁对象的作用:
把同步代码块锁住 只让一个线程在同步代码块中执行

保证锁的唯一性 所以锁的定义应该在run方法外部

  private int tacket = 100;
    Object object = new Object();

    @Override
    public void run() {
        while (true) {
            synchronized (object) {
                if (tacket > 0) {
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName() + 
                            "正在销售第 " + tacket + "张");
                    tacket--;
                }
            }
        }
    }

同步方法:
将共享数据封装在方法中 方法用Synchronized修饰
同步方法的锁就是this
this就是实现runnable接口的实现类对象 即this出现的本类对象
静态方法的锁对象是本类的class属性 即class文件的对象

  private static int tacket = 100;
    @Override
    public void run() {
        while (true)
        methodDemo1();
    }
//锁就是this this就是 创建本类实例(实现类的对象)
//    静态同步方法因为优先于对象存在 所以锁就是本类的.class文件对象
    private static synchronized void methodDemo1() {
        if (tacket > 0) {
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "正在销售第 " + tacket + "张");
            tacket--;
        }
    }

Lock锁接口 1.5版本以后 更灵活
在共享数据之前使用lock();方法 在之后使用unlock();方法 解锁
将unlock();放入finally中 使用

public class SynchronizedLock implements Runnable {
    private int tacket = 100;
     Lock lock=new ReentrantLock();

    @Override
    public void run() {
        while (true) {
             lock.lock();
                if (tacket > 0) {
                    try {
                        Thread.sleep(100);
                        System.out.println(Thread.currentThread().getName() + "正在销售第 " + tacket + "张");
                        tacket--;
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }finally {
//                        无论怎么样 最后一定要释放锁
                        lock.unlock();
                    }

                }

        }
    }
}

线程同步
wait notify保证了只有一个线程在执行 一个线程等待另一个线程去唤醒
wait notify 一定要添加锁对象 (确定哪个锁之间的同步相互通信 因为可能出现多个同步 多个锁 )
保证同步中的多个线程进行 等待或者唤醒
放在同步中保证同步的两个线程相互操作

public class ThreadWaitNotify {
    public static void main(String[] args) {
//        共享代码 锁对象
        Object o = new Object();
        new Thread() {
            @Override
            public void run() {
                synchronized (o) {
                    System.out.println("顾客吃包子告诉老板数量");
                    try {
                      o.wait();

                    } catch (InterruptedException e) {

                    }
                    System.out.println("吃了起来");
                }
            }
        }.start();

        new Thread() {
            @Override
            public void run() {
//                try {
//                    Thread.sleep(5000);

//                } catch (InterruptedException e) {
//                    e.printStackTrace();
//                }
                synchronized (o) {
                    System.out.println("老板做包子");
                    o.notify();
                }
                System.out.println("做完包子 可以吃了");

            }
        }.start();
    }
}

等待唤醒机制说明
使用Object类的wait和notify方法
要求 等待唤醒线程 只能有一个线程在执行
使用同步方法
等待唤醒对应的锁 应为同一个

实例 吃包子 做包子

创建一个类 描述包子
包子有皮 有馅 能证明包子是否存在

创建两个线程类实现Runnable接口
1 包子铺类
如果包子存在 则线程等待
如果包子不存在 则制作包子
分别制作两种不同馅的包子
并唤醒吃包子的线程

2 吃包子线程
如果包子 不存在 则线程等待
如果包子存在 则吃掉包子 并换馅

创建测试类

public class Baozi {
    private String pi;
    public String[] xian={"白菜馅","猪肉馅"};
    public int  xianIndex=0;
    public boolean flog =false;

}
//包子铺类 生产包子
public class Baozipu implements Runnable {
    private Baozi baozi;

    public Baozipu(Baozi baozi) {
        this.baozi = baozi;
    }

    @Override
    public void run() {
        while (true) {
            synchronized (baozi) {
                if (baozi.flog==true){
                    try {
                        baozi.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    try {
                        Thread.sleep(1000);
                        String name = baozi.xian[baozi.xianIndex % 2];
                        System.out.println("生产了" + name + "的包子!");
                        baozi.flog = true;
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } finally {
                        baozi.notify();
                    }

                }
            }
        }
    }
}
public class Chibaozi implements Runnable {

    private Baozi baozi;

    public Chibaozi(Baozi baozi) {

        this.baozi = baozi;
    }

    @Override
    public void run() {
        while (true) {
            synchronized (baozi) {
                if (baozi.flog==false) {
                    try {
                        baozi.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    String name = baozi.xian[baozi.xianIndex % 2];
                    baozi.xianIndex++;
                    try {
                        Thread.sleep(1000);
                        System.out.println("我吃到了" + name + "的包子");
                        baozi.flog = false;
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } finally {
                        baozi.notify();
                    }
                }
            }
        }
    }
}
public class BaoziTest {
    public static void main(String[] args) {
        Baozi baozi = new Baozi();
       new Thread(new Baozipu(baozi)).start();
       new Thread(new Chibaozi(baozi)).start();

    }
}

线程池

降低资源消耗 不需要重复创建线程
提高了响应速度

public class ThreadPoolDemo {
    public static void main(String[] args) {
        ExecutorService threadPool = Executors.newFixedThreadPool(5);
        threadPool.submit(new RunnableImpl());
        //pool-1-thread-1正在执行
        threadPool.submit(new RunnableImpl());
        threadPool.submit(new RunnableImpl());
        threadPool.submit(new RunnableImpl());
        //pool-1-thread-4正在执行
        //pool-1-thread-3正在执行
        //pool-1-thread-1正在执行
        //pool-1-thread-2正在执行

//        销毁线程池 不建议使用
        threadPool.shutdown();
    }
}
上一篇 下一篇

猜你喜欢

热点阅读