`
xiao1zhao2
  • 浏览: 38987 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java中的并发:同步

阅读更多
目录
  1.安全问题的产生
  2.同步:synchronized
  3.并发中的三个概念
  4.happens-before原则
  5.修饰符:volatile

 

1.安全问题的产生

  如果我们想实现这样一个功能:利用线程并发模拟多个网点售同一种票.例:

				public class ThreadDemo {

					public static void main(String[] args) {
						Ticket ticket = new Ticket();
						new Thread(ticket).start();
						new Thread(ticket).start();
						new Thread(ticket).start();
					}
				}

				class Ticket implements Runnable {
					final Random r = new Random();
					private int num = 10;

					@Override
					public void run() {
						while (true) {
							if (num > 0) {
								try {
									Thread.sleep(r.nextInt(100));
								} catch (InterruptedException e) {
								}
								System.out.println(Thread.currentThread().getName() + "..." + num--);
							} else {
								break;
							}
						}
					}
				}

 

在运行结果中,输出了0和-1,一共买了12张票!
  由于每个线程执行的过程是不可控的,多个线程同时访问一个资源(本例中的num)时,会导致程序运行结果并不是想看到的结果,便会产生线程安全问题,这个资源被称为共享资源.

2.同步

  2.1synchronized
    基本上所有的并发模式在解决线程安全问题时,都采用"序列化访问临界资源"的方案,即同一时刻只能有一个线程访问共享资源,也称作同步互斥访问.通常是在访问临界资源的代码前面加上一个锁,当访问完临界资源后释放锁,让其他线程继续访问.java提供了synchronized关键字实现同步互斥访问,synchronized既可以修饰代码块,也可以用来修饰方法.
    每一个对象都拥有一个锁标记(monitor),也称为监视器,线程只有获取了该对象的锁才能访问.使用synchronized关键字来标记一个方法或者代码块,当某个线程调用该对象的synchronized方法或者访问synchronized代码块时,这个线程便获得了该对象的锁,其他线程暂时无法访问这个方法,只有等待这个方法执行完毕或者代码块执行完毕,这个线程才会释放该对象的锁,其他线程才能执行这个方法或者代码块.
  2.2同步块

				public class ThreadDemo {

					public static void main(String[] args) {
						Ticket ticket = new Ticket();
						new Thread(ticket).start();
						new Thread(ticket).start();
						new Thread(ticket).start();
					}
				}

				class Ticket implements Runnable {
					final Random r = new Random();
					private int num = 10;
					private Object obj=new Object();
					@Override
					public void run() {
						while (true) {
							synchronized(obj){
								if (num > 0) {
									try {
										Thread.sleep(r.nextInt(100));
									} catch (InterruptedException e) {
									}
									System.out.println(Thread.currentThread().getName() + "..."
											+ num--);
								} else {
									break;
								}
							}
						}
					}
				}

 

 2.3同步方法

				public class Test {

					public static void main(String[] args) {
						Ticket ticket = new Ticket();
						new Thread(ticket).start();
						new Thread(ticket).start();
						new Thread(ticket).start();
					}
				}

				class Ticket implements Runnable {
					final Random r = new Random();
					private int num = 10;
					private boolean isEmpty = false;

					@Override
					public void run() {
						while (!isEmpty) {
							sell();
						}
					}

					public synchronized void sell() {

						if (num > 0) {
							try {
								Thread.sleep(r.nextInt(100));
							} catch (InterruptedException e) {
							}
							System.out
									.println(Thread.currentThread().getName() + "..." + num--);
						} else {
							isEmpty = true;
						}
					}
				}

 

当一个线程正在访问一个对象的synchronized方法,那么其他线程不能访问该对象的其他synchronized方法,但是能访问该对象的非synchronized方法;如果一个线程A需要访问对象object1的synchronized方法fun1,另外一个线程B需要访问对象object2的synchronized方法fun1,即使object1和object2是同一类型,因为他们访问的是不同的对象,不会产生线程安全问题.同步方法是同步块的简写,效率相对较低.

2.4类的锁
    每个类也会有一个锁,它可以用来控制对static数据成员的并发访问.改进后线程安全的单例模式:

				public class Single {

					private static Single instance;

					private Single() {
					}

					public static Single getInstance() {
						if (instance == null) {// 解决了效率问题
							synchronized (Test.class) {// 解决了并发安全
								if (instance == null) {
									instance = new Single();
								}
							}
						}
						return instance;
					}
				}

 

3.并发中的三个概念

  3.1原子性    即一个操作或者多个操作要么全部执行并且执行的过程不会被任何因素打断,要么就都不执行.java内存模型只保证了基本读取和赋值是原子性操作.由于synchronized和Lock能够保证任一时刻只有一个线程执行该代码块,保证了大范围的原子性.
  3.2可见性
    当多个线程访问同一个变量时,一个线程修改了这个变量的值,其他线程能够立即看得到修改的值.java中通过volatile修饰的变量可以保证可见性.
  3.3有序性
    程序执行的顺序按照代码的先后顺序执行.
    指令重排序:处理器为了提高程序运行效率,可能会对输入代码进行优化,它不保证程序中各个语句的执行先后顺序同代码中的顺序一致,但是它会保证程序最终执行结果和代码顺序执行的结果是一致的.
    java内存模型具备一些先天的"有序性",不需要通过任何手段就能够得到保证的有序性,通常也称为 happens-before原则.如果两个操作的执行次序无法从happens-before原则推导出来,那么它们就不能保证它们的有序性,虚拟机可以随意地对它们进行重排序.

4.happens-before原则

  1)程序次序规则:一个线程内按照代码顺序,在前面的操作先行发生于在后面的操作
  2)锁定规则:一个unLock操作先行发生于后面对同一个锁的lock操作
  3)volatile变量规则:对一个变量的写操作先行发生于后面对这个变量的读操作
  4)传递规则:如果操作A先行发生于操作B,而操作B又先行发生于操作C,则可以得出操作A先行发生于操作C
  5)线程启动规则:Thread对象的start()方法先行发生于此线程的每个一个动作
  6)线程中断规则:对线程interrupt()方法的调用先行发生于被中断线程的代码检测到中断事件的发生
  7)线程终结规则:线程中所有的操作都先行发生于线程的终止检测,我们可以通过Thread.join()方法结束、Thread.isAlive()的返回值手段检测到线程已经终止执行
  8)对象终结规则:一个对象的初始化完成先行发生于他的finalize()方法的开始

5.修饰符:volatile

  使用volatile修饰变量,保证其可见性并且禁止进行指令重排序.要实现线程安全必须保证变量是独立的,且对变量的操作是原子性的.volatile一般用于状态标志量.

3
3
分享到:
评论
4 楼 戢_时光 2015-07-31  
xiao1zhao2 写道
liaokangli 写道
2.3同步方法 程序还是存在并发问题

我这里测试没有问题,哪里有问题还请指出


当然是存在并发问题 你的程序是很对的  (// 解决了并发安全  ) 这个是重点。
3 楼 xiao1zhao2 2015-07-30  
liaokangli 写道
2.3同步方法 程序还是存在并发问题

我这里测试没有问题,哪里有问题还请指出
2 楼 liaokangli 2015-07-30  
2.3同步方法 程序还是存在并发问题
1 楼 戢_时光 2015-07-30  

转一个。

相关推荐

    Java并发:基础线程同步

    In this chapter, we will cover: 1.Synchronizing a method; 2.Arranging independent attributes in synchronized classes; 3.Using conditions in synchronized code; 4.Synchronizing a block of code with a ...

    Java并发编程:同步容器

    为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器、并发容器、阻塞队列、Synchronizer(比如CountDownLatch)。我们来讨论下同步容器。  一.为什么会出现同步容器?  在...

    Java并发编程:设计原则与模式(第二版)

    java并发方面的两大名著之一。读者将通过使用java.lang.thread类、synchronized和volatile关键字,以及wait、notify和notifyall方法,学习如何初始化、控制和协调并发操作。此外,本书还提供了有关并发编程的全方位...

    java并发编程:juc、aqs

    Java 并发编程中的 JUC(java.util.concurrent)库以及其核心组件 AQS(AbstractQueuedSynchronizer)在构建高性能、可伸缩性的多线程应用方面具有重要的地位。 AQS 是 JUC 中的核心组件,它提供了一个框架,让...

    java并发编程:juc线程池

    通过深入了解 Java 并发编程和 JUC 线程池,您可以更好地管理线程之间的协作和同步,充分发挥多核处理器的优势,确保程序稳定运行。 了解 Java 并发编程和 JUC 线程池是现代软件开发者的必备知识。通过掌握并发编程...

    Java并发:同步容器&并发容器

     在Java并发编程中,经常听到同步容器、并发容器之说,那什么是同步容器与并发容器呢?同步容器可以简单地理解为通过synchronized来实现同步的容器,比如Vector、Hashtable以及SynchronizedList等容器,如果有多个...

    java并发编程:线程基础

    本资源致力于向您介绍 Java 并发编程中的线程基础,涵盖了多线程编程的核心概念、线程的创建和管理,以及线程间通信的基本方法。通过深入学习,您将建立扎实的多线程编程基础,能够更好地理解和应用多线程编程。 多...

    Java多线程编程总结

    Java线程:线程的同步-同步方法 Java线程:线程的同步-同步块 Java线程:并发协作-生产者消费者模型 Java线程:并发协作-死锁 Java线程:volatile关键字 Java线程:新特征-线程池 Java线程:新特征-有返回值的线程 ...

    java多线程编程总结

    Java线程:线程的同步-同步方法 Java线程:线程的同步-同步块 Java线程:并发协作-生产者消费者模型 Java线程:并发协作-死锁 Java线程:volatile关键字 Java线程:新特征-线程池 Java线程:新特征-有返回值的线程 ...

    Java高并发编程,构建并发编程知识体系,提升面试成功率

    本课程将结合大量图示及代码演示,带你掌握多线程并发编程(线程安全,线程调度,线程封闭,同步容器等)与高并发处理思路与手段(扩容,缓存,队列,拆分等),构建完整的并发与高并发知识体系,胜任实际开发中并发...

    Java并发编程实战

    1.1 并发简史 1.2 线程的优势 1.2.1 发挥多处理器的强大能力 1.2.2 建模的简单性 1.2.3 异步事件的简化处理 1.2.4 响应更灵敏的用户界面 1.3 线程带来的风险 1.3.1 安全性问题 1.3.2 活跃性问题 1.3.3 ...

    Java 线程总结

    Java线程:线程的同步-同步方法 Java线程:线程的同步-同步块 Java线程:并发协作-生产者消费者模型 Java线程:并发协作-死锁 Java线程:volatile关键字 Java线程:新特征-线程池 Java线程:新特征-有返回值的线程 ...

    Java Core Sprout:基础、并发、算法

    同步关键字原理 多线程的三大核心 对锁的一些认知 ReentrantLock实现原理 ConcurrentHashMap 的实现原理 如何优雅地使用和理解线程池 深入理解线程通信 一个线程召集的诡异事件 线程池中你不可错过的一些细节 『ARM...

    java高并发相关知识点.docx

    线程安全:Java中的线程安全,包括同步方法和同步块等。 死锁:Java中的死锁,包括如何避免死锁和如何解除死锁。 性能优化:Java中的性能优化,包括JVM参数调优、代码优化、使用并发框架等。 并行计算:Java中的并行...

    Java 7并发编程实战手册

    《Java 7并发编程实战手册》是Java 7并发编程的实战指南,介绍了Java 7并发API中大部分重要而有用的机制。全书分为9章,涵盖了线程管理、线程同步、线程执行器、Fork/Join框架、并发集合、定制并发类、测试并发应用...

    java多线程笔记

    二、Java中的线程 3 三、Java中关于线程的名词解释 3 四、线程的状态转换和生命周期 4 Java线程:创建与启动 7 Java线程:线程名称的设定及获取 10 Java线程:线程栈模型与线程的变量 12 Java线程:线程的调度-休眠 ...

    java内存模型与并发技术

    阿里巴巴专家讲座——java内存模型与并发技术。 主要内容: 学习java并发理论基础:Java Memory Model 学习java并发技术基础:理解同步是如何工作 分析程序什么时候需要同步 几个典型的并发设计策略

    并发Java程序同步操作的有效删除

    并发Java程序同步操作的有效删除

Global site tag (gtag.js) - Google Analytics