2019.08.20 JAVA中的Thread概念(三)

一. 线程同步(synchronized)
(1)什么是线程同步:就是同一个线程操作同一个资源。如果线程不同步就是引发线程安全问题。
(2)如果同时有多个线程操作同一个资源,就会带来问题,如何避免问题的产生,可以使用互斥锁(synchronized),这是一个关键字。作用在同一份资源上时,给这个资源加上一把锁,你其他线程不许进来,等我处理结束后再说。

举个线程不安全的例子:

package thread;

public class Test2 {
	public static class Bus extends Thread{
		int tickets=50;
		@Override
		public void run() {
			while(true){
				String name=Thread.currentThread().getName();
				if(tickets>=1){
					
					try {
						Thread.sleep(50);
					} catch (InterruptedException e) {
						// TODO Auto-generated catch block
						e.printStackTrace();
					}
					System.out.println(name+"售出"+tickets--+"号座位");
				}
			}
		}
	}
	
	public static void main(String[] args) {	
		Bus bus=new Bus();
		Thread window1 =new Thread(bus);
		window1.setName("window1");
		window1.start();
		Thread window2 =new Thread(bus);
		window2.setName("window2");
		window2.start();
		Thread window3 =new Thread(bus);
		window3.setName("window3");
		window3.start();
	}

}

在这里插入图片描述
从打印结果可以看出,不仅出现了重复座位,而且还出现了 0 号座位,存在极大的bug。出现bug的原因:多个线程参与同一个数据的操作,如上代码中,多个线程同时卖50张票,却没有给操作同一个资源加锁,就会出现这种bug。

(3)解决方法

  • 同步代码块
    修改 run(),添加 synchronized(Object obj) 关键字。这里一般传入 this ,this 即 Bus 类的对象。
    public static class Bus extends Thread{
    		int tickets=50;
    		@Override
    		public void run() {
    			while(true){
    				String name=Thread.currentThread().getName();
    				synchronized (this) {
    					if(tickets>=1){				
    						try {
    							Thread.sleep(50);
    						} catch (InterruptedException e) {
    							e.printStackTrace();
    						}
    						System.out.println(name+"售出"+tickets--+"号座位");
    					}
    				}
    			}
    		}
    	}
  • 同步方法
 public static class Bus extends Thread{
    		int tickets=50;
    		boolean flag=true;
    		@Override
    		public void run() {
    			while(flag){
    				
    				sell();
    			}
    		}
    		public synchronized void sell(){
    			String name=Thread.currentThread().getName();
    			if(tickets>=1){	
    				try {
    					Thread.sleep(50);
    				} catch (InterruptedException e) {
    					// TODO Auto-generated catch block
    					e.printStackTrace();
    				}
    				System.out.println(name+"售出"+tickets--+"号座位");
    				flag=true;
    			}else{
    				flag=false;
    			}
    		}
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值