java线程怎么加锁?java线程生命周期是怎样的?

阳光 2022-04-01 21:35:31 java常见问答 7719

程序员在开发Java多线程应用程序中,各个线程之间由于要共享资源,必须用到锁机制,可是对于一个新手java程序员来说不清楚java线程怎么加锁?下面来我们就来给大家讲解一下java线程加锁方法。

1.对象锁(同步块)——锁某一个对象

对象锁:顾名思义给对象上锁

当A线程访问一个object的时候,首先会获取该对象的对象锁,然后访问锁定的代码,而B线程访问一个对象object多顶的代码区时,因为线程A之前获取到对象object的对象锁,因此线程B访问同步代码区的时候会被阻塞,直到线程A执行完同步代码块,然后释放对象锁,B线程获取对象object的对象锁,才能进行访问同步代码块

用法:synchronized (this){

//—-同步代码块———

}

this表示当前访问同步代码区的对象

特别注意:只要同步代码区的是同一个对象时才能实现互斥访问的效果

示例:

package com.demo;
public class SyncBlock
{
    public static int count;
    public static Object lock = new Object();
    public static void main(String[] args) throws InterruptedException
    {
        Thread t1 = new Thread(new MyRunnable(), "线程1");
        Thread t2 = new Thread(new MyRunnable(), "线程2");
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("count=" + count);
    }
    static class MyRunnable implements Runnable
    {
        @Override
        public void run()
        {
            for (int i = 0; i < 10000; i++)
            {
                synchronized(lock)
                {
                    count++;
                }
            }
        }
    }
}

2.类锁——锁当前类(方法上有static关键字修饰)

类锁:顾名思义给类上锁–>具体就是给类的字节码上锁

当线程A访问类锁锁定的同步代码块时,线程B访问同步代码块,因为A之前已经获取类锁,因此线程B只能A执行完同步代码块后才能获取类锁进行访问

示例:

package com.demo;
public class SyncClass
{
    private static int count;
    public static void main(String[] args) throws InterruptedException
    {
        Thread t1 = new Thread(new MyRunnable());
        Thread t2 = new Thread(new MyRunnable());
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(count);
    }
    static class MyRunnable implements Runnable
    {
        public static synchronized void increase()
        {
            count++;
        }
        @Override
        public void run()
        {
            for (int i = 0; i < 10000; i++)
            {
                increase();
            }
        }
    }
}

结果:同上

3.实例锁——锁的是当前实例(方法上没有static修饰,表示只能锁当前实例)

同一个实力访问的线程才是安全的,不同的线程访问则不安全

示例:

package com.demo;
/**
* @author dxchen
*
*/
public class SyncInstance
{
    public static int count;
    public static void main(String[] args) throws InterruptedException
    {
        MyRunnable target = new MyRunnable();
        Thread t1 = new Thread(target, "线程1");
        Thread t2 = new Thread(target, "线程2");
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(count);
    }
    static class MyRunnable implements Runnable
    {
        /**
        * 对count操作的实例方法
        */
        public synchronized void increase()
        {
            count++;
        }
        @Override
        public void run()
        {
            for (int i = 0; i < 10000000; i++)
            {
                increase();
            }
        }
    }
}

java线程生命周期是怎样的?

下图显示了一个线程完整的生命周期。

java线程怎么加锁?java线程生命周期是怎样的?.jpg

1.新建状态

使用 new 关键字和 Thread 类或其子类建立一个线程对象后,该线程对象就处于新建状态。它保持这个状态直到程序 start() 这个线程。

2.就绪状态

当线程对象调用了start()方法之后,该线程就进入就绪状态。就绪状态的线程处于就绪队列中,要等待JVM里线程调度器的调度。

3.运行状态

如果就绪状态的线程获取 CPU 资源,就可以执行 run(),此时线程便处于运行状态。处于运行状态的线程最为复杂,它可以变为阻塞状态、就绪状态和死亡状态。

4.阻塞状态

如果一个线程执行了sleep(睡眠)、suspend(挂起)等方法,失去所占用资源之后,该线程就从运行状态进入阻塞状态。在睡眠时间已到或获得设备资源后可以重新进入就绪状态。可以分为三种:

5.等待阻塞

运行状态中的线程执行 wait() 方法,使线程进入到等待阻塞状态。

6.同步阻塞

线程在获取 synchronized 同步锁失败(因为同步锁被其他线程占用)。

7.其他阻塞

通过调用线程的 sleep() 或 join() 发出了 I/O 请求时,线程就会进入到阻塞状态。当sleep() 状态超时,join() 等待线程终止或超时,或者 I/O 处理完毕,线程重新转入就绪状态。

8.死亡状态:

一个运行状态的线程完成任务或者其他终止条件发生时,该线程就切换到终止状态。

其实java线程是一个动态执行的过程,它也是有一个从产生到死亡的过程,直到最后死亡状态就是线程终止。最后大家如果想要了解更多java入门知识,敬请关注奇Q工具网。

推荐阅读:

Java程序员有必要再学一门前端吗?Java如何学好前端?

mybatis工作原理会调用get方法吗?mybatis支持延迟加载吗?

java线程怎么做标记?java线程如何同步?