创建多线程的几种方式

第一种方式:通过继承Thread类创建

new Thread() {// new Threah(){}表示创建一个匿名子类的实例对象,{}内是子类的代码
			// 重写父类的run方法
			public void run() {
				while (true) {
					try {// 必需try...catch
						Thread.sleep(3000);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
					System.out.println(Thread.currentThread().getName());
				}
			}
		}.start();

 第二种方式:通过实现Runnable接口创建

// new Thread(new Runnable(){}).start();表示调用Threah对象接受的Runnable对象的run方法.
		new Thread(new Runnable() {
			public void run() {
				while (true) {
					try {
						Thread.sleep(3000);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
					System.out.println(Thread.currentThread().getName());
				}
			}
		}).start();

 Thread内部run方法的代码:

 public void run() {
	if (target != null) {
	    target.run();
	}
    }

 target 是Runnable子类的实例对象,若通过Thread子类实现,则优先调用子类的run方法.其实Thread类也是实现了 Runnable接口.

使用第二种方式创建的好处:

            1、 多个线程共享同一份资源

            2、可以避免由于java单继承特性带来的局限性。

 

无论是继承还是实现都需要重写run方法,此方法为线程的主体.

 

模拟多个线程共享同一份资源

问题:一个火车站当天去往某地的车票一共有50张,由多个售票点负责售票.

若通过继承方式如下:

class Demo extends Thread
{
	private int ticket = 50 ;
	public void run()
	{
		while(this.ticket>0)
		{
			System.out.println("卖票:"+this.ticket--) ;
		}
	}
};
public class ThreadDemo04
{
	public static void main(String args[])
	{
		// 准备四个售票点
		Demo d1 = new Demo() ;
		Demo d2 = new Demo() ;
		Demo d3 = new Demo() ;
		Demo d4 = new Demo() ;

		d1.start() ;
		d2.start() ;
		d3.start() ;
		d4.start() ;
	}
};

 但这样打印的结果是:每个售票点都售票50张,不符合实际要求!当然你可以把ticket设置成为static,这样也可以符合要求.

第二种方式:

 

class Demo05 implements Runnable
{
	private int ticket = 50 ;
	public void run()
	{
		while(this.ticket>0)
		{
			System.out.println("卖票:"+this.ticket--) ;
		}
	}
};
public class ThreadDemo05
{
	public static void main(String args[])
	{
		// 四个售票点应该控制同一个资源:50
		Demo05 d = new Demo05() ;
		Thread t1 = new Thread(d) ;
		Thread t2 = new Thread(d) ;
		Thread t3 = new Thread(d) ;
		Thread t4 = new Thread(d) ;

		t1.start() ;
		t2.start() ;
		t3.start() ;
		t4.start() ;
	}
};

 由于不同的线程调用的是同一个对象(Demo05 d = new Demo05() )的run方法,因此访问的ticket 都是同一个对象的成员属性.

 

但这两份代码都存在线程安全问题:ticket到最后可能会变成一个负数,如在while循环开始加一段:

		try {
				Thread.sleep(300);
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}

 

这样ticket最后打印的值就会出现负数!  解决的方法线程同步.在下一节里讲

 

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