多线程


1.进程与线程

进程:
  是一个正在执行中的程序。
  每一个进程执行都有一个执行顺序。该顺序是一个执行路径,或者叫一个控制单元。
线程:
  就是进程中的一个独立的控制单元。
  线程在控制着进程的执行。

    一个进程中至少有一个线程。

    每当java命令执行一个类的时候,实际上都会起动一个jvm(java 虚拟机),每一个jvm在操纵系统中起动了一个进程java.exe.该进程中至少一个线程负责java程序的执行;而且这个线程运行的代码存在于main方法中。
该线程称之为主线程。
【注意:】主线程也有可能在子线程结束之前结束。并且子线程不受影响,不会因为主线程的结束而结束。

扩展:
    其实更细节说明jvm,jvm启动不止一个线程,还有负责垃圾回收机制的线程。

2.线程的创建

  2.1创建线程的两种方法
    通过对api的查找,java已经提供了对线程这类事物的描述。就是Thread类(该类实现了Runnable接口)。
创建线程有两种方法:一种是继承Thread类,另一种是实现Runnable接口。
   一、创建线程的第一种方式:继承Thread类。
 步骤:
   1、定义类继承Thread。
   2、复写Thread类中的run方法。
  目的:将自定义的代码存储在run方法中,让线程运行。
   3、调用线程的start方法,该方法有两个作用:启动线程、调用run方法。

  二、另一种方法:实现Runnable接口。
   1、定义类实现Runnable接口。
   2、覆盖接口中的run方法
  将线程要运行的代码存放在该run方法中。
   3、通过Thread类建立线程对象
   4、将Runnable接口的子类对象作为实际参数传递给Thread类的构造函数。
  为什么要将Runnable接口的子类对象传递给Thread的构造函数
  因为自定义的run方法所属的对象是Runnable接口的子类对象。
  所以要让线程去执行对象的run方法。就必须明确该run方法的所属对象。
 5、调用Thread类的start方法开启线程并调用Runable接口子类的run方法。
【疑问:】为什么我们不能直接调用run()方法呢?
 因为线程的运行需要本地操作系统的支持。start()方法中调用了private native void start0(),native表示调用了本地操作系统的函数。
2.2实现方式和继承方式的区别:
 
 一、如果一个类继承Thread,则不适合资源共享。但是如果实现了Runable接口的话,则很容易的实现资源共享。
 二、实现方式避免了单继承的局限性,在定义线程时,建议使用实现方式。
 三、增加程序的健壮性,代码可以被多个线程共享,代码和数据独立。
2.3多线程的随机性
发现运行结果每一次都不一样。
因为多个线程都获取cpu的执行权。cpu执行到谁,谁就运行。
我们可以形象的把多线程的运行称为在互相抢夺cpu的执行权。
这就是多线程的一个特性:随机性。

2.4线程对象以及名称
线程都有自己的默认的名称。
:Thread-编号 该编号从0开始。

static Thread currentThread();获取当前线程对象。
setName();设置线程名称,或者构造函数。
getName();获取线程的名称。

3.多线程的安全问题

问题的原因:
 当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,另一个线程参与进来执行。导致共享数据的错误。
解决办法:
 对多条操作共享数据的语句,只能让一个线程都执行完。在执行过程中,其他线程不可以参与执行。

 java对于多线程的安全问题提供了专业的解决办法:就是同步代码块。
4.同步

4.1同步
synchronized(对象)
{
 同步代码块
}
对象如同锁,持有锁的线程可以在同步中执行。
没有持有锁的线程即使获取cpu的执行权,也进不去,因为没有获取锁。
【同步前提】

1、多线程,必须两个或者两个以上的线程。
2、必须多个线程使用同一个锁。

好处:解决了多线程的安全问题。
弊端:多个线程需要判断锁,较为消耗资源。


【同步函数】
同步函数用的是哪一个锁?
函数需要被对象调用,那么函数都有一个所属对象引用,就是this。
所以同步函数使用的锁是this。

【静态同步函数】
静态同步函数的锁是.class对象。
类名.class   类型为Class

静态同步函数使用的锁就是该方法所在类的字节码文件对象。

4.2死锁

原因:同步中嵌套同步。

5.线程间通信
思考1:wait(),notify(),notifyAll(),用来操作线程为什么定义在Object类中?
 1、这些方法存在与同步中。
  因为要对持有监视器(锁)的线程操作,而只有同步才具有锁。
 2、使用这些方法时必须要标识所属的同步的锁。
  只有同一个锁上的等待线程可以被同一个锁上的notify唤醒。也就是说等待和唤醒必须是同一个锁。
 3、锁可以是任意对象,所以任意对象调用的方法一定定义在Object中。

思考2:wait(),sleep()有什么区别?

 wait():释放资源,释放锁。
 sleep():释放资源,不释放锁。
6.Lock锁


jdk1.5中提供了多线程升级解决方案。
将同步synchronized替换成显示Lock操作。
将Object中的wait,notify ,notifyAll,替换了Condition对象。
该对象可以Lock锁中的   newCondition方法进行获取。
可以实现只唤醒对方操作。

Lock 实现提供了比使用 synchronized 方法和语句可获得的更广泛的锁定操作。此实现允许更灵活的结构,可以具有差别很大的属性,可以支持多个相关的 Condition 对象。

随着灵活性的增加,也带来了更多的责任。不使用块结构锁就失去了使用 synchronized 方法和语句时会出现的锁自动释放功能。所以一定要执行释放锁功能:unlock();
   如:
  Lock l = ...;
  l.lock();
  try {
    // access the resource protected by this lock
  } finally {
   l.unlock();
  }
7.停止线程
stop方法已经过时。

如何停止线程?
只有一种:run方法结束。
开启多线程运行,运行代码通常是循环结构。

只要控制住循环,就可以让run方法结束,就是线程结束。


wait() 当线程处于了冻结状态。就不会读取到标记,那么线程就不会结束。
当没有指定的方式让冻结的线程恢复到运行状态时,这时需要对冻结状态进行清除。

强制让线程恢复到运行状态中来,这样就可以操作标记让线程结束。
Thread中提供了该方法interrupt();

8.Thread中的其他方法

【setDaemon()】
 
 将该线程标记为守护线程或用户线程。当正在运行的线程都是守护线程时,Java 虚拟机退出。
该方法必须在启动线程前调用。

【join()】
等待该线程终止。

当A线程执行到了B线程的join方法时,A线程就会等待,等B线程都执行完,A才会执行。
join可以用来临时加入线程执行。

【yield()】
暂停当前正在执行的线程对象,并执行其他线程。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值