美文网首页Java
复习JavaSE 10.多线程

复习JavaSE 10.多线程

作者: 第二套广播体操 | 来源:发表于2019-03-08 19:49 被阅读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();

    }
}

线程池

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

  • java.util.concurrent.Executors 线程工厂
  • static ExecutorService newFixedThreadPool(int nThreads)
  • 创建一个线程池,该线程池重用固定数量的从共享无界队列中运行的线程。
  • ExecutorService方法 Future<?> submit(Runnable task)
  • 提交一个可运行的任务执行,并返回一个表示该任务的未来。
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();
    }
}

相关文章

  • 复习JavaSE 10.多线程

    并发:指两个或者多个事件在同一时间段同时发生 例如cpu在多个线程中来回切换并行:指 两个或者多个事件在同一时刻发...

  • 第一个“30天计划”

    目前情况 java课本周就结束了,就差准备最后一节课和出试卷了。 个人的javaSE几乎复习完毕。但是多线程、枚举...

  • Chapter 10. 多线程

    阅读原文 Chapter 10. 多线程 10.1 多线程概述 基本概念:程序 - 进程 - 线程 程序(prog...

  • javase(多线程)

    1. 进程和线程 进程:正在进行的程序。每一个进程执行都有一个执行顺序,该顺序是一个执行路径,或者叫一个控制单元。...

  • 2018-12-11

    10. HashMap在多线程环境下使用需要注意什么?为什么? 在多线程环境下,需要使用ConcurrentHas...

  • java学习路线

    javaSE java基础语法 java文件操作 java网络操作 java多线程 java数据库操作 java ...

  • JAVASE-多线程

    Thread 多线程为我们解决了程序中需要并发执行多个任务的操作,可以通过创建一个线程来负责执行需要执行的任务。创...

  • JavaSE-多线程

    目录 [TOC] 第一章:多线程基础 想要设计一个程序,边打游戏边听歌,怎么设计? 要解决上述问题,得使用多进程或...

  • JavaSE之数组复习

    一维数组 介绍 数组是一种引用类型 数组是一种简单的数据结果,线性的结构 数组是一种容器,可以来存储其他元素,数组...

  • JavaSE之String复习

    基本 我们先来看一下String在Jdk中的定义 我们发现这是一个String类被finl进行修饰了,name就可...

网友评论

    本文标题:复习JavaSE 10.多线程

    本文链接:https://www.haomeiwen.com/subject/rnsdpqtx.html