笔记-java之多线程概念


活动地址:CSDN21天学习挑战赛

学习的最大理由是想摆脱平庸,早一天就多一份人生的精彩;迟一天就多一天平庸的困扰。各位小伙伴,如果您:
想系统/深入学习某技术知识点…
一个人摸索学习很难坚持,想组团高效学习…
想写博客但无从下手,急需写作干货注入能量…
热爱写作,愿意让自己成为更好的人…

多线程

线程:是一个程序内部的一条执行路径。
如果程序中只有一条执行路径,那么这个程序就是单线程的程序。

多线程:是指从软硬件上实现多条执行流程的技术。

Thread类
java是通过java.lang.Thread类来代表线程的。

多线程的实现方案一:继承Thread类

1.定义一个子类MyThread继承线程类java.lang.Thread,重写run()方法。
2.创建MyThread类的对象
3.调用线程对象的start()方法启动线程(启动后还是执行run方法的)

多线程的实现方案二:实现Runnable接口

1.定义一个线程任务类MyRunnable实现Runnable接口,重写run()方法
2.创建MyRunnable任务对象
3.把MyRunnable任务对象交给Thread处理
4.调用线程对象的start()方法启动线程
优点:线程任务类只是实现接口,可以继续继承类和实现接口,扩展性强。
缺点:多一层对象包装,如果线程有执行结果是不可以直接返回的。

线程安全问题:
多个线程同时操作同一个共享资源的时候可能可能会出现业务安全问题,称为线程安全问题。

线程安全问题出现的原因:

1.存在多线程并发
2.同时访问共享资源
3.存在修改共享资源

线程同步的核心思想:
加锁,把共享资源进行上锁,每次只能一个线程进入访问,访问完毕后解锁,然后其他线程才能进来。

线程同步方式一:
同步代码块:
作用:把出现线程安全问题的核心代码给上锁。
原理:每次只能一个线程进入,执行完毕后自动解锁,其他线程才可以进来执行。
synchronized(同步锁对象){
//操作共享资源的代码(核心代码)
}

锁对象的规范要求

理论上锁对象只要对于当前同时执行的线程来说是同一对象即可。
建议使用共享资源作为锁对象
对于实例方法建议使用this作为锁对象
对于静态方法建议使用字节码(类名.class)对象作为锁对象

线程同步方法二:
同步方法:
作用:把出现安全问题的核心方法给上锁
原理:每次只能一个线程进入,执行完毕后自动解锁,其他进程才可以进来执行。

修饰符 synchronized 返回值类型 方法名称(形参列表){
//操作共享资源的代码
}

同步方法底层原理:

同步方法其实底层原理也是隐式锁对象的,只是锁的范围是整个方法代码。
如果方法是实例方法:同步方法默认用this作为锁对象。但是代码要高度面向对象。
如果方法是静态方法:同步方法默认用类名.class作为锁对象

同步代码块锁的范围更小,同步方法锁范围更大。

线程同步方法三:
Lock锁

public class LockTest {
    private static Lock lock = new ReentrantLock();
    private static  int count = 0;
    private static void add() {
        try {
            //加锁
            lock.lock();
            count++;
            //重入锁
            reduce();
        } finally {
            // 记得释放锁
            lock.unlock();
            System.out.println(count);
        }
    }
    private static void reduce(){
        // 加锁
        lock.lock();
        count--;
        // 释放锁
        lock.unlock();
    }
    public static void main(String[] args) throws InterruptedException {
        for (int i = 0; i < 100; i++) {
            new Thread(LockTest::add).start();
        }
    }
}

线程通信
线程通信就是线程间相互发送数据,线程间共享一个资源即可实现线程通信。

线程通信常见形式:
通过分享一个数据的方式实现
根据共享数据的情况决定自己该怎么做,以及通知其他线程怎么做。

线程通信的前提:线程通信通常是在多个线程操作同一个共享资源的时候需要通信,且要保证线程安全。

线程池
线程池就是一个可以复用线程的技术。

不使用线程池的问题:
如果用户每发起一个请求,后台就创建一个新线程来处理,下次新任务来了又要创建新线程,而创建的新线程的开销是很大的,这样会严重影响系统的性能。

ExecutorService接口代表线程池

如何得到线程池对象:
方式一:使用ExecutorService的实现类ThreadPoolExecutor自创建一个线程池对象。
方式二:使用Executors(线程池的工具类)调用方法返回不同特点的线程池对象。

临时线程什么时候创建?
新任务提交时发现核心线程都在忙,任务队列也满了,并且还可以创建临时进程,此时才会创建临时进程。
什么时候会拒绝任务?
核心线程和临时线程都在忙,任务队列也满了,新的任务过来的时候才会开始拒绝任务。

ThreadPoolExecutor创建线程池对象实例

ExecutorService pools = new ThreadPoolExecutor(3,5,8,TimeUnit.SECONDS,new ArrayBlockingQueue<>(6),Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值