java线程

来源:互联网 发布:网络词汇用语2017最新 编辑:程序博客网 时间:2024/05/01 21:45


Thread t1;

Object a;

Thread类;

调用区别:Thread.sleep();   Thread.yield();

a.wait();

t1.jion();      t1.setDaemon(true); 

 notify();与 notifyAll();直接调用即可


ThreadA 中 有个 ThreadB, 调用ThreadB.join() 后ThreadB线执行,然后才是ThreadA

t.join(int wait_time);将t线程加入到当前线程,后面的代码想要执行需要满足以下条件之一:

1、线程t死了;
2、等待时间超过wait_time;

如果没有指定wait_time就只能等线程t死了才行了···

join() 理解成 强行抢占资源 比较好 我觉得发生机制 在栈空间里 就像楼主画的图那样 谁调用join()谁抢占资源 先运行 要么运行完 要么设置占用资源时间


sleep:将运行线程转到阻塞状态,持有对象锁,sleep可以在非同步块调用

yield:将运行线程转到就绪(可运行)状态,持有对象锁

wait:释放对象锁,只能在同步方法或块中调用,在对象上调用wait()方法(不是在线程上调用)。wait是Object类的方法


线程离开运行状态的3种方法:
1、调用Thread.sleep():使当前线程睡眠至少多少毫秒(尽管它可能在指定的时间之前被中断)。
2、调用Thread.yield():不能保障太多事情,尽管通常它会让当前运行线程回到可运行性状态,使得有相同优先级的线程有机会执行。
3、调用join()方法:保证当前线程停止执行,直到该线程所加入的线程完成为止。然而,如果它加入的线程没有存活,则当前线程不需要停止。
 
除了以上三种方式外,还有下面几种特殊情况可能使线程离开运行状态:
1、线程的run()方法完成。
2、在对象上调用wait()方法(不是在线程上调用)。
3、线程不能在对象上获得锁定,它正试图运行该对象的方法代码。
4、线程调度程序可以决定将当前运行状态移动到可运行状态,以便让另一个线程获得运行机会,而不需要任何理由。

java.lang.Object的类的三个方法来学习:
 void notify() 
    唤醒在此对象监视器上等待的单个线程。 
 void notifyAll() 
    唤醒在此对象监视器上等待的所有线程。 
 void wait() 


在使用synchronized关键字时候,应该尽可能避免在synchronized方法或synchronized块中使用sleep或者yield方法,因为synchronized程序块占有着对象锁,你休息那么其他的线程只能一边等着你醒来执行完了才能执行。不但严重影响效率,也不合逻辑。
同样,在同步程序块内调用yeild方法让出CPU资源也没有意义,因为你占用着锁,其他互斥线程还是无法访问同步程序块。当然与同步程序块无关的线程可以获得更多的执行时间。

Java中的多线程是一种抢占式的机制 而不是分时机制。抢占式机制指的是有多个线程处于可运行状态,但是只有一个线程在运行。 

共同点: 
1. 他们都是在多线程的环境下,都可以在程序的调用处阻塞指定的毫秒数,并返回。 

2. wait()和sleep()都可以通过interrupt()方法 打断线程的暂停状态 ,从而使线程立刻抛出InterruptedException。 
   如果线程A希望立即结束线程B,则可以对线程B对应的Thread实例调用interrupt方法。如果此刻线程B正在wait/sleep /join,则线程B会立刻抛出InterruptedException,在catch() {} 中直接return即可安全地结束线程。 
   需要注意的是,InterruptedException是线程自己从内部抛出的,并不是interrupt()方法抛出的。对某一线程调用 interrupt()时,如果该线程正在执行普通的代码,那么该线程根本就不会抛出InterruptedException。但是,一旦该线程进入到 wait()/sleep()/join()后,就会立刻抛出InterruptedException 。 

不同点: 
1. Thread类的方法:sleep(),yield()等 
   Object的方法:wait()和notify()等 
   
2. 每个对象都有一个锁来控制同步访问。Synchronized关键字可以和对象的锁交互,来实现线程的同步。 
   sleep方法没有释放锁,而wait方法释放了锁,使得其他线程可以使用同步控制块或者方法。 

3. wait,notify和notifyAll只能在同步控制方法或者同步控制块里面使用,而sleep可以在任何地方使用 
4. sleep必须捕获异常,而wait,notify和notifyAll不需要捕获异常   

 

 

 线程的调度 
        线程调度器按线程的优先级高低选择高优先级线程(进入运行中状态)执行,同时线程调度是抢先式调度,即如果在当前线程执行过程中,一个更高优先级的线程进入可运行状态,则这个线程立即被调度执行。


抢先式调度又分为:时间片方式和独占方式。在时间片方式下,当前活动线程执行完当前时间片后,如果有其他处于就绪状态的相同优先级的线程,系统会将执行权交给其他就绪态的同优先级线程;当前活动线程转入等待执行队列,等待下一个时间片的调度。 
在独占方式下,当前活动线程一旦获得执行权,将一直执行下去,直到执行完毕或由于某种原因主动放弃CPU,或者是有一高优先级的线程处于就绪状态。




守护线程与普通线程写法上基本么啥区别,调用线程对象的方法setDaemon(true),则可以将其设置为守护线程,该方法必须在启动线程前调用。。
守护线程使用的情况较少,但并非无用,举例来说,JVM的垃圾回收、内存管理等线程都是守护线程。还有就是在做数据库应用时候,使用的数据库连接池,连接池本身也包含着很多后台线程,监控连接个数、超时时间、状态等等。
public static void main(String[] args) { 
                Thread t1 = new MyCommon(); 
                Thread t2 = new Thread(new MyDaemon()); 
                t2.setDaemon(true);        //设置为守护线程 


                t2.start(); 
                t1.start(); 
        } 
JRE判断程序是否执行结束的标准是所有的前台执线程行完毕了,而不管后台线程的状态

0 0
原创粉丝点击