什么是进程--进程是系统创建的
Thread 类 线程 是程序中的执行线程,java虚拟机允许应用程序并发的运行多个执行线程
进程:是一个正在执行的程序,每一个进程都有一个执行顺序,该顺序是一个执行路径,或者叫一个控制单元,进程用于定义空间,用于封装控制单元 (java有编译进程,和运行进程)
线程:就是进程中的一个独立的控制单元,线程在控制着进程的的执行, Java VM 启动的时候会有一个进程java.exe,
该进程中至少负责一个线程负责java程序的执行,而且这个线程的运行的代码存在于main方法中,该线程称为主线程,虚拟机启动的时候就是多线程,有多条执行的程序叫做:多线程程序
扩展:其实更细说明jvm , jvm启动不止一个线程,还有负责垃圾回收机制的线程
问题? 如何在自定义的代码中,自定义一个线程呢?
通过api的查找,java已经提供了对线程这类事物的描述,就是Thread类创建线程的第一种方式,继承Thread类
步骤 : 1.定义类继承Thread
2,复写Thread类中的run方法
(目的:将自定义的代码存储下run方法中,让线程去执行) run功能是执行底层封装run的方法
3,调用线程start方法 (该方法两个作用,启动线程,调入run方法)
多线程存在的意义:
发现运行结果每一次都不同
因为多个线程都获取cpu的执行权,cpu执行到谁,谁就运行,明确一点,在某一时刻,只能有一个程序的运行(多核除外)
cpu在快速的切换,已达到看上去是同时运行的效果,我们可以形象把多线程的运行行为在互相抢夺cpu的执行权
这是多线程的一个特性: 随机性,谁抢到谁执行,至于执行多长,cpu说了算 ---主线程是在执行main中的方法
为什么要覆盖run方法呢?
Thread类用于描述线程, 该功能就定义了一个功能,用于储存线程要一下的代码,该储存过程就是run方法 ,也就是说Thread类中的run方法,用于存储线程要运行的代码
sleep(time) 设置冻结的时间段 wait() 等待 notify( ) 唤醒
在没有执行资格的情况下-叫冻结状态 冻结可以分为两个 一个是睡眠一个是等待 --没有执行资格的情况下是冻结状态
有执行资格的情况下-叫临时状态
既有资格又有执行权的叫做-运行状态
在运行时不小心挂了stop() 停止 -- 叫消亡状态 当线程在没有执行的内容后就挂了
当你开启多个线程时,但是COU(在某一时刻)只能执行一个线程 , 把剩下的线程都在临时状态上,它在等待CPU的执行权(具备运行资格,但没有执行权)
线程获取名称的方式有,getName() 原来线程都要自己默认的名称
static 对象 currentThread() 返回对当前正在执行的线程对象的引用
设置线程名称:setName() 或者构造函数
需求:简单的买票程序,多个窗口同时买票,
1定义类实现Runnable接口 2.覆盖Runnable接口中的run方法(将线程要运行的代码存放在run方法中) 3.通过Thread类建立线程对象 4.将Runnable接口的之类对象作为实际参数传递给Thread类的构造函数(为什么要将Runnable接口的之类对象传递给Thread的构造函数,因为自定义的run方法所属的对象是Runnable接口的之类对象,所有要让线程去指定对象的run方法就必须明确该run方法所属的对象) 5.调用Thread类的stare方法开启线程并调用Runnable接口的之类的run方法
实现方式的好处:避免单继承的局限性 在定义线程时,建议使用继承方式
两种方式的区别:继承Thread:线程代码存放Thread之类run方法中,
继承Runnable,线程代码存在接口的之类的run方法;
线程在强执行权,我们可以同步打印线程的的顺序,但是执行全部我们是无法控制的
在这里对象就如同一把锁持有锁的线程可以在同步中执行,没有持有锁的线程即使获取CPU的执行权,也进不去,因为没有获取锁
单线程的查询程序不需要加同步锁
在线程同步的前提,1.我们必须要有两个或者两个以上的线程 2.必须是多个线程使用同一把锁
必须保证同步中只能有一个线程在运行
好处:解决了多线程的问题 弊端:多线程需要判断锁,较为消耗资源
在一个程序中是否有安全问题,如果有,该怎么解决?
1.明确这些代码是多线程运行代码,
2.明确共享数据
3.明确多线程运行代码中那些数据是操作共享数据的,
多线程的业务处理:,业务有可能是产生一个生产者,还有可能出来多个消费者,多个线程的使用,但是这是为什么呢?
又有可能是一个消费者,,两个生产者,但是消费者只消费了一个, 剩下一个就不知道怎么弄了
生产者和消费者,都有两个,t1--t2--t3--t4
stop方法已过时
如何停止线程? 有一种,run方法结束.开启多线程运行,运行代码通常是循环结构,只有控制住循环,就可以让run方法结束,也就是线程结束
特殊方法当线程处于冻结状态,就不会读取到标记,那么线程就不结束如果线程在调用Object类wait(),wait(long)或者wait(long,int),或者该类的join(long,int)或者sleep(long,int),方法程序中受阻则其终端状态将被清除,还会收到一个异常
中断状态,不是停止线程,唤醒时会发生异常,中断异常,冻结状态强制被清除
线程分为,前台线程和后台线程,当前台线程关闭,后台线程就会自动关闭
守护线程:setDaemon将该线程标记为守护线程或用户线程
先开启两主线程, 这个时候的执行权还在主线程手里,当主线程碰到 join() 方法时主线程就会停下来,冻结,主线程碰到谁的join, 它就等谁
JOIN()特点,当A线程执行到了B线程的.join()方法时,A就会等待,等B线程执行都完毕 A才会执行
join 可以用来临时加入线程执行 当它结束了,主线程才会结束,就是等待该线程的join挂了
当主线程在运行时我们可以临时加入一个线程,让这个线程运行完,在执行下面的线程
toString 方法返回该线程的字符串表示形式,包括线程名称,优先级和线程组,优先级,表示CPU的执行权
setPriority()方法,更改线程的优先级,优先级一共就十级,一到十,默认是五级
yield:暂停当前正在执行的线程对象,并执行其他线程
我们学了多线程,但是在什么时候我们是怎么使用多线程的?
当某些代码,需要同时被执行时,就用单独的线程去封装