synchronized同步方法简单介绍(一)

非线程安全和线程安全

非线程安全:其实会在多个线程对同一个对象中的实例变量进行并发访问时产生,结果就是脏读。

线程安全:获得的实例变量的值经过同步处理,不会出现脏读的现象。

一个方法中的变量不存在非线程安全问题,永远都是线程安全的。这是因为方法内部的变量是私有特性造成的。

代码如下解读

public class HasSelfPrivateNum {

	public void addI(String username) {
		try {
			int num = 0;//注意该变量在方法内部
			if (username.equals("a")) {
				num = 100;
				System.out.println("a set over!");
				Thread.sleep(2000);
			} else {
				num = 200;
				System.out.println("b set over!");
			}
			System.out.println(username + " num=" + num);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

}
public class ThreadA extends Thread {

	private HasSelfPrivateNum numRef;

	public ThreadA(HasSelfPrivateNum numRef) {
		super();
		this.numRef = numRef;
	}

	@Override
	public void run() {
		super.run();
		numRef.addI("a");
	}

}

public class ThreadB extends Thread {

	private HasSelfPrivateNum numRef;

	public ThreadB(HasSelfPrivateNum numRef) {
		super();
		this.numRef = numRef;
	}

	@Override
	public void run() {
		super.run();
		numRef.addI("b");
	}

}

public class Run {

	public static void main(String[] args) {

		HasSelfPrivateNum numRef = new HasSelfPrivateNum();

		ThreadA athread = new ThreadA(numRef);
		athread.start();

		ThreadB bthread = new ThreadB(numRef);
		bthread.start();

	}

}

把上面的代码改成如下:不加同步出现脏读,加上没有问题

结论:两个线程访问同一个对象中的同步方法时一定是线程安全的

public class HasSelfPrivateNum {

	//该变量放在方法的外面 不安全
	private int num = 0;

	//加上synchronized后多个线程并发访问,只能排队访问资源,不会出现线程安全问题
	 public synchronized void addI(String username) {
		try {
			if (username.equals("a")) {
				num = 100;
				System.out.println("a set over!");
				Thread.sleep(2000);
			} else {
				num = 200;
				System.out.println("b set over!");
			}
			System.out.println(username + " num=" + num);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

}

再对上面的main()代码改造。

public class Run {

	public static void main(String[] args) {
		
		//两个线程访问两个不同的实例的相同的同步方法,结果异步执行
		HasSelfPrivateNum numRef1 = new HasSelfPrivateNum();
		HasSelfPrivateNum numRef2 = new HasSelfPrivateNum();

		ThreadA athread = new ThreadA(numRef1);
		athread.start();

		ThreadB bthread = new ThreadB(numRef2);
		bthread.start();

	}

}

总结:关键字synchronized获得的锁都是对象锁,不是说的吧一段代码或者方法当做锁,上面的案例就是那个线程先执行带synchronized关键字的方法,那个线程就持有该方法所属对象的锁Lock,其他线程只能等待,前提是多个线程访问同一个对象。这里不做synchronized锁的是对象代码阐述。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值