线程介绍,线程与进程区别,如何使用多线程,Thread类,Runnable接口,补充知识(方法重载,方法重写)

简介: 线程介绍,线程与进程区别,如何使用多线程,Thread类,Runnable接口,补充知识(方法重载,方法重写)

引言:搞进程是为了什么呢?

满足并发编程,这样的需求,cpu多个核心,应用程序做出对应调整,让代码可以把多个核心充分利用起来~,当需要支持多个任务的时候——进程就十分关键了,多进程已经很好的实现了并发编程的效果,但是缺点也很明显,就是太重了,如果我们进程大规模的创建销毁,则开销就会比较大(普通用户一般不会,但是服务器就不一定了)——于是聪明的猿们想了个办法


一、💛线程的介绍

线程(轻量级的进程):创建进程的时候,只分配自己一个简单的PCB(进程控制块,c,结构),而不去分配后续的这些内存硬盘资源,这样不就更快了没,这样既可以并发的去执行任务,又可以提升创建/销毁速度。

但是这时候又有疑问了🤔️

只创建一个pcb,没有分配后续,但是线程搞出来就是为了执行任务,执行任务有需要消耗这些资源(不给钱,人家也不能给你干活对吧😶)聪明的猿->

方法:创建的还是进程,创建进程的时候,把资源都分配好~,后续创建的线程,让线程在进程内部(进程和线程之间的关系,可以认为进程包含了线程),后续进程中的新的线程,直接复用前面的进程这里创建好的资源~

