题1:
Given:11. Runnable r = new Runnable() {
12. public void run() {
13. System.out.print("Cat");
14. }
15. };
16. Thread t = new Thread(r) {
17. public void run() {
18. System.out.print("Dog");
19. }
20. };
21. t.start();
What is the result?
A. Cat
B. Dog
C. Compilation fails.
D. The code runs with no output.
E. An exception is thrown at runtime.单独看上面这道题不难。但是,当接着看下面这道题时,第一印象觉得是不是少了Thread的run()方法:题2:
Given:1. public class Threads5 {
2. public static void main (String[] args) {
3. new Thread(new Runnable() {
4. public void run() {
5. System.out.print("bar");
6. }}).start();
7. }
8. }What is the result?
A. Compilation fails.
B. An exception is thrown at runtime.
C. The code executes normally and prints "bar".
D. The code executes normally, but nothing prints.
请问如何来区分该选择runnable中、还是Thread中的run()方法呢?thx。

解决方案 »

  1.   

    Thread不是抽象类,它不需要重写run()也可以跑
      

  2.   

    public class Thread implements Runnable
    Thread 实现了 接口 不是必须实现接口中的方法吗?
    这里,用重写就能解释吧
    你绝的少了,其实是 Thread 匿名类中直接重写了run()
      

  3.   

    thx。
    那么看第二道题:
    1. public class Threads5 {
    2.        public static void main (String[] args) {
    3.            new Thread(new Runnable() {
    4.                public void run() {
    5.                    System.out.print("bar");
    6.            }}).start();
    7.        }
    8. }
    如果没有重写Thread的run()方法,Thread也可以start,但是应该就不会让运行Runnable中的run()方法吧。这道题却输出了bar。这怎么解释呢?thx。
      

  4.   

    run
    public void run()如果该线程是使用独立的 Runnable 运行对象构造的,则调用该 Runnable 对象的 run 方法;否则,该方法不执行任何操作并返回。 
    Thread 的子类应该重写该方法。 
    指定者:
    接口 Runnable 中的 run
    jdk的说明    /* What will be run. */
        private Runnable target;    public void run() {
    if (target != null) {
        target.run();
    }
        }run方法的源码,Thread本身的run方法不做任何操作,这个查看一下api和源代码就搞定了
      

  5.   

    这个你要看JDK的源码实现(jdk安装目录下有个src.zip):
    public
    class Thread implements Runnable {
    …………
    private Runnable target;  //保存初始化时传入的实现Runnable接口的对象
    ……………………………………
    }
    看他的start和run方法实现:
    public synchronized void start() {
           if (threadStatus != 0)
                throw new IllegalThreadStateException();
            group.add(this);
            start0();
            if (stopBeforeStart) {
        stop0(throwableFromStop);
    }
    }
    start实际上是去调用start0()(native method),由其去调用run方法。再看Thread类的run实现:
     public void run() {
    if (target != null) {
        target.run();
    }
        }
    那么第一题很好理解了,因为你override了Thread的run方法,所以不会去调用target Runnable的run方法,所以输出Dog;第二题当然输出bar,因为初始化Thread时传入了用匿名内部类实现Runnable接口的对象,所以target不为null,而本身的run方法没有override,所以就会调用该匿名内部类(即target Runnable)实现的run方法。
      

  6.   


    Thread继承了Runnable呢的run
      

  7.   

    看下源码就知道了,Thread里有成员变量Runnable target,当构造函数的参数有Runnable时就会把target赋值为参数,Thread的run方法会判断target是否为null,不为null就调用target的run方法
    如果重写了Thread的run方法并不调用super.run();,那target的run方法自然就不会再被调用了
      

  8.   

    显然是不喜欢看API的楼主。。
      

  9.   

    看源代码
    private Runnable target;    public void run() {
        if (target != null) {
            target.run();
        }
    }
    所以如果传入的target不为空的话,就调其run方法,而第一题中,传入的target却是不为空,但是Thread类的run方法被覆盖了,也就是说,和上面源代码中的方法没有关系了,所以会直接调用重写的run方法,
    第二题,解释同上,因为传入的target不为空
      

  10.   

    个人理解:Thread类的run()方法默认是执行创建它的Runabble接口对象的run()方法,第一种形式实际上是覆盖了这个默认的run()方法
      

  11.   


    . public class Threads5 {
    2.        public static void main (String[] args) {
    3.            new Thread(new Runnable() {
    4.                public void run() {
    5.                    System.out.print("bar");
    6.            }}).start();
    7.        }
    8. }这个拆分一下,就会得到跟上面一样的实现:
    Runnable r = new Runnable() {
            public void run() {
                System.out.print("bar");
            }
     };
     Thread t = new Thread(r);
     t.start();这里Thread没重写,了
    因为Thread的源代码 public void run() {
      if (target != null) {
      target.run();
      }
    }
    传进去的target 是runnable对象不为空,就会调用Rubnable的run()
    显示 bar
      

  12.   

    源码是最好的答案, 不喜欢翻源码搞java很吃亏的阿!
      

  13.   

    第一题是因为你重写了Thread类的run()方法,所以源代码
    public void run() {
      if (target != null) {
      target.run();
      }
    }
    就被覆盖了,,所以它不会指向Runnable的run()方法;所以输出Dog!
    第二题是你写了一个内部匿名类实现Runnable的接口,重写了Runnable的run()方法;输出bar