Java中Runnable和Thread的区别

        在Java中可有两种方式实现多线程,一种是继承Thread类,一种是实现Runnable接口。
Thread类是在java.lang包中定义的。一个类只要继承了Thread类同时覆写了本类中的run()方法就可以实现多线程操作了,但是一个类只能继承一个父类,这是此方法的局限。
①使用Thread方式启动多线程:
class MyThread extends Thread{  
	private String name;  
	public MyThread(String name) {  
		super();  
		this.name = name;  
	}  
	public void run(){  
		for(int i=0;i<10;i++){  
			System.out.println("线程开始:"+this.name+",i="+i);  
		}  
	}  
}  
public class ThreadDemo01 {  
	public static void main(String[] args) {  
		MyThread mt1=new MyThread("线程a");  
		MyThread mt2=new MyThread("线程b");  
		mt1.run();  
		mt2.run();  
	}  
} 

运行结果:
Thread Run:Thread A,i=0
Thread Run:Thread A,i=1
Thread Run:Thread A,i=2
Thread Run:Thread A,i=3
Thread Run:Thread A,i=4
Thread Run:Thread B,i=0
Thread Run:Thread B,i=1
Thread Run:Thread B,i=2
Thread Run:Thread B,i=3
Thread Run:Thread B,i=4

        不过此时运行的结果并不是我们所预料中的那样,而是先执行完线程A,然后再执行线程B。导致这样的结果是因为程序中使用了run方法启动线程,通过查找源代码:
public void run() {
	if (target != null) {
		target.run();
	}
}

        因为线程是跟操作系统相关的,run方法并没有操作系统相关的操作,所以也就不会产生多线程的结果啦。通过查看jdk文档可以发现start方法:当调用start方法时,JVM会调用run方法。下面使用start方法启动线程:
public class ThreadDemo01 {  
	public static void main(String[] args) {  
		MyThread mt1=new MyThread("Thread A");  
		MyThread mt2=new MyThread("Thread B");  
		//mt1.run();  
		//mt2.run();  
		mt1.start() ;
		mt2.start() ;
	}  
} 

运行结果:
Thread Run:Thread A,i=0
Thread Run:Thread A,i=1
Thread Run:Thread B,i=0
Thread Run:Thread B,i=1
Thread Run:Thread B,i=2
Thread Run:Thread B,i=3
Thread Run:Thread A,i=2
Thread Run:Thread A,i=3
Thread Run:Thread A,i=4
Thread Run:Thread B,i=4

        可以看到此时线程A和线程B就并行执行了。
②通过实现Runnable接口实现多线程
因为在Java中一个class只能继承一个父类,所以在实际开发中很少有使用到Thread,而是使用Runnable接口。
class MyThread02 implements Runnable{  
	private String name;  
	public MyThread02(String name) {  
		this.name = name;  
	}
	public void run(){  
		for(int i=0;i<100;i++){  
			System.out.println("Thread Run:"+this.name+",i="+i);  
		}  
	}  
}
public class ThreadDemo02 {  
	public static void main(String[] args) {  
		MyThread mt1=new MyThread("Thread A");  
		MyThread mt2=new MyThread("Thread B");  
		
		new Thread(mt1).start() ;
		new Thread(mt2).start() ;
	}  
} 

运行结果:
Thread Run:Thread B,i=0
Thread Run:Thread B,i=1
Thread Run:Thread A,i=0
Thread Run:Thread B,i=2
Thread Run:Thread A,i=1
Thread Run:Thread B,i=3
Thread Run:Thread A,i=2
Thread Run:Thread A,i=3
Thread Run:Thread B,i=4
Thread Run:Thread A,i=4


两种实现方式的区别:
Thread方式:
 
  •   通过继承Thread类
  •   重写run方法
  •   在main函数中通过new Thread子类,然后调用start方法开启多线程

Runnable方式:
 
  •   通过实现接口Runnable
  •   重写run方法
  •   在main函数中通过new Thread传入Runnable对象,然后调用start方法开启多线程

        在程序开发中只要是多线程肯定永远以实现Runnable接口为主,因为实现Runnable接口相比继承Thread类有如下好处:
  • 避免单继承的局限,一个类可以继承多个接口。
  • 适合于资源的共享

实例(以卖票程序为例Thread方式实现,系统中有5张票,每个线程相当于一个售票机):
public class MyThread03 extends Thread{  
	private String name ;
	private int ticket=5;
	public MyThread03(String name){
		this.name = name ;
	}
	public void run(){  
		for(int i=0;i<20;i++){  
			if(this.ticket>0){  
				System.out.println(name+"卖票,剩余:"+this.ticket--);  
			}  
		}  
	}  
}
public class ThreadTicket {  
	public static void main(String[] args) {  
		Thread mt1=new MyThread03("Thread A");  
		Thread mt2=new MyThread03("Thread B");  
		Thread mt3=new MyThread03("Thread C");  
		
		mt1.start();
		mt2.start();
		mt3.start();
	}  
} 

运行结果:
Thread A卖票,剩余:5
Thread C卖票,剩余:5
Thread B卖票,剩余:5
Thread C卖票,剩余:4
Thread A卖票,剩余:4
Thread C卖票,剩余:3
Thread B卖票,剩余:4
Thread C卖票,剩余:2
Thread A卖票,剩余:3
Thread A卖票,剩余:2
Thread C卖票,剩余:1
Thread B卖票,剩余:3
Thread A卖票,剩余:1
Thread B卖票,剩余:2
Thread B卖票,剩余:1

        从运行结果可以看出,此时三个售票机并没有共享5张票,而是各自买了5张票,这是不合理的。
实例(Runnable接口方式):
public class MyThread04 implements Runnable{ 
	private int ticket=5;
	public void run(){  
		for(int i=0;i<20;i++){  
			if(this.ticket>0){  
				System.out.println(Thread.currentThread().getName() +"卖票,剩余:"+this.ticket--);  
			}  
		}  
	}  
}
public class ThreadTicket02 {  
	public static void main(String[] args) {  
		Runnable rn=new MyThread04();  
		
		new Thread(rn,"Thread A").start();
		new Thread(rn,"Thread B").start();
		new Thread(rn,"Thread C").start();
	}  
} 

运行结果:
Thread A卖票,剩余:5
Thread A卖票,剩余:2
Thread C卖票,剩余:3
Thread B卖票,剩余:4
Thread A卖票,剩余:1

        此时使用Runnable方法向三个Thread中传递同一个对象,即可实现资源的共享。当然此时剩余票数并没有顺序,因为此时并没有对程序进行同步处理,可在run方法中添加synchronized同步声明:
public class MyThread04 implements Runnable{ 
	private int ticket=5;
	public synchronized void run(){  
		for(int i=0;i<20;i++){  
			if(this.ticket>0){  
				System.out.println(Thread.currentThread().getName() +"卖票,剩余:"+this.ticket--);  
			}  
		}  
	}  
}

运行结果:
Thread B卖票,剩余:5
Thread B卖票,剩余:4
Thread B卖票,剩余:3
Thread B卖票,剩余:2
Thread B卖票,剩余:1

        最后通过查看源代码可知道其实Thread已经实现了Runnable接口:class Thred implements Runnable。
        这是自己第一篇博客,语言组织方面没有那么完善,同时博客内容中有不少是从别的地方借鉴的,希望能够培养自己以后写博客的习惯,同时也能提高自己的能力。

你可能感兴趣的:(java,多线程,thread,Runnable)