其实💨💨一个进程至少包含一个进程,最初创建出来的这个,可以视为是一个只包含一个线程的进程(此时创建需要分配资源,此时第一个线程的开销可能比较大的,但是后续再这个进程里创建进程,就可以省略分配资源的过程,资源是已经有了的

补充:线程也是通过PCB描述的(看出来了某些佬们制作的时候,可能偷懒了,统一设置一样的了,此时一个PCB对应一个线程,多个PCB对应一个进程

⚠️⚠️PCB(线程)内存,指针,文件描述符表,同一个进程的多个PCB这两个字段内容相同但是上下文,状态,记账信息···PCB支持调度的属性,则这些每个PCB不同。

原因:同一个进程的这些线程,共用一份资源(内存+硬盘),但是每个线程独立去cpu上调度(状态,上下文,优先级,记账信息,各自有各自的一份)

💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖 💖

记住:进程:操作系统进行资源分配的基本单位

           线程:操作系统进行调度执行的基本单位

线程不能太多,线程调度的开销,反而拖慢整个程序效率。

二、线程产生的问题 🌝  🌝

多线程的优势:稳定,一点问题不会发生,多进程的隔离性

1.同时看上一个鸡大腿,两个人可能会冲突,这样的冲突,可能带来bug,线程安全问题(多线程编程里面,最关键的问题)

2.假如有个大哥心情焦躁把桌子掀翻了(乌鸦:难办?难办就都别办了),一旦某个进程,执行过程中出现异常,并且这个异常没有背很好的处理,此时,就可能会导致整个进程终止(进程中的所有进程也就随之终止)

——所以说线程也不一定就比进程更好,只是有优势:更轻量,销毁速度快

多进程,多线程(JAVA常用)多本质并发编程实现模型,实际上,还存在其他并发编程的实现模型。

3.补充 进程1假如访问进程2才叫越界,但是线程1,线程2是共用一个空间。

三、面试经典问题(小总结)😋

谈谈进程和线程的区别和联系

1.进程包含线程,都是为了实现并发编程的方式,线程比进程更加轻量。

2.进程是系统分配资源的基本单位线程是系统调度执行的基本单位,创建进程的时候把分配资源(虚拟地址空间——内存资源文件描述符表——硬盘资源)的工作给做了,后续创建线程,直接共用之前的资源即可。

3.进程有独立的地址空间,彼此之间不会相互影响,进程的独立性->系统稳定性,多个进程共用这一份地址空间,一个线程一旦抛出异常,就可能导致整个进程异常结束->多个线程之间

⛄️⛄️⛄️

线程是轻量,但是也有成本,但是在互联网圈子,高并发的服务器太多了,要处理的并发量太多了,非常频繁的创建/销毁线程,开销也不可以被忽视了。聪明的猿们->

两个办法:🐻

1.轻量级编程:协程/纤程                java标准库里,没有配置(目前),有一些第三方库实现了协程,GO天然支持协程(也是热门的一大原因)

2.线程池:池(pool)计算机中非常经典的思想方法,把一些要释放的资源,不着急释放,先放到池子里,以备后续使用,申请资源的时候,也先提前把资源请好,也放到一个池子里面,后续申请的时候也比较方便。


线程~本身是操作系统的概念,操作系统也提供了一些API供猿们使用,java中,把操作系统封装,把操作系统API,又进行封装,提供Thread类。java追求跨平台,JVM 能屏蔽不同操作系统的差异。

另外谈到java简单复习一个小的知识点 🐷方法重载和方法重写(基础,还是很重要的)

方法重载:方法名相同 ,参数列表不同,返回值不作要求

方法重写:是父类和子类之间,子类的方法类似于覆盖这种重写了父亲的方法。方法并且不可以是静态的,子类的访问限定修饰符一定要大于父类的,final修饰的不能被重写,返回值一样的。

myThread.start():创建进程,此处这个start就是在创建进程,🐵(这个操作就会在底层调用操作系统的API,同时还会在操作系统的内核里创建出对应的PCB结构,并且加入对应的链表中···)此时,这个新创建出来的线程,就参与到CPU的调度之中,这个线程接下来要执行的工作,就是上面重写run()这个方法。

那么肯定也有人会有问题,假如说我们使用run()方法行不行呢 ,  可以,但是二者的区别是,一个使用后是产生了线程,一个就是单纯的使用了普通方法。

 

我们看下图,调整了一下main中有while循环,线程中也存在一个while循环,两个都是死循环,使用start方法执行,(相当于我们创建了一个线程,去执行下面的代码)则两个循环都在执行,是交替来做的,两个线程,分别执行自己的 循环,这两个线程都能参与到cpu的调度之中,这两个线程while循环并发式的执行。

假如说这个地方还用run的话,就是单线程执行,那就会一直执行run方法,并未创建新线程,每个线程都是一个独立执行流,每个线程都可以独自执行一段代码,多个线程之间是并发关系~。

上图的代码自行复制

class MyThread extends  Thread {
    @Override
    public void run() {
        while (true) {
            System.out.println("hello world");
        }
    }
}
    public  class  Demo {
        public static void main(String[] args) {
            MyThread myThread = new MyThread();
            myThread.start();
            while (true) {
                System.out.println("love");
            }
        }
    }

创建进程之后,查看线程方法

1.idea调试器 main这个主线程,Thread新的线程

2.jocnsole 找路径去找,SDK,双击本地进程然后去打开,看堆栈跟踪,线程的调用栈,方法之间调用关系,尤其是当前程序basil,查看胰腺癌这里每个线程的调用栈,就可以大概知道,哪个代码出现卡死的情况

介绍一个静态方法(Thread的),java中sleep(让他转的走的慢一点),🕙🕙🕙Sleep函数可以使计算机程序(进程,任务或线程)进入休眠,使其在一段时间内处于非活动状态。当函数设定的计时器到期,或者接收到信号、程序发生中断都会导致程序继续执行前提.注意继承的类里面只可以try catch处理异常,而不能选择去抛异常(因为他的父类没有抛异常,他这个重写也不可以),但是下面的while是可以抛异常的,因为他不需要继承父类

class MyThread extends  Thread {
    @Override
    public void run() {
        while (true) {
            System.out.println("hello world");
            try {
                Thread.sleep(1000);  //必须try catch
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
    public  class  Demo {
        public static void main(String[] args) throws InterruptedException {
            MyThread myThread = new MyThread();
            myThread.start();
            while (true) {
                System.out.println("love");
                Thread.sleep(1000);      //抛异常,try catch都可以
            }
        }
    }

用完sleep之后,此处也并非是严格的交替,也存在一些顺序反过来的情况,当这两个线程进行sleep之后,就会进入阻塞状态,当时间到,系统会重新唤醒这两个线程,并且恢复对线程的调度,当这两个线程被唤醒,谁先后,可以视为随机。

但其实也不是随机,系统在进行多个线程调度的时候,没有一个明确的顺序,而是按照这样随机的方式进行调度~这样的随机调度的过程,称为抢占式执行

Java通过Thread创建线程的方式方法,还有很多写法

1.创建一个类,继承Thead,重写run方法

2.创建一个类,实现Runnable(实现interface,Java中interface一般都是形容词词性 哈哈哈没啥用),重写run方法

class  MyRunnable implements  Runnable{
    @Override
    public void run() {
        while(true){
            System.out.println( "hello world");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
    public  class  Demo {
        public static void main(String[] args) throws InterruptedException {
            MyRunnable runnable = new MyRunnable();
            Thread t=new Thread(runnable);
            t.start();
            while (true) {
                System.out.println("love");
                Thread.sleep(1000);
            }
        }
    }

Runnable,本身没有start方法,绕不开start,这是必须的一步,

Thread是要完成的操作,放到Thread的run中

Runnable这里,则是分开了,要把完成的工作放到Runnable,再让Runnable和Thread配合,

Runnable:好处线程要执行的任务和线程本身,进一步的解耦合了,(相当于把任务和线程拆开了Runnable,并发编程的方式,来完成某个工作具体细节,

使用多线程——使用Runnable搭配线程使用,

线程池——可以使用Runnable搭配线程池使用

协程——--Runnable搭配协程

 


相关文章
|
8月前
|
安全 算法 Java
Java 多线程:线程安全与同步控制的深度解析
本文介绍了 Java 多线程开发的关键技术,涵盖线程的创建与启动、线程安全问题及其解决方案,包括 synchronized 关键字、原子类和线程间通信机制。通过示例代码讲解了多线程编程中的常见问题与优化方法,帮助开发者提升程序性能与稳定性。
346 0
|
5月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
269 1
|
5月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
285 1
|
8月前
|
数据采集 监控 调度
干货分享“用 多线程 爬取数据”:单线程 + 协程的效率反超 3 倍,这才是 Python 异步的正确打开方式
在 Python 爬虫中,多线程因 GIL 和切换开销效率低下,而协程通过用户态调度实现高并发,大幅提升爬取效率。本文详解协程原理、实战对比多线程性能,并提供最佳实践,助你掌握异步爬虫核心技术。
|
9月前
|
Java 数据挖掘 调度
Java 多线程创建零基础入门新手指南:从零开始全面学习多线程创建方法
本文从零基础角度出发,深入浅出地讲解Java多线程的创建方式。内容涵盖继承`Thread`类、实现`Runnable`接口、使用`Callable`和`Future`接口以及线程池的创建与管理等核心知识点。通过代码示例与应用场景分析,帮助读者理解每种方式的特点及适用场景,理论结合实践,轻松掌握Java多线程编程 essentials。
617 5
|
数据采集 Java 数据处理
Python实用技巧:轻松驾驭多线程与多进程,加速任务执行
在Python编程中,多线程和多进程是提升程序效率的关键工具。多线程适用于I/O密集型任务,如文件读写、网络请求;多进程则适合CPU密集型任务,如科学计算、图像处理。本文详细介绍这两种并发编程方式的基本用法及应用场景,并通过实例代码展示如何使用threading、multiprocessing模块及线程池、进程池来优化程序性能。结合实际案例,帮助读者掌握并发编程技巧,提高程序执行速度和资源利用率。
647 0
|
5月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
238 6
|
8月前
|
Java API 微服务
为什么虚拟线程将改变Java并发编程?
为什么虚拟线程将改变Java并发编程?
390 83
|
10月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
367 0
|
5月前
|
Java 调度 数据库
Python threading模块:多线程编程的实战指南
本文深入讲解Python多线程编程,涵盖threading模块的核心用法:线程创建、生命周期、同步机制(锁、信号量、条件变量)、线程通信(队列)、守护线程与线程池应用。结合实战案例,如多线程下载器,帮助开发者提升程序并发性能,适用于I/O密集型任务处理。
477 0

热门文章

最新文章