多线程基本知识

一、创建线程的三种方式

        其实创建线程远远不止有3种方式,但这里只记录三种。

1、通过继承Thread类来创建一个线程: 
步骤1:定义一个继承Thread类的子类: 
步骤2:构造子类的一个对象: Thread t1 = new Thread(); 
步骤3:启动线程: t1.start(); 
至此,一个线程就创建完成了。 

2、通过实现Runnable接口来创建Thread线程: 
步骤1:创建实现Runnable接口的类: 
步骤2:创建一个类对象: Runnable runnable = new SomeRunnable(); 
步骤3:由Runnable创建一个Thread对象: Thread t2 = new Thread(runnable); 
步骤4:启动线程: t2.start(); 
至此,一个线程就创建完成了。 
 
3、与方法2类似,通过实现Callable接口来创建Thread线程。
步骤1:创建实现Callable接口的类; 
步骤2:创建一个类对象: Callable callable = new SomeCallable(); 
步骤3:由Callable创建一个FutureTask对象: FutureTask ft= new FutureTask(callable); 
步骤4:由FutureTask创建一个Thread对象: Thread oneThread = new Thread(ft); 
步骤5:启动线程: oneThread.start(); 
至此,一个线程就创建完成了。 

可参考:推荐:http://www.importnew.com/25286.html

二、创建线程三种方式的demo

package resource.java.ordinary.mul.thread;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

/**
 * 线程Demo
 * 
 * @author xiao
 */
public class TranditionThread {
	public static void main(String[] args) throws InterruptedException, ExecutionException {
		/*
		 * 创建线程的方法一:
		 */
		Thread thread = new Thread() {
			@Override
			public void run() {
				// excuMethod(1);
			}
		};
		thread.start();

		/*
		 * 创建线程方法二:
		 */
		Thread thread2 = new Thread(new Runnable() {
			@Override
			public void run() {
				// excuMethod(2);
			}
		});
		thread2.start();

		/*
		 * 创建线程方法三:
		 */
		// FutureTask是一个包装器,它通过接受Callable来创建,它同时实现了 Future和Runnable接口。
		FutureTask<String> ft = new FutureTask<String>(new Callable<String>() {

			@Override
			public String call() throws Exception {
//				excuMethod(3);
				System.out.println("hi~~ 此处有个新线程");
				return "FutureTask 返回something";
			}
		});
		Thread t3 = new Thread(ft);
		t3.start();
		String result = ft.get();
		System.out.println(result);// 输出: FutureTask 返回something

		/*
		 * 问题:此方法运行的是excuMethod(4)方法还是excuMethod(5)方法??
		 */
		new Thread(new Runnable() {

			@Override
			public void run() {
//				excuMethod(4);
			}
		}) {
			@Override
			public void run() {
				// excuMethod(5);
			}
		}.start();
	}

	private static void excuMethod(int flag) {
		while (true) {
			try {
				Thread.sleep(500);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println(flag + "  " + Thread.currentThread().getName());
		}
	}

}

        对于上面问题的一个解答:

        答案:运行标识为5的线程。

        思路:这个方法的结构是这样的:

	new Thread( Runnable.run(){ 
		// 标识为4的线程 
	}){ run(){ 
		// 标识为5的线程 
    }}.start(); 

        原因:在Thread.class中,Thread是实现了Runnable接口的。在运行了Thread.start()方法后,先在子类中找run()方法,找到则用子类的方法,找不到在用父类的方法。在这题中,标识为5的线程所在的run()方法已经重写了父类的方法,所以最终运行的是excuMethod(5)方法。

三、关于synchronized关键字

1、synchronized的四种用法:

(1)修饰方法

(2)修饰一个代码块

(3)修饰一个静态方法

(4)修饰一个类

2、synchronized使用的demo

/**
 * 线程互斥:synchronized
 * @author xiao
 *
 */
public class TranditionThreadSynchronized {
	public static void main(String[] args) {
		new TranditionThreadSynchronized().init();
	}

	public void init() {
		final Output ootput = new Output();
		
		// 第一个线程
		new Thread(new Runnable() {
			@Override
			public void run() {
				while (true) {
					try {
						Thread.sleep(1000);
						ootput.output2("xiao");
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			}
		}).start();

		// 第二个线程
		new Thread(new Runnable() {
			@Override
			public void run() {
				while (true) {
					try {
						Thread.sleep(1000);
						ootput.output2("hag");
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			}
		}).start();
	}

	class Output {
		public void output(String name) {
			int len = name.length();
			// synchronized(XXX){}中,XXX表示的是要锁住的对象
			// 此this指的是和output2中实现的效果一样,锁的是Output对象
			synchronized (this) {
				for (int i = 0; i < len; i++) {
					System.out.print(name.charAt(i));
				}
				System.out.println();
			}
		}

		public synchronized void output2(String name) {
			int len = name.length();
			for (int i = 0; i < len; i++) {
				System.out.print(name.charAt(i));
			}
			System.out.println();
		}
	}
}

        思考:如果需要在类Output中加入output3()方法(如下代码块所示),且output1()和output3()需要保持互斥,则需要做些什么?         

            public static synchronized void output3(String name) {
                int len = name.length();
                for (int i = 0; i < len; i++) {
                    System.out.print(name.charAt(i));
                }
                System.out.println();
            }

        答案:需要做的事情有:

    (1)内部类Output需要添加关键字static,以此变为外部类。

    (2)将output1()中的synchronized (this)修改为synchronized (Output.class),使output1()方法中锁定的是Output类。

猜你喜欢

转载自my.oschina.net/u/3696939/blog/1815537
今日推荐