传统线程之同步锁(二)

一. 线程安全

线程安全问题是指程序中公用的东西被多个线程访问,比如:类的静态变量

线程互斥:是指两个线程之间不可以同时运行,他们会相互排斥,必须等待一个线程运行完毕,另一个才能运行


二. 同步锁

有什么办法可以解决线程安全问题呢?那就是在程序中加锁

Java有两种加锁的方法:

1. 在代码块中加锁 synchronized (this) { ... }

2. 在方法上加锁 public synchronized void xxx(){ ... }


示例代码:

public class TraditionalThreadSynchronized {

	public static void main(String[] args) {
		new TraditionalThreadSynchronized().init();
	}

	private void init() {
		final Outputer outputer = new Outputer();

		// 线程1
		new Thread(new Runnable() {
			@Override
			public void run() {
				while (true) {
					try {
						Thread.sleep(10);
						outputer.output1("1111111111");
					} catch (Exception e) {
						e.printStackTrace();
					}
				}
			}
		}).start();

		// 线程2
		new Thread(new Runnable() {
			@Override
			public void run() {
				while (true) {
					try {
						Thread.sleep(10);
						outputer.output1("2222222222");
					} catch (Exception e) {
						e.printStackTrace();
					}
				}
			}
		}).start();
	}

	class Outputer {
		public void output1(String name) {
			// 同步代码块
			synchronized (this) {
				for (int i = 0; i < name.length(); i++) {
					System.out.print(name.charAt(i));
				}
				System.out.println();
			}
		}
		
		// 同步方法
		public synchronized void output2(String name) {
			for (int i = 0; i < name.length(); i++) {
				System.out.print(name.charAt(i));
			}
			System.out.println();
		}
	}

}

线程1和线程2因为都要调用output1方法并打印传入的name字符串,cpu就会来回的在两个线程之间切换,

有可能线程1打印到一半的时候就切换到线程2上,这显然是我们不想看到的,所以在代码内容中加锁,

这样可以保证一个线程调用方法结束才会执行下一个线程。


上面代码中的this指的是outputer对象,它就是一把锁,两个线程使用同一把锁才能实现同步。

而在方法上加锁也可以实现线程同步,在方法上使用synchronized关键字也是把this作为锁。

其实output1方法和output2方法也是同步的,因为他们都是使用this作为锁。


思考一个问题:如果把一个方法定义为静态的即:public static synchronized

那么它使用什么作为锁呢?答案是:该类的字节码对象xxx.class


三. Synchonized和ThreadLocal

ThreadLocal和Synchonized都用于解决多线程并发访问,但是它们之间有本质区别,synchonized是利用锁机制,使变量或代码块在某一时刻只能被一个线程访问,而ThreadLocal为每一个线程提供一个变量副本,每个线程在同一时间访问到的并不是同一个对象,这样就隔离了多个线程对数据的共享,而Synchronized却正好相反,它用于在多个线程间通信时能够获得数据共享。

总结:synchonized用于线程间数据共享,ThreadLocal用于线程间数据隔离。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值