首页
学习
活动
专区
工具
TVP
发布

吉林乌拉

专栏成员
113
文章
108793
阅读量
23
订阅数
搜索引擎
在现在互联网如日中天的时代,即使你不是互联网行业的人,你也一定会用过谷歌或者百度。因为他们已经影响了我们生活的方方面面,为我们提供了很多的便利。那么在互联网行业的人我们除了使用它们,我们还迫切地想知道它们到底是怎么实现的。
吉林乌拉
2020-11-13
1.2K0
设计模式之装饰者模式
在之前的设计模式文章中楼主已经介绍过了,要尽量针对接口编程,而不要针对实现编程。因为这样我们的程序比较方便扩展,又遵循了设计模式的基本原则。既然要针对接口编程,那么势必会创建大量的子类来实现。但有些时候并不是所有的业务都可以通过创建子类就可以实现的,反而通过创建大量子类,而增加了程序的不可扩展性。所以今天楼主分享一下设计模式中另一种模式叫装饰者模式。装饰者模式运用了对象组合的方式,可以做到在运行时动态的装饰类,这也是装饰者模式的由来。那么在介绍装饰者模式之前,我们和其他的设计模式一样,我们先看一个简单的例子。我们将以游戏中角色为例。我们知道在游戏中角色可以使用很多不同的武器,在使用不同的武器时,用户角色的攻击力就会不同,那么下面的例子我们将创建3个不同的武器分别为刀、剑、枪,并为这3个武器分别初始化不同的攻击力。下面为具体的代码。
吉林乌拉
2019-09-25
5430
设计模式之观察者模式
今天我继续和大家分享一下设计模式中的知识,今天我们来看一下观察者模式。观察者模式也可以叫发布订阅模式,在实际的场景中有很多时候会遇到这种设计模式。在现实的生活中可以将这种模式理解为报纸订阅服务。也就是说,如果用户订阅了某个报社的报纸,那么报社在收到这个订阅请求后,就会每天把最新的报纸送到用户的手中,如果某一天用户不想继续看这家报社的报纸了,那么就可以取消这个订阅,那么这时报社又收到这个用户取消订阅的请求,然后把这个用户从以后的送报纸用户的名单中删除掉。所以第二天在给其他用户送报纸的时候,就不会继续给这个用户送了。通过上面这个小的例子使我们知道所谓观察者也就是上述例子中的用户,那么这个用户在观察什么呢?答案可显而知,也就是观察订阅这家报社的报纸有没有最新的,如果有最新的报纸,那么报社就会自动将新的报纸,送到自己的手中。那么报纸如果没有最新的呢,也就是没有更新呢?例如在法定假日期间,(并不是所有的报纸都是按天发版的,还有一些报纸法定假日停刊),那么这时用户就不会收到新的报纸了。
吉林乌拉
2019-09-20
5790
数据库事务管理
在日常的软件开发中除了需要考虑软件性能指标外,还需要特别考虑的地方就是软件的安全性了,提到安全性,那我们就不得不考虑事务管理。也就是在同一个事务下,对数据库的操作要么全成功提交,只要有一个失败,那么已经更新的数据也必须回滚。在spring中使用事务比较简单,因为spring中不但提供了和底层事务无关的事务对象,还提供了声明性事务的功能,目的是让程序从事务代码中解耦,方便我们随时随地的添加事务。
吉林乌拉
2019-09-12
5970
ThreadPoolExecutor的使用
ThreadPoolExecutor也就是线程池。它就是Java为我们开发多线程程序时提供的一个开发框架。它可以统一的管理线程的创建、销毁、优化、监控等,在使用线程池时比我们直接使用原始的线程类更加方便。既然线程池这么方便,那它到底是怎么实现上述的功能呢?下面我们先看一下当用线程池启动一个线程时它的流程图。
吉林乌拉
2019-08-15
2.1K0
多线程中的死锁是啥意思?
死锁是在开发多线程时才会遇到的。原因就是不同的线程都在等待其它线程释放锁,而其它线程由于一些原因迟迟没有释放,这就造成了所有的线程都开始等待程序出现了假死的现象。说白了这就是一个BUG。我们用下面简单的程序来模拟一下死锁发生的现象。
吉林乌拉
2019-08-14
1.1K0
Java中volatile关键字的使用
volatile关键字的作用就是使变量在多个线程间可见。这到底是什么意思呢?我们先看下面的事例然后在详细说明。
吉林乌拉
2019-08-14
8280
join()方法的使用
在多线程开发中常常遇到了一个问题就是希望某一个线程在执行完毕后在执行主线程。在Java中通常用join()方法来实现需求。join()方法的目的是所属的线程正常运行run()方法中的逻辑,当前线程无限期阻塞,也就是说一直等待所属线程执行完,当前线程才会执行,底层实现原理是调用了wait()方法。
吉林乌拉
2019-08-14
6630
ThreadLocal详细的使用
我们知道在Java中用static关键字可以实现变量的共享,那么在多线程环境中可以用ThreadLocal让每一个线程都有自己的私用数据。首先我们先看一下共享变量的实例。
吉林乌拉
2019-08-14
7010
Jvm内存划分
在Java开发中我们不用考虑对内存的管理,是因为Jvm帮我们做了很多工作。Jvm为了满足不同的用途将内存大体划分了下面几部分。如下图:
吉林乌拉
2019-08-14
6200
当Jvm遇到new关键字
在上一篇我们知道了Jvm的内存划分,这一篇我们来具体分析一下,当我们用new关键字创建一个新对象时,Jvm都做了哪些工作。当虚拟机执行到new指令时,发现它是关键字,于是会检查这个指令参数是否能在常量池中找到相关的引用,并检查这个引用所代表的类是否被加载、解析和初始化过。如果没有被加载,则先执行类的加载过程。当类加载成功后,虚拟机就会为新创建的对象分配内存。分配内存无非就是在Java堆上划出一部分区域来作为新对象的存储空间。但在实际的处理时是比较麻烦的。例如怎么找到空闲的内存,如何划分固定大小内存。我们知道Java虚拟机规范中所说虚拟机中的堆内存可以是连续的也可以是不连续的。所以虚拟机在为对象分配新内存时,就要根据这两种不同的方式来采取不同的实现。这两种方式分别叫作:指针碰撞和空闲列表。正面我们分别看一下它们具体的实现方式。
吉林乌拉
2019-08-14
5380
OutOfMemoryError异常
我们知道在Java虚拟机内存中,除了程序计数器外,其它的内存区域都可能会发生OutOfMemoryError异常。本文将用具体的事例来演示在什么情况下会出现OutOfMemoryError异常,并以此来演示一下相关的虚拟机参数。
吉林乌拉
2019-08-14
3920
垃圾收集算法
垃圾收集器是Java虚拟机中自带的功能,它的目的是帮助我们管理内存,正是因为有它的存在所以,我们在开发时,基本不用考虑内存溢出等问题。基本不用考虑不代表,一定不会遇到内存溢出等问题,在上一篇中我们用简单的方法模拟了一些内存溢出的问题,在这一篇我们将重点分享一下,Java中垃圾收集器的一些实现算法知识,了解这方面的知识有助于,在内存溢出时,快速的解决问题。垃圾收集器的主要功能有3个。
吉林乌拉
2019-08-14
4190
类加载
我们知道在运行Java程序时,首先需要把源代码编译成二进制文件也就是class文件,然后虚拟机才能执行。那虚拟机在执行class文件时,都进行了哪些步骤呢。下面我们将详细分享一下。当类也就是class文件被加载到虚拟机内存开始,到卸载出内存为止。它将要执行以下7个步骤:
吉林乌拉
2019-08-14
4870
Java中类加载器
在上一篇类加载中我们介绍了一个类要加载到内存中要分为7个步骤。其中第一步就是加载也就是通过类的全限定名来获取类的二进制字节流。在Java中把上述加载的过程定义了一个模块叫做类加载器,目的是可以让用户自己决定如何加载一个类。类加载器虽然只是实现类的加载动作,但它在Java中起到的作用却远远要比类加载的功能要重要的多。原因就是类加载器在加载的过程中,会有一些特殊的特性来保证Java的运行安全。例如,每一个类加载器,都有一个独立的类名称空间。说白点就是如果要比较两个类是否相等,必须有一个前提,就是这两个类必须是同一个类加载器加载的,否则,即使比较的是同一个类,如果它们是由不同的类加载器加载的,那么这两个类也是不相等的。除了上述特性外,还有一个非常重要的特性就是双亲委派模式。在介绍双亲委派模式之前我们先看一下在虚拟机中一共都有哪些类加载器。
吉林乌拉
2019-08-14
5220
字节码执行引擎
在上一篇我们介绍了类加载器的相关功能,在这一篇中我们在分享一下虚拟机中的另一个非常重要的功能字节码执行引擎。我们知道Java虚拟机的主要任务就是加载class文件并执行其中的字节码。加载class的功能是由类加载器实现的,那么执行其中字节码的功能就是由字节码执行引擎执行的。下图为虚拟机的基本结构图。
吉林乌拉
2019-08-14
5700
Java内存模型
在Java虚拟机中定义了一种内存模型也就是JMM。目的是屏蔽各种硬件和操作系统的内存差异,以解决Java跨平台时能达到统一的内存访问效果。下面我们了解一下在JMM中内存是怎么划分的。
吉林乌拉
2019-08-14
4350
volatile变量
关键字volatile是虚拟机提供的一种轻量级的同步机制,但在做多线程开发时,大部分的人都不太习惯喜欢用volatile,而是直接用关键字synchronized来进行同步。但有些时候在解决线程安全问题时,使用volatile关键字要比使用synchronized同步函数更好一些。下面我们详细分析一下在Java中关键字volatile到底是一个什么样的东东,底层是怎么实现的。
吉林乌拉
2019-08-14
8210
ArrayList源码解析
上在一篇中我们已经介绍过了ArrayList集合类是List接口的实现类,所以它会默认具有List接口的相关特性。所以在这里我们就可以说ArrayList是一个能够保证元素的插入顺序并且可以保存重复元素的集合类。除了上述的特性外,ArrayList和其它集合类相比还可以保存null元素到集合类中(并不是所有的集合类都支持此功能)。ArrayList集合类底层是通过动态数组的方式实现的。动态数组的意思是说ArrayList的底层数组大小是可以动态改变的。我们知道在Java中数组的大小是不可以改变的,也就是说如果数组初始化成功,那么在使用时就一定是这么大的数组了。如果在使用时超过了数组的最大索引时,那么虚拟机就会抛出异常。既然Java中数组的大小是不可改变的,那么ArrayList底层是怎么实现动态数组功能的呢。
吉林乌拉
2019-08-14
3360
LinkedList源码解析
在这一篇中我们主要介绍LinkedList集合类。它和ArrayList不同的是,LinkedList底层是通过双向链表的方式实现的。下面我们介绍一下双向链表的知识。在上一篇中我们知道ArrayList底层数组在处理业务有一个很大的性能问题,就是如果我们从数组的中间位置要删除一个元素要付出很大的代价,原因就是将元素删除之后,这个元素后面的元素都要向数组的前端移动,所以会造成性能的损失,同样,在数组的中间位置插入元素时,也会有上述等问题。于是Java的设计者们为了解决ArrayList的性能问题时,于是LinkedList诞生了。因为它底层是采用双向链表的方式实现的,所以不会出现上述等问题。下面我们详细了解一下链表这个数据结构。
吉林乌拉
2019-08-14
3590
点击加载更多
社区活动
【纪录片】中国数据库前世今生
穿越半个世纪,探寻中国数据库50年的发展历程
Python精品学习库
代码在线跑,知识轻松学
博客搬家 | 分享价值百万资源包
自行/邀约他人一键搬运博客,速成社区影响力并领取好礼
技术创作特训营·精选知识专栏
往期视频·千货材料·成员作品 最新动态
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档