Java线程安全

本文探讨了Java中线程安全的问题,通过两个示例展示了在多线程环境下,不加对象锁和加对象锁(使用synchronized)对访问同一实例数据的影响。在没有加锁的情况下,会出现线程不安全状态;而在run方法上加上对象锁,则能确保线程安全。
摘要由CSDN通过智能技术生成
线程安全概念:当多个线程访问某一个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的。

synchronized:可以在任意对象及方法上面加锁,而加锁的这段代码称为“互斥区”或“临界区”。

示例:多个线程访问同一个实例数据

1、情形一:方法不加对象锁

该情形会出现线程不安全的状态。

package com.sunft.base.sync001;

/**
 * 线程安全概念:当多个线程访问某一个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的。
 * synchronized:可以在任意对象及方法上面加锁,而加锁的这段代码称为“互斥区”或“临界区”。
 * @author sunft
 *
 */
public class MyThread extends Thread {
	
	private int count = 5;
	
	@Override
	public void run() {
		count --;
		System.out.println(this.currentThread().getName() + ":count=" + count);
	}

	public static void main(String[] args) {
		MyThread myThread = new MyThread();
		Thread t1 = new Thread(myThread, "t1");
		Thread t2 = new Thread(myThread, "t2");
		Thread t3 = new Thread(myThread, "t3");
		Thread t4 = new Thread(myThread, "t4");
		Thread t5 = new Thread(myThread, "t5");
		
		t1.start();
		t2.start();
		t3.start();
		t4.start();
		t5.start();
	}

}

控制台打印信息:会出现线程安全问题

t2:count=2
t5:count=0
t1:count=1
t4:count=1
t3:count=2

2、情形二:给run方法加上对象锁

该情形能够保证线程安全。

package com.sunft.base.sync001;

/**
 * 线程安全概念:当多个线程访问某一个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的。
 * synchronized:可以在任意对象及方法上面加锁,而加锁的这段代码称为“互斥区”或“临界区”。
 * @author sunft
 *
 */
public class MyThread extends Thread {
	
	private int count = 5;
	
	/**
	 * 注意Thread类中方法没有synchronized,
	 * 在这里加上
	 */
	@Override
	public synchronized void run() {
		count --;
		System.out.println(this.currentThread().getName() + ":count=" + count);
	}

	public static void main(String[] args) {
		MyThread myThread = new MyThread();
		Thread t1 = new Thread(myThread, "t1");
		Thread t2 = new Thread(myThread, "t2");
		Thread t3 = new Thread(myThread, "t3");
		Thread t4 = new Thread(myThread, "t4");
		Thread t5 = new Thread(myThread, "t5");
		
		t1.start();
		t2.start();
		t3.start();
		t4.start();
		t5.start();
	}

}

控制台打印信息:

t1:count=4
t3:count=3
t2:count=2
t4:count=1
t5:count=0
示例总结:
当多个线程访问myThread的run方法时,以排队的方式进行处理(这里的排队是按照CPU分配的先后顺序而定的),一个线程想要执行synchronized修饰的方法里的代码,首先是尝试获得锁,如果拿到锁,执行synchronized代码体内容;拿不到锁,这个线程就会不断的尝试获得这把锁,直到拿到为止,而且是多个线程同时去竞争这把锁。(也就是会有锁竞争的问题)。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值