Java线程那点事

1、如何控制线程执行的顺序?
第一种办法是通过join方法去保证多线程的顺序性的特性
join:让主线程等待子线程结束以后才能够进行运行

//
static Thread thread1 = new Thread(){
        @Override
        public void run() {
            System.out.println("thread 1");
        }
    };
    static Thread thread2 = new Thread(){
        @Override
        public void run() {
            System.out.println("thread 2");
        }
    };
    static Thread thread3 = new Thread(){
        @Override
        public void run() {
            System.out.println("thread 3");
        }
    };
    public static void main(String[] args) throws InterruptedException {
        thread1.start();
        thread1.join();
        thread2.start();
        thread2.join();
        thread3.start();
    }

运行结果:
thread 1
thread 2
thread 3

第二种是通过ExecutorService这个方法,这是Java5的时候出来的。

static Runnable runnable1=()->{

        System.out.println("RunningfromLambda 1");

    };
    static Runnable runnable2=()->{

        System.out.println("RunningfromLambda 2");

    };
    static Runnable runnable3=()->{

        System.out.println("RunningfromLambda 3");

    };
    static ExecutorService executorService = Executors.newSingleThreadExecutor();
    
    public static void main(String[] args) {
        executorService.submit(runnable1);
        executorService.submit(runnable2);
        executorService.submit(runnable3);
        executorService.shutdown();
    }

运行结果:
RunningfromLambda 1
RunningfromLambda 2
RunningfromLambda 3

2、Java中Volatile和Synchronized的区别?
首先什么是JMM(Java Memory Model)?
它主要是用来处理并发过程中如何处理可见性、原子性、有序性的问题。

并发编程中有两个关键问题
a.线程之间如何通信 ;
方法:wait() 、notify()、notifyall()
线程通信有两种,一种是共享内存(隐式通信),第二种是消息传递(显示通信)。

b.线程之间如何同步
在共享内存的并发模型中,同步是显示做的;可以通过Synchronized线程排队访问做到。
在消息传递的并发模型中,由于消息的发送必须在消息接收之前,所以同步是隐式。

3、定位内存可见性问题
什么是对象内存共享,什么不是?
每一个线程都会有一个私有的本地内存,例如线程A想和线程B进行通信,线程A会先把本地内存的信息共享到主内存的共享变量里面去,然后B会到主内存里面去拿这个共享变量到B的本地内存,这个过程就完成了一个线程通信。

那么线程A什么时候去刷新共享这个内存?线程B什么时候去主内存里面去拿,这就是线程安全的核心问题。
那么怎么去解决这个问题呢?
通过Volatile和Synchronized解决。
那么使用Volatile会发生什么呢?

1、对于使用了Volatile的变量进行写操作的时候,JVM会向处理器发送一条Lock前缀的指令。会把这个变量所在缓存行的数据写回到系统内存
2、在多多处理器的情况下,保证各个处理器缓存一致性的特点,就回实现缓存一致性协议。就是每个处理器会嗅探到总线传过的数据进行检测,自己缓存的值是不是过期了,当处理器发现自己缓存的内存地址被修改以后,它就会把当前缓存的处理器设置为失效状态,这个时候处理器对它进行修改的时候,它会重新去把这个值读准确,读到自己的处理器缓存里面。

Volatile:可以做到原子性、可见性;不能做到复合操作的原子性。

synchronized:可重入锁、互斥性、可见性
我们来编译一下这段代码,看下会有什么东西

public class App {

    public static void main(String[] args) {
        synchronized(App.class){

        }
    }
    public static synchronized void test(){

    }
}

{
  public com.tuling.springcloud.bean.App();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 6: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lcom/tuling/springcloud/bean/App;

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=3, args_size=1
         0: ldc           #2                  // class com/tuling/springcloud/bean/App
         2: dup
         3: astore_1
         4: monitorenter(synchronized核心)
         5: aload_1
         6: monitorexit(synchronized的核心)
         7: goto          15
        10: astore_2
        11: aload_1
        12: monitorexit
        13: aload_2
        14: athrow
        15: return
      Exception table:
         from    to  target type
             5     7    10   any
            10    13    10   any
      LineNumberTable:
        line 9: 0
        line 11: 5
        line 12: 15
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      16     0  args   [Ljava/lang/String;
      StackMapTable: number_of_entries = 2
        frame_type = 255 /* full_frame */
          offset_delta = 10
          locals = [ class "[Ljava/lang/String;", class java/lang/Object ]
          stack = [ class java/lang/Throwable ]
        frame_type = 250 /* chop */
          offset_delta = 4

  public static synchronized void test();
    descriptor: ()V
    flags: ACC_PUBLIC, ACC_STATIC, ACC_SYNCHRONIZED
    Code:
      stack=0, locals=0, args_size=0
         0: return
      LineNumberTable:
        line 15: 0
}

4: monitorenter(synchronized核心)
6: monitorexit(synchronized的核心)
在这里插入图片描述
它会先去获得一个锁,如果成功了然后再去调用这个对象,如果失败了进入一个队列,开始执行下一个线程。

Lock和Synchronized的区别
1、Synchronized线程执行出现异常的时候释放

2、Synchronized的缺陷
Lock可以主动释放锁,而Synchronized是被动的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我! 毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值