java高并发----个人学习理解汇总记录

1.首先,需要理解几个概念

1.同步(Synchronous):同步方法调用一旦开始,调用者必须等到前面的方法调用返回后,才能继续后续的行为,依次直到完成所有。

2.异步(Asynchronous):异步方法调用更像一个消息传递,一旦开始,方法调用就会立即返回,调用者就可以继续后续的操作。异步方法通常会在另外一个线程中“真实”地执行。整个过程,不会阻碍当前调用者的工作。

3.并发(Concurrency):在一个时间段内,多件事情在这个时间段内交替执行。比如说:有两个队列,进入同一个入口,交替进入。

4.并行(Parallelism):多件事情在同一个时刻同事发生。比如说:有两个队列,分别进如两个入口。互不干扰。

注意:实际上,如果系统内只有一个CPU,而使用多进程或者多线程任务,那么真实环境中这些任务不可能是真实并行的,毕竟一个CPU一次只能执行一条指令,在这种情况下多进程或者多线程就是并发的,而不是并行的(操作系统会不停地切换多任务)。真实的并行也只可能出现在拥有多个CPU的系统中(比如多核CPU)。

5.临界区:临界区用来表示一种公共资源或者说共享数据,可以被多个线程使用,但是每一次只能有一个线程使用它,一旦临界区资源被占用,其他线程要想使用这个资源就必须等待。

6.阻塞(Blocking):一个线程占用了临界区资源,其他的线程就必须在这个临界区中等待,会导致线程挂起,这种情况就是阻塞。

7.非阻塞(Non-Blocking):没有一个线程可以妨碍其他线程执行,所有的线程都会尝试不断向前执行。

8.死锁(Deadlock):几个线程都不释放资源,也得不到自己所需要的资源,导致线程都没办法继续进行下去。

注意:可以通过jstack命令进行查看,其中所需要的pid可以通过命令行获取到,也可以通过jdk提供的jconsole.exe工具直接查看当前线程的pid。不懂自行百度

9.饥饿(Starvation):指某一个或者多个线程因为种种原因无法获得所要的资源,导致一直无法执行。比如它的优先级可能太低,而高优先级的线程不断抢占它需要的资源,导致低优先级线程无法工作

10.活锁(Livelock):比如两个线程都秉承着“谦让”的原则,主动将资源释放给他人使用,可能会导致资源不断地在两个线程间跳动,而没有一个线程可以同时拿到所有资源正常执行。这种情况就是活锁

总结:这里的概念学习起来比较枯燥,多进行理解,孰能生巧。

2.并发级别概念

根据控制并发的策略,我们可以把并发的级别分为阻塞无饥饿无障碍无锁无等待几种。

1.阻塞:一个线程是阻塞的,那么在其他线程释放资源之前,当前线程无法继续执行。当我们使用synchronized关键字或者重入锁时,我们得到的就是阻塞的线程。

2.无饥饿:线程之间是有优先级的,那么线程调度的时候总是会倾向于先满足高优先级的线程。其中分为非公平锁与公平锁两种情况。公平锁不管新来的线程优先级多高,要想获得资源,就必须乖乖排队,这样所有的线程都有机会执行

3.无障碍:无障碍是一种最弱的非阻塞调度。两个线程如果无障碍地执行,那么不会因为临界区的问题导致一方被挂起。大家一起修改共享数据,把数据改坏了怎么办呢?对于无障碍的线程来说,一旦检测到这种情况,它就会立即对自己所做的修改进行回滚,确保数据安全。

4.无锁:无锁的并行都是无障碍的。在无锁的情况下,所有的线程都能尝试对临界区进行访问,但不同的是,无锁的并发保证必然有一个线程能够在有限步内完成操作离开临界区。

5.无等待:无锁只要求有一个线程可以在有限步内完成操作,而无等待则在无锁的基础上更进一步扩展。它要求所有线程都必须在有限步内完成,这样不会引起饥饿问题。如果限制这个步骤的上限,还可以进一步分解为有界无等待和线程数无关的无等待等几种,他们之间的区别只是对循环次数的限制不同

3.并行的两个重要定律

为什么要使用并行程序的问题前面已经进行了简单的探讨。总的来说,最重要的应该是处于两个目的。

第一,为了获得更好的性能;

第二,由于业务模型的需要,确实需要多个执行实体。

在这里,更加关注第一种情况,也就是有关性能的问题。将串行程序改造为并发程序,一般来说可以提高程序的整体性能,但是究竟能提高多少,甚至说究竟是否真的可以提高,还是一个需要研究的问题。目前,主要有两个定律对这个问题进行解答,一个是Amdahl定律,另外一个是Gustafson定律。

