多线程面试题-1

如何在两个线程之间共享数据

如果每个线程执行的代码执行的代码相同,那么可以使用同一个runnable对象,这个runnable里面的数据共享。
如果每个线程执行的代码不同,这时候需要用不同的Runnable对象。

锁&线程&线程池

死锁、活锁、饥饿、无锁
volatile 会禁止语义重排序
CyclicBarrier 可 重 用 , CountDownLatch 不可重用

原子性是指一个线程的操作是不能被其他线程打断,同一时间只有一个线程对一个 变量进行操作。在多线程情况下,每个线程的执行结果不受其他线程的干扰。
可见性是指某个线程修改了某一个共享变量的值,而其他线程是否可以看见该共享变量修改后的值。像 CPU 的缓存优化、硬件优化、指令重排及对 JVM 编译器的优化,都会出现可见性的问题。
指令重排只会保证当前线程执行结果一致,但指令重排后势必会影响多线程的 执行结果。虽然重排序优化了性能,但也是会遵守一些规则的,并不能随便乱排序, 只是重排序会影响多线程执行的结果。

守护线程就是守护用户线程,当用户线程全部执行完结束之后,守护线程才会跟着结束。也就是守护线程必 须伴随着用户线程,如果一个应用内只存在一个守护线程,没有用户线程,守护线 程自然会退出。
Yield 方法可以暂停当前正在执行的线程对象,让其它有相同优先级的线程执行。 它是一个静态方法而且只保证当前线程放弃 CPU 占用而不能保证使其它线程一定 能占用 CPU,执行yield()的线程有可能在进入到暂停状态后马上又被执行。
重入锁,指的是以线程为单位,当一个线程获取对象锁之后,这个线程可以再 次获取本对象上的锁,而其他的线程是不可以的。
CAS,全称为 Compare and Swap,即比较-替换。假设有三个操作数:内存值 V、 旧的预期值 A、要修改的值 B,当且仅当预期值 A 和内存值 V 相同时,才会将内 存值修改为 B 并返回 true,否则什么都不做并返回 false。当然 CAS 一定要 volatile 变量配合。
java.lang.Thread中有一个方法叫holdsLock(),它返回true如果当且仅当当前线程拥有某个具体对象的锁。
ThreadLocal 的作用是提供线程内的局部变量,这种变量在线程的生命周期内起作 用,减少同一个线程内多个函数或者组件之间一些公共变量的传递的复杂度。用来 解决数据库连接、Session 管理等。

不可变对象保证了对象的内存可见性,对不可变对象的读取不需要进行额外的同步 手段,提升了代码执行效率。
多线程的上下文切换是指 CPU 控制权由一个已经正在运行的线程切换到另外一个 就绪并等待获取 CPU 执行权的线程的过程。
如果线程是因为调用了 wait()、sleep()或者 join()方法而导致的阻塞,可以中断线程, 并且通过抛出 InterruptedException 来唤醒线程;如果线程遇到了 IO 阻塞,无能为力, 因为 IO是操作系统实现的,Java 代码并没有办法直接接触到操作系统。

Thread.sleep(0)

由于 Java 采用抢占式的线程调度算法,因此可能会出现某条线程常常获取到 CPU 控制权的情况,为了让某些优先级比较低的线程也能获取到 CPU 控制权,可以使用Thread.sleep(0)手动触发一次操作系统分配时间片的操作,这也是平衡 CPU 控制 权的一种操作。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值