java线程笔记

进程与线程的区别

进程是指正在运行的程序,几乎所有的操作系统都支持进程的概念,所有运行的任务通常对应一个进程,当一个程序进入内存执行,即变成了一个进程,进程是处于运行中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位。线程是进程的执行单元,一个进程可以拥有多个线程,多个线程之间共享进程所拥有的全部资源。一个进程中的多个线程可以并发执行,线程的执行是抢占式的,线程的调度和管理由进程本身负责完成,
进程之间不能共享内存,但线程之间共享内存很容易,
系统创建进程是需要为进程分配系统资源,但创建线程则代价小得多

java中创建线程的方式
  1. 通过继承Thread类
  • 定义继承Thread类的子类,重写run()方法,run方法即为线程执行体
  • 创建该子类的实例对象,即创建了线程对象
  • 调用该对象的start()方法启动执行该线程
  1. 通过实行Runnable接口
  • 定义实现Runnable接口的类,同样实现run()方法
  • 创建该类的实例对象,以该对象作为Thread的target创建Thread实例对象,该Thread对象才是真正的线程对象
  • 调用该Thread对象的start() 方法启动线程
  1. 使用callable接口和FutureTask类
  • 创建Callbale接口的实现类,并且实现call()方法,并且该call方法有返回值
  • 创建Callable的实例对象,并用FutureTask类来包装Callable对象,该FutureTask对象封装了callable对象的call方法的返回值
  • 使用FutureTask对象作为Thread对象的target创建并启动线程
  • 调用FutureTask对象的get方法来获得子线程执行结束后的返回值
线程创建方式对比

采用接口的方式创建线程打破了类单继承的局限,还可以继承其他类,只此一点 推荐使用接口方式
同样 采用接口实现方式多个进程之间可以共享一个target对象,适合多个线程来处理同一份资源的情况

线程同步安全问题

采用同步代码块synchornized(锁定对象)
或者同步方法 public synchronized 返回值 方法名(参数){//code}
或者同步锁(Lock) 定义一个锁对象 在方法的开头加锁,方法结尾 解锁

线程睡眠方法sleep() 与线程让步yield()方法的不同

sleep方法会将线程进入阻塞状态,直到经过阻塞时间才会转人就绪状态,而yield方法不会将线程阻塞,只是将线程强制进入就绪状态,让线程调度器重新调度一次,完全有可能某个线程调用的yield暂停之后,立即再次获得处理器资源被执行。
sleep方法会调用抛出异常所以要么捕捉该异常要么抛出异常,而调用yield方法没有声明抛出任何异常。

线程通信可以借助于Object类的wait() notify() 和 notifyAll()方法

wait() 方法导致当前线程等待,直到其他线程调用该同步监视器的notify()方法或notifyAll()方法唤醒该线程,调用wait方法会释放对当前同步监视器的锁定
notify() 唤醒在此同步监视器上等待的单个线程,如果所有线程都在此同步监视器上等待,则会唤醒其中一个线程,选择是任意的,只有当前线程放弃对同步监视器的锁定后(即调用wait()方法),才可以执行被唤醒的线程
notifyAll() 唤醒在此同步监视器上等待的所有线程,只有当前线程放弃对同步监视器的锁定后,才可以执行被唤醒的线程

经典的生产者消费者例子

通过使用一个flag标识判断是否可以生产 是否可以消费
还可以通过阻塞队列控制线程通信 BlockingQueue

包装线程不安全的集合

通过Collections提供的静态方法将这些集合包装成线程安全的集合,比如ArrayList HashSet TreeSet HashMap TreeMap 等都是线程不安全的

你可能感兴趣的:(java线程笔记)