关于这两个定律,不是很理解,研究更深入客自行百度,总的来说,提升性能的方法:想办法提升系统并行的比例,同时增加CPU数量。

4.JMM(java内存模型)相关的一些概念

JMM关键技术点都是围绕着多线程的原子性、可见性、有序性来建立的

1.原子性:是指操作是不可分的,要么全部一起执行,要么不执行。在java中,其表现在对于共享变量的某些操作,是不可分的,必须连续的完成。比如a++,在操作a++的过程中,其他线程不会改变a的值。

注意:java中实现原子操作的方法大致有2种:锁机制、无锁CAS机制

2.可见性:可见性是指一个线程对共享变量的修改,对于另一个线程来说是否是可以看到的。比如:线程A对共享变量的修改要被线程B及时看到的话,需要进过以下步骤:1.线程A在自己的工作内存中修改变量之后,需要将变量的值刷新到主内存中 2.线程B要把主内存中变量的值更新到工作内存中

注意:可以使用volatilesynchronized来实现

3.有序性:有序性指的是程序按照代码的先后顺序执行。为了性能优化,编译器和处理器会进行指令冲排序,有时候会改变程序语句的先后顺序

5.理解进程和线程

1.进程:我们经常使用windows系统,经常会看见.exe后缀的文件,双击这个.exe文件的时候,这个文件中的指令就会被系统加载,那么我们就能得到一个关于这个.exe程序的进程。

2.线程:线程是轻量级的进程,是程序执行的最小单元,使用多线程而不是多进程去进行并发程序的设计,是因为线程间的切换和调度的成本远远小于进程,线程的所有状态在java.lang.Thread中的State枚举中有定义:

  New:表示刚刚创建的线程,这种线程还没有开始执行

  RUNNABLE:运行状态,线程的start()方法调用后,线程会处于这种状态

  BLOCKED:阻塞状态。当线程在执行的过程中遇到了synchronized同步块,但这个同步块被其他线程已获取还未释放时,当前线程将进入阻塞状态,会暂停执行,直到获取到锁。当线程获取到锁之后,又会进入到运行状态(RUNNABLE)

  WAITING:等待状态。和TIMEWAITING都表示等待状态,区别是WAITING会进入一个无时间限制的等,而TIMEWAITING会进入一个有限的时间等待,那么等待的线程究竟在等什么呢?一般来说,WAITING的线程正式在等待一些特殊的事件,比如,通过wait()方法等待的线程在等待notify()方法,而通过join()方法等待的线程则会等待目标线程的终止。一旦等到期望的事件,线程就会再次进入RUNNABLE运行状态。

  TERMINATED:表示结束状态,线程执行完毕之后进入结束状态。

注意:从NEW状态出发后,线程不能在回到NEW状态,同理,处理TERMINATED状态的线程也不能在回到RUNNABLE状态 

个人理解:单个CPU一次只能运行一个任务,任一时刻,CPU总是运行一个进程,其他进程处于非运行状态。其中,一个进程可以包括多个线程,一个进程的内存空间是共享的,每个线程都可以使用这些共享内存。需要注意的是:一个线程使用某些共享内存时,其他线程必须等它结束,才能使用这一块内存

注意:以上全部都是基本概念的理解,对于高并发的代码技巧掌握,不要急于求成,更多的是要着重于每一个基础知识点的逐步积累,万丈高楼平地起,一起加油骚年们。

原文地址:https://www.cnblogs.com/Roger2Lj/p/11562509.html

时间: 2024-10-12 08:44:51

java高并发----个人学习理解汇总记录的相关文章

java 高并发知识点学习总结(一)

并发 : 同时拥有两个或多个线程,如果程序在单核处理器上运行,多个线程交替的换入或者换出内存,这些线程是同时存在的,每个线程都处于执行过程中的某个状态,如果运行在多核处理器上,此时,程序中的每个线程都将分配到一个处理器核上,因此可以同时运行. 为什么需要cpu cache: cpu 的频率太快,快到主存跟不上,这样在处理器始终周期内,CPU常常需要等待主存,浪费资源.所以cache得出现,是为了缓解CPU和内存之间速度的不匹配问题.   CPU cache 有什么意义:    1) 时间局部性:

JAVA高并发程序设计学习-JDK并发包:同步控制一

JDK内部提供了大量的API和框架,这里主要介绍三部分 多线程同步控制方法 线程池,提高线程调度的性能 JDK的并发容器 重入锁:java.util.concurrent.locks.ReenterLock 在代码中,类ReenterLock实现了Runnable,其中有static的变量i,在run()方法中会对i进行自增操作. 自增操作的步骤为:get-set,先获取值再增加值. 如果在这里不进行控制的话,会导致get的值不是最新set的值. 因此,在自增的时候使用锁进行控制,保证get-s

