Java线程

一、创建线程的两种方式

1. 继承Thread类 类

1) 定义子类继承Thread类。
2) 子类中重写Thread类中的run方法。
3) 创建Thread子类对象,即创建了线程对象。
4) 调用线程对象start方法:启动线程,调用run方法。

2. 实现Runnable 接口

1)定义子类,实现Runnable接口。
2)子类中重写Runnable接口中的run方法。
3)通过Thread类含参构造器创建线程对象。
4)将Runnable接口的子类对象作为实际参数传递给Thread类的构造方法中。
5)调用Thread类的start方法:开启线程,调用Runnable子类接口的run方法。

二、Thread 类的有关方法

1)void start(): 启动线程,并执行对象的run()方法。
2)run(): 线程在被调度时执行的操作。
3)String getName(): 返回线程的名称。
4)void setName(String name):设置该线程名称。
5)static currentThread(): 返回当前线程。
6)static void yield(): : 线程让步

  • 暂停当前正在执行的线程,把执行机会让给优先级相同或更高的线程。
  • 若队列中没有同优先级的线程,忽略此方法。

7)join() : : 当某个程序执行流中调用其他线程的 join() 方法时,调用
线程将被阻塞,直到 join() 方法加入的 join 线程执行完为止。

  • 低优先级的线程也可以获得执行。

8)static void sleep(long millis) :(指定时间:毫秒)

  • 令当前活动线程在指定时间段内放弃对CPU控制,使其他线程有机会被执行,时间到后重排队。

  • 抛出InterruptedException异常。

9)stop(): 强制线程生命期结束。
10)boolean isAlive(): : 返回boolean,判断线程是否还活着。

三、线程的分类

Java中的线程分为两类:一种是 守护线程,一种是 用户线程。

  • 它们在几乎每个方面都是相同的,唯一的区别是判断JVM何时离开。
  • 守护线程是用来服务用户线程的,通过在start()方法前调用thread.setDaemon(true)可以把一个用户线程变成一个守护线程。
  • Java垃圾回收就是一个典型的守护线程。
  • 若JVM中都是守护线程,当前JVM将退出。

四、线程的生命周期

要想实现多线程,必须在主线程中创建新的线程对象。Java语言使用Thread类及其子类的对象来表示线程,在它的一个完整的生命周期中通常要经历如下的 五种状态:

1、新建: 当一个Thread类或其子类的对象被声明并创建时,新生的线程对象
处于新建状态。
2、就绪:处于新建状态的线程被start()后,将进入线程队列等待CPU时间片,
此时它已具备了运行的条件。
3、运行:当就绪的线程被调度并获得处理器资源时,便进入运行状态, run()
方法定义了线程的操作和功能。
4、阻塞:在某种特殊情况下,被人为挂起或执行输入输出操作时,让出 CPU
并临时中止自己的执行,进入阻塞状态。
5、死亡:线程完成了它的全部工作或线程被提前强制性地中止。

五、多线程的安全问题

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

六、锁

1、Synchronized 的使用方法

Java对于多线程的安全问题提供了专业的解决方式: 对于多线程的安全问题提供了专业的解决方式:

同步代码块
  • synchronized ( 对象){
    // 需要被同步的代码;
    }
    synchronized 锁的对象可以是this线程本身,也可以是其他对象。

  • synchronized 还可以放在方法声明中,表示整个方法为同步方法。例如:

public synchronized void show (String name){
    …//方法体
}

2、互斥锁

在Java语言中,引入了对象互斥锁的概念,来保证共享数据操作的完整性。

  • 每个对象都对应于一个可称为“互斥锁”的标记,这个标记用来保证在任一时刻,只能有一个线程访问该对象。
  • 关键字synchronized 来与对象的互斥锁联系。当某个对象用synchronized修饰时,表明该对象在任一时刻只能由一个线程访问。
  • 同步的局限性:导致程序的执行效率要降低。
  • 同步方法(非静态的)的锁为this。
  • 同步方法(静态的)的锁为当前类本身。
    单例模式就是很好的例子:
class Singleton {
    private static Singleton instance = null;
    private Singleton(){}
    public static Singleton getInstance(){
        if(instance==null){
            synchronized(Singleton.class){
                if(instance == null){
                    instance=new Singleton();
                } 
            } 
        }
        return instance;
    }  
}
class TestSingleton{
    public static void main(String[] args){
        Singleton s1=Singleton.getInstance();
        Singleton s2=Singleton.getInstance();
        System.out.println(s1==s2);
    } 
}

3、释放锁的操作

  • 当前线程的同步方法、同步代码块执行结束。
  • 当前线程在同步代码块、同步方法中遇到break、return终止了该代码块、该方法的继续执行。
  • 当前线程在同步代码块、同步方法中出现了未处理的Error或Exception,导致异常结束。
  • 当前线程在同步代码块、同步方法中执行了线程对象的wait()方法,当前线程暂停,并释放锁。

4、不会释放锁的操作

  • 线程执行同步代码块或同步方法时,程序调用Thread.sleep()、Thread.yield()方法暂停当前线程的执行。
  • 线程执行同步代码块时,其他线程调用了该线程的suspend()方法将该线程挂起,该线程不会释放锁(同步监视器)。
    注意:应尽量避免使用suspend()和resume()来控制线程。

5、线程的死锁问题

1、 死锁

  • 不同的线程分别占用对方需要的同步资源不放弃,都在等待对方放弃自己需要的同步资源,就形成了线程的死锁。

2、 解决方法

  • 专门的算法、原则
  • 尽量减少同步资源的定义

七、线程通信

wait() 与 notify() 和 notifyAll()
  • wait():令当前线程挂起并放弃CPU、同步资源,使别的线程可访问并修改共享资源,而当前线程排队等候再次对资源的访问。
  • notify():唤醒正在排队等待同步资源的线程中优先级最高者结束等待。
  • notifyAll ():唤醒正在排队等待资源的所有线程结束等待。
  • 注意:Java.lang.Object提供的这三个方法只有在synchronized方法或synchronized代码块中才能使用,否则会报java.lang.IllegalMonitorStateException异常。并且当前线程必须具有对该对象的监控权(加锁)。
1、wait()方法
  • 在当前线程中调用方法: 对象名.wait()。
  • 使当前线程进入等待(某对象)状态 ,直到另一线程对该对象发出 notify (或notifyAll) 为止。
  • 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)。
  • 调用此方法后,当前线程将释放对象监控权 ,然后进入等待
  • 在当前线程被notify后,要重新获得监控权,然后从断点处继续代码的执行。
2、notify()/notifyAll()
  • 在当前线程中调用方法: 对象名.notify()。
  • 功能:唤醒等待该对象监控权的一个线程。
  • 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值