Java多线程面试题
2016-07-26 20:41
225 查看
1.线程与进程的区别是什么:
1)线程是进程的子集,一个进程可以有很多线程
2)所有的线程共享一片相同的内存空间,每个线程都拥有单独的
栈内存用来存储本地数据
3)进程是资源分配和拥有的单位,线程是处理器调度的基本单位
4)二者均可并发执行
2.Thread
类中的 start () 和 run () 方法有什么区别?
当你调用
run ()方法的时候,只会是在原来的线程中调用,没有新
的线程启动,start
()方法才会启动新线程
3.Java
中 Runnable 和 Callable 有什么不同?
它们的主要区别是
Callable 的 call () 方法可以返回值和抛出异
常,而
Runnable 的 run ()方法没有这些功能。Callable 可以返
[b]回装载有计算结果的
Future 对象。
[/b]
4.([b]JMM)Java
内存模型是什么?[/b]
http://blog.csdn.net/zgmzyr/article/details/8798568
线程A与线程B之间如要通信的话,必须要经历下面2个步骤
首先,线程A把本地内存A中更新过的共享变量刷新到主内存中去。
然后,线程B到主内存中去读取线程A之前已更新过的共享变量。
JMM通过控制主内存与每个线程的本地内存之间的交互,来为java程序员提供内存可见性保证。
5.java中的Java
中的 volatile 变量是什么?
Volatile保证可见性Volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。
6.什么是线程安全?
如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的。
Vector
是用同步方法来实现线程安全的, 而和它相似的 ArrayList 不是线程安全的。
7. 如何在两个线程间共享数据?
(涉及到在两个线程间共享对象)用
wait 和 notify 方法实现了生产者消费者模型
notify
()方法不能唤醒某个具体的线程,所以只有一个线程在等待
[b]的时候它才有用武之地。而
notifyAll ()唤醒所有线程并允许他们
[/b]
[b]争夺锁确保了至少有一个线程能继续运行
[/b]
[b]8. Java
中的同步集合与并发集合有什么区别?[/b]
[b]Java1.5
之前程序员们只有同步集合来用且在多线程并发的时候会[/b]
[b]导致争用,阻碍了系统的扩展性。Java5
介绍了并发集合像 [/b]
[b]ConcurrentHashMap,不仅提供线程安全还用锁分离和内部分区[/b]
[b]等现代技术提高了可扩展性
[/b]
[b]9. Java
中堆和栈有什么不同?[/b]
[b]个线程都有自己的栈内存,用于存储本地变量,方法参数和栈调用,
[/b]
[b]而堆是所有线程共享的一片公用内存区域
[/b]
[b]10.什么是线程池?
为什么要使用它?[/b]
[b]创建线程要花费昂贵的资源和时间,如果任务来了才创建线程那[/b]
[b]么响应时间会变长,而且一个进程能创建的线程数有限。为了避[/b]
[b]免这些问题,在程序启动的时候就创建若干线程来响应处理,它[/b]
[b]们被称为线程池,里面的线程叫工作线程
[/b]
[b]11.如何写代码来解决生产者消费者问题?[/b]
比较低级的办法是用
wait 和 notify 来解决这个问题,比较赞的办
法是用
Semaphore 或者 BlockingQueue 来实现生产者消费者模
型
12.[b]Java
中 synchronized 和 ReentrantLock 有什么不同?[/b]
Java
在过去很长一段时间只能通过 synchronized 关键字来实现
互斥,它有一些缺点。比如你不能扩展锁之外的方法或者块边
界,尝试获取锁时不能中途取消等。Java
5 通过 Lock 接口提供
了更复杂的控制
13. 有三个线程
T1,T2,T3,怎么确保它们按顺序执行?
可以用线程类的
join ()方法在一个线程中启动另一个线程,另外
一个线程完成该线程继续执行。
14.Thread
类中的 yield 方法有什么作用?
Yield
方法可以暂停当前正在执行的线程对象,让其它有相同优先
级的线程执行。它是一个静态方法而且只保证当前线程放弃
CPU
占用而不能保证使其它线程一定能占用
CPU,执行 yield ()的线
程有可能在进入到暂停状态后马上又被执行
15.Java
中 ConcurrentHashMap 的并发度是什么?
oncurrentHashMap
把实际 map 划分成若干部分来实现它的可
扩展性和线程安全。这种划分是使用并发度获得的,它是
ConcurrentHashMap
类构造函数的一个可选参数,默认值为
16,这样在多线程情况下就能避免争用。
16. Java
多线程中调用 wait () 和 sleep ()方法有什么不同?
wait
()方法用于线程间通信,如果等待条件为真且其它线程被唤
醒时它会释放锁,而
sleep ()方法仅仅释放 CPU 资源或者让当前
线程停止执行一段时间,但不会释放锁。
17.虽然你可以使用
System.gc ()来进行垃圾回收,但是不保证能成功
18.单例模式:
欢双检锁JVM
的类加载和静态变量初始化特征来创建 Singleton
实例
第一种(懒汉,线程不安全):
Java代码
public class Singleton {
private static Singleton instance;
private Singleton (){}
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
第二种(懒汉,线程安全):
Java代码
public class Singleton {
private static Singleton instance;
private Singleton (){}
public static synchronized Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
第四种([b]饿汉,变种):[/b]
Java代码
public class Singleton {
private Singleton instance = null;
static {
instance = new Singleton();
}
private Singleton (){}
public static Singleton getInstance() {
return this.instance;
}
}
第五种(静态内部类):
Java代码
public class Singleton {
private static class SingletonHolder {
private static final Singleton INSTANCE = new Singleton();
}
private Singleton (){}
public static final Singleton getInstance() {
return SingletonHolder.INSTANCE;
}
}
第六种(枚举):
Java代码
public enum Singleton {
INSTANCE;
public void whateverMethod() {
}
}
这种方式是Effective Java作者Josh Bloch 提倡的方式,它不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象,可谓是很坚强的壁垒啊,不过,个人认为由于1.5中才加入enum特性,用这种方式写不免让人感觉生疏,在实际工作中,我也很少看见有人这么写过。
第七种(双重校验锁):
Java代码
public class Singleton {
private volatile static Singleton singleton;
private Singleton (){}
public static Singleton getSingleton() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}
1)线程是进程的子集,一个进程可以有很多线程
2)所有的线程共享一片相同的内存空间,每个线程都拥有单独的
栈内存用来存储本地数据
3)进程是资源分配和拥有的单位,线程是处理器调度的基本单位
4)二者均可并发执行
2.Thread
类中的 start () 和 run () 方法有什么区别?
当你调用
run ()方法的时候,只会是在原来的线程中调用,没有新
的线程启动,start
()方法才会启动新线程
3.Java
中 Runnable 和 Callable 有什么不同?
它们的主要区别是
Callable 的 call () 方法可以返回值和抛出异
常,而
Runnable 的 run ()方法没有这些功能。Callable 可以返
[b]回装载有计算结果的
Future 对象。
[/b]
4.([b]JMM)Java
内存模型是什么?[/b]
http://blog.csdn.net/zgmzyr/article/details/8798568
线程A与线程B之间如要通信的话,必须要经历下面2个步骤
首先,线程A把本地内存A中更新过的共享变量刷新到主内存中去。
然后,线程B到主内存中去读取线程A之前已更新过的共享变量。
JMM通过控制主内存与每个线程的本地内存之间的交互,来为java程序员提供内存可见性保证。
5.java中的Java
中的 volatile 变量是什么?
Volatile保证可见性Volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。
6.什么是线程安全?
如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的。
Vector
是用同步方法来实现线程安全的, 而和它相似的 ArrayList 不是线程安全的。
7. 如何在两个线程间共享数据?
(涉及到在两个线程间共享对象)用
wait 和 notify 方法实现了生产者消费者模型
notify
()方法不能唤醒某个具体的线程,所以只有一个线程在等待
[b]的时候它才有用武之地。而
notifyAll ()唤醒所有线程并允许他们
[/b]
[b]争夺锁确保了至少有一个线程能继续运行
[/b]
[b]8. Java
中的同步集合与并发集合有什么区别?[/b]
[b]Java1.5
之前程序员们只有同步集合来用且在多线程并发的时候会[/b]
[b]导致争用,阻碍了系统的扩展性。Java5
介绍了并发集合像 [/b]
[b]ConcurrentHashMap,不仅提供线程安全还用锁分离和内部分区[/b]
[b]等现代技术提高了可扩展性
[/b]
[b]9. Java
中堆和栈有什么不同?[/b]
[b]个线程都有自己的栈内存,用于存储本地变量,方法参数和栈调用,
[/b]
[b]而堆是所有线程共享的一片公用内存区域
[/b]
[b]10.什么是线程池?
为什么要使用它?[/b]
[b]创建线程要花费昂贵的资源和时间,如果任务来了才创建线程那[/b]
[b]么响应时间会变长,而且一个进程能创建的线程数有限。为了避[/b]
[b]免这些问题,在程序启动的时候就创建若干线程来响应处理,它[/b]
[b]们被称为线程池,里面的线程叫工作线程
[/b]
[b]11.如何写代码来解决生产者消费者问题?[/b]
比较低级的办法是用
wait 和 notify 来解决这个问题,比较赞的办
法是用
Semaphore 或者 BlockingQueue 来实现生产者消费者模
型
12.[b]Java
中 synchronized 和 ReentrantLock 有什么不同?[/b]
Java
在过去很长一段时间只能通过 synchronized 关键字来实现
互斥,它有一些缺点。比如你不能扩展锁之外的方法或者块边
界,尝试获取锁时不能中途取消等。Java
5 通过 Lock 接口提供
了更复杂的控制
13. 有三个线程
T1,T2,T3,怎么确保它们按顺序执行?
可以用线程类的
join ()方法在一个线程中启动另一个线程,另外
一个线程完成该线程继续执行。
14.Thread
类中的 yield 方法有什么作用?
Yield
方法可以暂停当前正在执行的线程对象,让其它有相同优先
级的线程执行。它是一个静态方法而且只保证当前线程放弃
CPU
占用而不能保证使其它线程一定能占用
CPU,执行 yield ()的线
程有可能在进入到暂停状态后马上又被执行
15.Java
中 ConcurrentHashMap 的并发度是什么?
oncurrentHashMap
把实际 map 划分成若干部分来实现它的可
扩展性和线程安全。这种划分是使用并发度获得的,它是
ConcurrentHashMap
类构造函数的一个可选参数,默认值为
16,这样在多线程情况下就能避免争用。
16. Java
多线程中调用 wait () 和 sleep ()方法有什么不同?
wait
()方法用于线程间通信,如果等待条件为真且其它线程被唤
醒时它会释放锁,而
sleep ()方法仅仅释放 CPU 资源或者让当前
线程停止执行一段时间,但不会释放锁。
17.虽然你可以使用
System.gc ()来进行垃圾回收,但是不保证能成功
18.单例模式:
欢双检锁JVM
的类加载和静态变量初始化特征来创建 Singleton
实例
第一种(懒汉,线程不安全):
Java代码
public class Singleton {
private static Singleton instance;
private Singleton (){}
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
第二种(懒汉,线程安全):
Java代码
public class Singleton {
private static Singleton instance;
private Singleton (){}
public static synchronized Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
第四种([b]饿汉,变种):[/b]
Java代码
public class Singleton {
private Singleton instance = null;
static {
instance = new Singleton();
}
private Singleton (){}
public static Singleton getInstance() {
return this.instance;
}
}
第五种(静态内部类):
Java代码
public class Singleton {
private static class SingletonHolder {
private static final Singleton INSTANCE = new Singleton();
}
private Singleton (){}
public static final Singleton getInstance() {
return SingletonHolder.INSTANCE;
}
}
第六种(枚举):
Java代码
public enum Singleton {
INSTANCE;
public void whateverMethod() {
}
}
这种方式是Effective Java作者Josh Bloch 提倡的方式,它不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象,可谓是很坚强的壁垒啊,不过,个人认为由于1.5中才加入enum特性,用这种方式写不免让人感觉生疏,在实际工作中,我也很少看见有人这么写过。
第七种(双重校验锁):
Java代码
public class Singleton {
private volatile static Singleton singleton;
private Singleton (){}
public static Singleton getSingleton() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}
相关文章推荐
- 程序员为什么要时刻保持危机感?
- 面试准备之--桶类排序
- StackOverflow程序员推荐:每个程序员都应读的30本书
- 剑指offer面试题39:二叉树深度以及判断平衡二叉树
- 剑指offer面试题38:数字在排序数组中出现的次数
- 最全前端面试问题及答案总结(转载自hawx1993)
- 数组去重 算法实现
- 面试准备之---排序
- java研发面试准备
- PHP高级程序员必学
- 为什么中国的程序员总被称为码农?
- FAE面试后的自我反思
- iOS新手求职(本人经历)
- 转 展望未来,总结过去10年的程序员生涯,给程序员小弟弟小妹妹们的一些总结性忠告
- .Net程序员应该掌握的正则表达式
- 【.Net码农】C#中的partial class(部分类)
- 国内一线互联网公司内部面试题库
- java面试题(有些是转载)
- 面试准备之--字符串旋转
- 晋升的为什么不是你