并发和并行
并发:指单位时间内运行的线程数
并行:真正意义上的同时运行
多线程的优点
1、充分利用CPU资源
2、代码模块化
使用多线程需要注意的问题
1、线程安全问题
2、线程死锁问题
3、线程太多会是服务器宕机问题
线程的实现
Java的Thread类中明确说明为两种方式:
第一种、继承Thread类
第二种、实现Runnable接口
Thread类是对线程的抽象,而Runnable是对任务或者 说是业务逻辑的抽象。而Runnable接口方法的执行也是通过将Runnable接口传入到Thread类进行执行。
线程中断
线程中断的几种方法:
第一种、suspend()、resume() 和 stop()等,强制中断,不管线程是否释放资源都会强制将线程中断,但是java将其标志为过期了,需要慎用,因为使用不好,会出现问题。例如:当我们进行写入文件操作的时候,如果强制中断则文件未写完就会出现问题,或者数据库插入或者修改的时候,如果没有执行完成,就强行关闭可能会导致死锁等问题。
第二种、interrupt(),设置中断标志位,是否中断由线程自己决定。表明线程是协作式的,而不是抢占式(强制终结),然后通过isInterrupt()和interrupted()方法检验中断标志位,但是interrupted会在判断标志位位true时将其改为false.
第三种、自己在线程类中设置中断标记,但是这种方式不建议使用。因为 run 方法里有阻塞调 用时会无法很快检测到取消标志,线程必须从阻塞调用返回后,才会检查这个取 消标志。这种情况下,使用中断会更好,因为,
一、一般的阻塞方法,如 sleep 等本身就支持中断的检查,
二、检查中断位的状态和检查取消标志位没什么区别,用中断位的状态还可 以避免声明取消标志位,减少资源的消耗。 注意:处于死锁状态的线程无法被中断
JAVA中线程的认识
run()和start()方法
1、Thread类是对线程的抽象,但是还没有真正交给操作系统管理,而start()方法才是真正意义上将线程与操作系统绑定。并且同一个线程实例不可以被多次start()否则会报错。我们可以在图中的start()方法的源码可以看到,他会判断线程的状态,如果线程已经被start()了,会抛出一个异常。
2、run()类似于普通类中的一个方法来实现业务逻辑
join()方法
这个方法j和这个英文单词的描述一样,插队,当我们在A线程里执行了join()B线程,这样只有在B执行完之后,A才会被执行.通过join()方法可以实现不同线程之间顺序执行。
yield()方法
这个方法是让出当前线程的CPU占用权,CPU会将当前线程以及其他线程通过时间片轮转算法就行再分配,还有可能继续将CPU分给该线程,注意:yield()方法只是让出CPU的占用权,但是不会释放其他资源,例如:锁、IO等等。
线程的优先级
在 Java 线程中,通过一个整型成员变量 priority 来控制优先级,优先级的范 围从 1~10,在线程构建的时候可以通过 setPriority(int)方法来修改优先级,默认 优先级是 5,优先级高的线程分配时间片的数量要多于优先级低的线程。(值越大优先级越高)
设置线程优先级时,针对频繁阻塞(休眠或者 I/O 操作)的线程需要设置较 高优先级,而偏重计算(需要较多 CPU 时间或者偏运算)的线程则设置较低的 优先级,确保处理器不会被独占。在不同的 JVM 以及操作系统上,线程规划会 存在差异,有些操作系统甚至会忽略对线程优先级的设定。
守护线程
器不会被独占。在不同的 JVM 以及操作系统上,线程规划会 存在差异,有些操作系统甚至会忽略对线程优先级的设定。
守护线程
一般用于资源调度,底层代码常用,但是平时用不到,由于守护线程随着主线程的死亡而死亡,因此,不能依靠finally关键字来做释放资源等事情。