`

线程总结

    博客分类:
  • Java
 
阅读更多

一、程序,进程,线程的区别

       程序是一段静态的代码,是软件执行的蓝本。

       进程可以理解为正在进行的程序,它是从程序加载、执行、执行完毕的整个过程,多个进程可以共享操作系统所管理的资源,比如剪切板。每个进程都单独占用一块内存,多个进程之间不能数据共享,必须通过网络交换数据。

       线程是比进程更小的执行单位,一个进程在执行过程中,可以产生多个线程。线程是一个进程内部的多个并行的运行单元,同一个进程的多个线程之间可以通过内存共享数据。通俗地讲,线程是运行在进程中的小“进程”。

 

二、Java中的多线程

       Java程序执行时,先启动了一个主线程,即main方法。在main方法中,可以开启其他线程。当main方法执行完毕的时候,程序不会退出,只有所有线程都执行完毕,Java程序才会退出。

       线程的使用方法:

       1.继承Thread类

       

public class A extends Thread{
	public void run(){
		//线程要干的活...
	}
}

class Main{
         public static void main(String args[]){
                    new A().start();//这样就执行了A中的run方法,当run执行完毕,线程结束

          }

}

 

 

       2.实现Runnable接口

 

 

/**
 *刷新Jpanel的线程
 *
 **/

public class UpdataCenter implements Runnable {

	JPanel center;

	public UpdataCenter(JPanel center) {
		this.center = center;
	}

	@Override
	public void run() {
		while (true) {
			if (!MainUI.pause) {
				center.repaint();
			}
			try {
				Thread.sleep(50);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
}

//调用
UpdataCenter st = new UpdataCenter(center);
		Thread th = new Thread(st);
		th.start();//开启线程

 

 

         当有多个线程执行的时候,由于CPU每次只处理一个,他们的处理顺序是不确定的,所以先开启的线程不一定先执行。

 

        当run方法执行完毕,线程变成死亡状态,释放了分配给线程对象的内存。如果在run未执行完重复调用start()会抛出IlIegalThreadStateException。

 

三、线程中的方法

       sleep(long a);休眠a毫秒,再往下执行。

       interrupt()可以吵醒休眠的线程,结束休眠。

       isAlive();线程在执行run的过程中返回true,开始执行前和执行结束后返回false。

 

四、线程同步

        同步:有多个线程同时操作一个变量,这个时候就没办法确定程序的正确性了,所以应该想办法解决这种 同时执行的时候对变量改变的不确定性,所以得先执行玩这个再执行其它,同时只做一件事。

        异步:多线程可以同时操作,不会发生结果错误。、

 

        比如在两个地方同时使用网银转账。本来银行卡剩余1000,在A处转出800,在B处转出500.最后本来卡内余额应该为-300。但是由于使用两个线程在同时取钱,这就造成了结果的不确定性。

 

线程A处从内存中读到money = 1000;对其进行-800的操作,money = 200;

线程B处从内存中读到money = 1000;对其进行-500的操作,money = 500;

两个线程计算完毕,将数据放回原地,而Cpu只能一次一次的放(同时只能处理一件事,只是速度非常快)CPU有可能先将200放回去,再将500放回去,那么卡内余额变成了500(200被覆盖),反之亦然。

 

       当然,还有一种情况是,线程A处从内存中读到money = 1000;对其进行-800的操作,money = 200,将其放回原地;在进行B的操作(线程执行顺序具有不确定性)。这个时候卡内余额变成了-300。

 

       如何避免第一种情况的发生?这就是线程同步需要处理的问题。

 

       Java中提供了关键字synchronized(同步),使用该关键字可以修饰方法,对象。

使用它修饰之后,当一个线程取出了这个对象对其修改或者这个方法对其执行。其余线程必须等待它先执行完毕。这样就避免了第一种错误。

 

       使用方法:

 

public class Main {
	public static void main(String[] args) {
		Account ac = new Account();
		new Card(ac, 800).start();
		new Card(ac, 400).start();
		
	}
}

package Thread20140715;

public class Card extends Thread {

	Account ac;
	int n;

	public Card(Account ac, int n) {
		this.ac = ac;
		this.n = n;
	}

	/**
	 * 异步:可以同时修改这块内存,导致结果出错。
	 * 
	 * 如果没在Account锁定,
	 * 
	 * 输出结果:-200 -200或者200 200或者600 600
	 * 
	 * 否则,结果为 200/600 -200
	 */
	public void run() {
		ac.getMoney(n);
		ac.getMoney1(n);
		ac.getMoney2(n);
	}

	/**
	 * 同步:只允许一个Account操作修改这块内存。
	 * 
	 * 输出结果: 200/600 -200
	 */
	// public void run() {
	// synchronized (ac) {// 锁定ac
	// ac.getMoney(n);
	// }
	// }

}

package Thread20140715;

public class Account {

	int money = 1000, money1 = 1000, money2 = 1000;

	/**
	 * 异步
	 * 
	 * @param n
	 *            要取多少钱
	 */
	public void getMoney1(int n) {
		try {
			Thread.sleep(500);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		money1 -= n;
		System.out.println("剩余金额1:" + money1);
	}

	/**
	 * 同步
	 * 
	 * @param n
	 *            要取多少钱
	 */
	public void getMoney2(int n) {
		synchronized (this) {// 锁定this对象,所有属性不可修改
			try {
				Thread.sleep(500);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			money2 -= n;
			System.out.println("剩余金额2:" + money2);
		}
	}

	/**
	 * 同步
	 * 
	 * @param n
	 *            要取多少钱
	 */
	public synchronized void getMoney(int n) {
		// 锁定方法,此时不锁定属性,
		// 所以其它方法仍然可对money进行修改,导致结果出错

		try {
			Thread.sleep(500);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		money -= n;
		System.out.println("剩余金额:" + money);

	}
}

输出结果:
剩余金额:200
剩余金额1:600
剩余金额:-200
剩余金额2:200
剩余金额1:600
剩余金额2:-200

由结果可知,剩余金额1是错误的结果,因为1没有使用同步(这个结果很随机,但是只有1可能错误)。

 

五、锁

用来锁定中间的代码。

        Lock lc = new ReentrantLock();

lc.lock();

        ……

lc.unlock();

 

六、计时器线程

       使用方法:创建Timer计时器对象,创建相应任务。调用schedule方法。传入任务,第一次执行延长的时间,以后执行的周期。

       Timer timer = new Timer();

       EnermyPlaneCreator task = new EnermyPlaneCreator();

       timer.schedule(task, 800, 20000);

 

       public class EnermyPlaneCreator extends TimerTask{

              public void run() {}

       }

 

七、线程使用模型

1.监听器模型

在A线程中增加标记flag,在B线程中对A中的flag进行操作。B为A的监听线程。

例如:flag初始值为0,表示A没开始执行。当A start之后,将flag赋值为1,这时B就可以知道此时A已经开始执行了。当A执行到某个操作,flag=2,这是B也可以知道A执行到哪里了。

 

2.生产消费模型

开启一个生产者线程A,一个消费者线程B。

在A中生成对象,在B中消费对象。(例如ArrayList的add,remove操作)

由于线程的执行顺序不确定,这个时候A可能还没有生成,B就去消费了。程序就会有问题。

可以进行判断,当消费的时候判断是否已经有了生成,如果没有生产,就通知生产者进行生产并等待生产者进行生产。

 

示例代码:

public class Main {

	// 存放产品对象的队列
	static ArrayList<Computer> coms = new ArrayList<Computer>();

	public static void main(String[] args) {

		// 生产者
		Produce p = new Produce();
		p.start();

		// 消费者
		Customer c = new Customer();
		c.start();
	}
}


public class Computer {

	String name;

	public Computer(String name) {
		this.name = name;
	}

	@Override
	public String toString() {//打印对象时调用
		return name;
	}

}

public class Produce extends Thread {

	int i = 0;

	public void run() {
		while (true) {
			try {
				synchronized (Main.coms) {
					if (Main.coms.size() > 0) {
						Main.coms.wait();
					} else {
						Computer com = new Computer("第" + i + "台电脑");
						Main.coms.add(com);
						System.out.println("生产了电脑:" + com);
						i++;
						Main.coms.notify();
					}
				}
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}

		}

	}

}

public class Customer extends Thread {

	public void run() {
		while (true) {
			try {
				synchronized (Main.coms) {
					// 如果产品不够,就等待
					if (Main.coms.size() <= 0) {
						Main.coms.wait();
					} else {
						Computer c = Main.coms.remove(0);
						System.out.println("-->消费了一台:" + c);
						Main.coms.notify();
					}
				}
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
}

 

 

 

 

 

 

 

 

0
0
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics