java的run代码 java runoob( 六 )


41. sleep 和 wait 有什么区别?
sleep:方法是线程类(Thread)的静态方法,让调用线程进入睡眠状态,让出执行机会给其他线程 , 等到休眠时间结束后,线程进入就绪状态和其他线程一起竞争cpu的执行时间 。因为sleep 是static静态的方法,他不能改变对象的机锁,当一个synchronized块中调用了sleep 方法,线程虽然进入休眠,但是对象的机锁没有被释放,其他线程依然无法访问这个对象 。
wait:wait是Object类的方法,当一个线程执行到wait方法时 , 它就进入到一个和该对象相关的等待池,同时释放对象的机锁,使得其他线程能够访问,可以通过notify,notifyAll方法来唤醒等待的线程
42. notify和 notifyAll有什么区别?
如果线程调用了对象的 wait方法,那么线程便会处于该对象的等待池中,等待池中的线程不会去竞争该对象的锁 。
当有线程调用了对象的 notifyAll方法(唤醒所有 wait 线程)或 notify方法(只随机唤醒一个 wait 线程),被唤醒的的线程便会进入该对象的锁池中,锁池中的线程会去竞争该对象锁 。也就是说,调用了notify后只要一个线程会由等待池进入锁池,而notifyAll会将该对象等待池内的所有线程移动到锁池中,等待锁竞争 。
优先级高的线程竞争到对象锁的概率大,假若某线程没有竞争到该对象锁,它还会留在锁池中,唯有线程再次调用 wait方法,它才会重新回到等待池中 。而竞争到对象锁的线程则继续往下执行,直到执行完了 synchronized 代码块,它会释放掉该对象锁,这时锁池中的线程会继续竞争该对象锁 。
43. 线程的 run和 start有什么区别?
每个线程都是通过某个特定Thread对象所对应的方法run来完成其操作的 , 方法run称为线程体 。通过调用Thread类的start方法来启动一个线程 。
start方法来启动一个线程,真正实现了多线程运行 。这时无需等待run方法体代码执行完毕 , 可以直接继续执行下面的代码;这时此线程是处于就绪状态,并没有运行 。然后通过此Thread类调用方法run来完成其运行状态,这里方法run称为线程体,它包含了要执行的这个线程的内容,Run方法运行结束,此线程终止 。然后CPU再调度其它线程 。
run方法是在本线程里的,只是线程里的一个函数,而不是多线程的 。如果直接调用run,其实就相当于是调用了一个普通函数而已,直接待用run方法必须等待run方法执行完毕才能执行下面的代码,所以执行路径还是只有一条,根本就没有线程的特征,所以在多线程执行时要使用start方法而不是run方法 。
44. 创建线程池有哪几种方式?
①. newFixedThreadPool(int nThreads)
创建一个固定长度的线程池,每当提交一个任务就创建一个线程,直到达到线程池的最大数量,这时线程规模将不再变化,当线程发生未预期的错误而结束时,线程池会补充一个新的线程 。
②. newCachedThreadPool
创建一个可缓存的线程池 , 如果线程池的规模超过了处理需求,将自动回收空闲线程,而当需求增加时,则可以自动添加新线程,线程池的规模不存在任何限制 。
③. newSingleThreadExecutor
这是一个单线程的Executor , 它创建单个工作线程来执行任务,如果这个线程异常结束,会创建一个新的来替代它;它的特点是能确保依照任务在队列中的顺序来串行执行 。
④. newScheduledThreadPool(int corePoolSize)
创建了一个固定长度的线程池,而且以延迟或定时的方式来执行任务,类似于Timer 。
45. 线程池都有哪些状态?
线程池有5种状态:Running、ShutDown、Stop、Tidying、Terminated

推荐阅读