复习JavaSE 10.多线程

并发:指两个或者多个事件在同一时间段同时发生 例如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();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容

  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    胜浩_ae28阅读 5,094评论 0 23
  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    小徐andorid阅读 2,803评论 3 53
  •   一个任务通常就是一个程序,每个运行中的程序就是一个进程。当一个程序运行时,内部可能包含了多个顺序执行流,每个顺...
    OmaiMoon阅读 1,664评论 0 12
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,952评论 1 18
  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,448评论 1 15