JAVA高并发程序设计学习:Synchronized同步代码块具体使用方法

多线程同时对资源进行访问时,同步机制使得同一时间内只能有一个线程对资源进行操作. 同步机制可以用Synchronized实现. 当Synchronized修饰一个方法的时候,该方法称为同步方法. 当Synchronized方法执行完成或者异常时会释放锁. 会有同学对synchronized修饰方法,静态方法,对象时具体对哪些东西加锁不是很明白,这里会进行详细的讲解. synchronized修饰方法时,会对类实例进行加锁,该实例的所有synchronized方法必须等当前锁释放后才能访问. sy

imooc课程:Java高并发秒杀API 记录

Java高并发秒杀API之业务分析与DAO层 Java高并发秒杀API之Service层 Java高并发秒杀API之web层 Java高并发秒杀API之高并发优化 除了并发部分外的这个web开发的总结:https://www.imooc.com/video/11737 springmvc运行流程 BOOTSTRAP 和 JS 的使用技巧.(模块化.不混乱.抽取字典方便修改) JQ插件(countDown && cookie 等) 原文地址:https://www.cnblogs.com/a

跟着阿里p7一起学java高并发 - 第18天:玩转java线程池,这一篇就够了

java中的线程池,这一篇就够了 java高并发系列第18篇文章. 本文主要内容 什么是线程池 线程池实现原理 线程池中常见的各种队列 自定义线程创建的工厂 常见的饱和策略 自定义饱和策略 线程池中两种关闭方法有何不同 扩展线程池 合理地配置线程池 线程池中线程数量的配置 什么是线程池 大家用jdbc操作过数据库应该知道,操作数据库需要和数据库建立连接,拿到连接之后才能操作数据库,用完之后销毁.数据库连接的创建和销毁其实是比较耗时的,真正和业务相关的操作耗时是比较短的.每个数据库操作之前都需要创

Java高并发秒杀API之业务分析与DAO层

课程介绍 高并发和秒杀都是当今的热门词汇,如何使用Java框架实现高并发秒杀API是该系列课程要研究的内容.秒杀系列课程分为四门,本门课程是第一门,主要对秒杀业务进行分析设计,以及DAO层的实现.课程中使用了流行的框架组合SpringMVC+Spring+MyBatis,还等什么,赶快来加入吧! 第1章 课程介绍 本章介绍秒杀系统的技术内容,以及系统演示.并介绍不同程度的学员可以学到什么内容. 第2章 梳理所有技术和搭建工程 本章首先介绍秒杀系统所用框架和技术点,然后介绍如何基于maven搭建项

Java高并发,如何解决,什么方式解决

对于我们开发的网站,如果网站的访问量非常大的话,那么我们就需要考虑相关的并发访问问题了.而并发问题是绝大部分的程序员头疼的问题, 但话又说回来了,既然逃避不掉,那我们就坦然面对吧~今天就让我们一起来研究一下常见的并发和同步吧. 为了更好的理解并发和同步,我们需要先明白两个重要的概念:同步和异步    1.同步和异步的区别和联系 所谓同步,可以理解为在执行完一个函数或方法之后,一直等待系统返回值或消息,这时程序是出于阻塞的,只有接收到 返回的值或消息后才往下执行其它的命令. 异步,执行完函数或方法

如何处理java高并发问题---之页面静态化

转载:如何处理java高并发问题---之页面静态化 解决java中的高并发问题可以从硬件软件等方面入手,硬件如:服务器:软件如:系统缓存.页面静态化等.这里我写的是页面静态化的简单小例子.之所以将这个小例子记录下来是因为之前对页面静态化有误解,原以为静态化页面就是在项目编写中前端页面利用html就可以.但是在参考网上资料后发现理解误差很大,以下是我在参考了一些网上资料后写的实现页面静态化的简单实例: 项目结构图 1.业务处理类Servlet.java(这里为了方便采用的servlet,可根据项目

java高并发

转自:https://www.cnblogs.com/lr393993507/p/5909804.html 对于我们开发的网站,如果网站的访问量非常大的话,那么我们就需要考虑相关的并发访问问题了.而并发问题是绝大部分的程序员头疼的问题, 但话又说回来了,既然逃避不掉,那我们就坦然面对吧~今天就让我们一起来研究一下常见的并发和同步吧. 为了更好的理解并发和同步,我们需要先明白两个重要的概念:同步和异步    1.同步和异步的区别和联系 所谓同步,可以理解为在执行完一个函数或方法之后,一直等待系统返