• 0
  • 0
分享
  • 用好单例设计模式,代码性能提升300%——软件测试圈
  • 恬恬圈 2023-05-12 16:09:49 字数 3374 阅读 1048 收藏 0

  大家好,今天给大家分享一个写代码的设计模式,就是我们最最耳熟能详的单例设计模式。

  可能很多人都听说过这个单例设计模式了,甚至都写的贼溜,但是今天给大家说说用这个这个单例设计模式,咱们是怎么把代码的性能大幅度提升的,单例模式跟代码性能的关系,恐怕很多兄弟还没认真研究过呢!

  一次请求执行流程

  首先我们先来看看什么叫做单例模式,要理解单例模式,我们就得先说说不用单例模式的时候,我们平时创建对象是怎么弄的。

  平时创建对象这个简单吧,比如我们搞一个对外的web接口,然后再接口收到一个请求的时候,就创建一个对象。

  这个伪代码如下:

  @RestController("/user")
  public class Controller {
    
    private UserService userService;
    
    @RequestMapping("/create")
    public Response create(CreateUserRequest request) {
      User user = new User(request);
      
      UserService userService = new UserService();
      userService.add(user);
      
      return Response.success();
    }
    
  }

  上面那段代码极为的简单,假设你有一个Controller对外提供一个http接口,然后每次你通过浏览器发送一个创建用户的请求。

  也就是针对/user/create这个url的请求,发送一个CreateUserRequest请求参数,代码里就会通过new关键字,搞出来一个User对象,然后再通过new关键字创建一个UserService组件来,接着把User对象交给UserService组件去插入这个用户数据到数据库里去,这段代码基本但凡是懂java的应该都能看懂。

  但是这里有一个问题,大家知道每次处理请求的时候,这段代码运行他会干什么事情吗?

  其实有最关键的点就是,他每次请求过来都会在内存里创建一个User对象和一个UserService对象,那这些对象是如何创建的呢?

  java代码是如何运行的?

  下面就得给大家来揭秘一下这个代码运行的底层原理了,首先呢,当我们启动一个java程序的时候,一定会启动一个jvm进程,比如说上面那段代码,你可能是通过spring boot这类框架用main方法启动的,也可能是把他打包以后放到tomcat里去运行的。

  如果你是直接运行main方法来启动的,那么就会直接启动一个jvm进程,如果你是把代码打包以后放tomcat里运行的,那么tomcat自己本身就是一个jvm进程,如下图。

  接着呢,其实你启动的JVM进程,会把你写好的代码加载到内存里来然后运行你写的代码,你的代码运行起来以后,他就可以干你希望他干的事情了,比如说接收浏览器发送的http请求,然后创建一些对象,插入数据库等等,如下图所示。

1-1.jpg

  那么这个时候,有一个很关键的点,就是你的代码运行的时候用new User()和new UserService()创建出来的对象扔哪儿去了?

  很简单,你的JVM进程是有一块自己的内存区域可以用的,而且就他可以用,这块区域叫做堆内存,这就类似于咱们自己家盖个小别墅,弄一块院子自己可以在里面种花种草一样,别人不能在你家院子里种黄瓜和大蒜,对不对,如下图。

1-2.jpg

  那么接着呢,上面我们写的那段代码,大家注意一下,每次收到一个请求,都会创建一个User对象和一个UserService对象,对不对?

  所以说,随着你不停的发送请求不停的发送请求,咱们的代码是不是会不停的创建对象不停的创建对象,然后咱们的堆内存里,**对象是不是就会变的越来越多,越来越多?如下图。

1-3.jpg

  堆内存满了后怎么办?

  那么我问大家一个问题,堆内存是一块内存空间,他是可以无限制的一直放入对象的吗?

  当然不是了,当你的对象越来越多,太多的时候,就会把这块内存空间给塞满,塞满了以后他就放不下新的对象了,这个时候怎么办呢?

  他会触发一个垃圾回收的动作,就是JVM进程自己偷偷摸摸开了一个垃圾回收线程,这个线程就专门盯着我们的堆内存,感觉他快满了,就把里面的对象清理掉一部分,这就叫做垃圾回收,如下图。

1-5.jpg

  但是每次垃圾回收都有一个问题,他因为要清理掉一些对象,所以往往会在清理对象的时候,避免你再创建新的对象了,不然就跟你妈妈打扫你的房间一样,人家一边在打扫垃圾,结果你还不停的吃东西往地下扔垃圾,你妈妈不打你屁股才怪,对吧?所以一般垃圾回收的时候,会让JVM进程停止工作,别创建新的对象了,如下图。

1-6.jpg

  那么在垃圾回收进行中,JVM进程停止运行的这个期间,是不是会导致一个问题,那就是你的用户发送过来的请求就没人处理了,没错,这个时候用户会感觉每次发送请求那是卡住,一直卡着没有返回,此时系统性能是处于一个极差的状态的,如下图。

1-7.jpg

  用单例模式如何优化系统性能呢?

  那么这个时候问题来了,回到这篇文章的主体,就是用单例模式如何优化系统性能呢?

  其实针对上面的问题,很多小伙伴可能已经发现了,如果想要优化系统性能,有一个关键的点就是尽量创建少一些的对象,避免堆内存频繁的塞满,也就可以避免频繁的垃圾回收,更可以避免频繁的JVM进程停顿,进而避免系统请求频繁的卡顿无响应。

  **那如何少创建一些对象呢?**单例模式就是一个很好的办法,对我们来说,完全可以让UserService这个对象就只创建一次,不要每次请求重复的创建他,让一个对象就创建一次,就是单例模式,单例模式有很多种写法,其中一种写法如下:

  @RestController("/user")
  public class Controller {
    
    private UserService userService;
    
    @RequestMapping("/create")
    public Response create(CreateUserRequest request) {
      User user = new User(request);
      
      UserService userService = UserSerivce.getInstance();
      userService.add(user);
      
      return Response.success();
    }
    
  }
  public class UserService {
    
    private UserService() {}
    
    private static class Singleton {
      static UserService userService = new UserService(); 
    }
    
    public static UserService getInstance() {
      return Singleton.userService;
    }
    
  }

  大家可以看到上面的代码,我们在UserService中定义了一个私有化的静态内部类Singleton,在Singleton里定义了一个静态变量UserService对象,这样的话,Singleton这个类只会被加载一次,只有类加载的时候才会实例化一个静态变量UserService对象,后续每次通过getInstance()方法都是直接获取这唯一一个对象就可以了,不会重复创建对象。

  这就是单例模式的一种写法,也是企业开发中最常用的一种写法,用了单例模式后,就可以大幅度降低我们创建的对象数量,避免堆内存频繁塞满,频繁垃圾回收,频繁JVM进程停顿影响请求性能,这样往往可以帮助我们更好的提升系统的性能。



作者:石杉的架构笔记    

来源:http://www.51testing.com/html/94/n-5099494.html

  • 【留下美好印记】
    赞赏支持
登录 后发表评论
+ 关注

热门文章

    最新讲堂

      • 推荐阅读
      • 换一换
          • 在Java中,异常情况分为Exception(异常)和Error(错误)两大类,Java异常通常是指程序运行过程中出现的非正常情况,如用户输入错误、除数为零、需要处理的文件不存在、数组下标越界等,对于异常情况的出现,可以采用异常处理,以保证程序可以正常的执行。Java中定义两种类型的异常和错误:1. JVM(Java虚拟机) 异常:由 JVM 抛出的异常或错误。例如:NullPointerException 类,ArrayIndexOutOfBoundsException 类,ClassCastException 类。2. ...
            0 0 927
            分享
          • Postman是一个可扩展的API开发和测试协同平台工具,可以快速集成到CI/CD管道中。旨在简化测试和开发中的API工作流。Postman 工具有 Chrome 扩展和独立客户端,推荐安装独立客户端。Postman 有个 workspace 的概念,workspace 分 personal 和 team 类型。Personal workspace 只能自己查看的 API,Team workspace 可添加成员和设置成员权限,成员之间可共同管理 API。当然我个人使用一般是不登录的,因为登录之后会自动将你的测试历史数据保存到账户里,你可以登陆网页端进行查看。 因为API的很多数据是很敏感的,...
            13 14 2339
            分享
          •   1、前言  大家好!我是Meng  前段时间,很荣幸被一合作伙伴邀请发表一篇文章,主题为"这些年,我所从事软件测试的一些感悟",正好趁着这个机会,我也好好总结一下。  2、测试培训  对于软件测试,在培训之前也不知道是干什么的,只知道有软件开发,之后才知道有软件测试这个行业。  刚开始也是在徘徊是做开发还是做测试,经过多次被测试机构的招生老师洗脑后,开始进行了软件测试培训之旅。那时候测试培训机构就那么几家,基本都是机构,个人的很少,培训机构是线下的,线上的也很少。培训价格方面也是过万了,跟现在的培训机构差不多,也就便宜了几千而已。  回头一想想,那时候教的内容,真的很少...
            0 0 1017
            分享
          • 1、目的本文档定义了软件缺陷管理流程和相关规则,确保软件缺陷管理的系统性和规范性,以保证项目研发质量。2、适用范围适用于部门项目研发过程的缺陷管理,对各阶段的缺陷管理过程进行指导和规范。3、定义3.1 术语缺陷(Defect):存在于软件之中偏差,可被激活,以静态形式存在于软件内部。Bug:缺陷一种表现形态,系统或程序存在的任何一种破坏正常运转能力的问题。3.2 缺陷定义(1)软件未达到需求规格说明书的功能; (2)软件出现了需求规格说明书指明不会出现的错误; (3)软件功能超出需求规格说明书的范围;(4)软件未达到需求规格说明书未指出但应达到的目标; (5)测试...
            12 12 1946
            分享
          • 软件测试面试中,介绍做过的项目,可以说是必不可少的一道面试题了,对于面试的同学来说,该自己发挥呢?把项目的所有功能噼里啪啦说一遍就完事了?当然不是,我们要搞清楚,面试官问这个题的目的。面试官主要想知道你参与项目的体量,你在项目中做了哪些有价值的事情,对整个项目起到了什么作用,你具备什么能力,这些能力是否能复用到新的工作岗位,能否带来更大的价值。清楚了面试官的目的,我们就可以开始发挥了,主要可以从以下几个方面展开:1、项目的背景主要包含项目面向的用户群体,用户量,常用的用户场景,甚至可以说下商业模式。例如这是一个B/S架构的系统,分为代理商前台和运营后台,用户量大概在5w左右,主要的用户场景是,...
            0 0 1363
            分享
      • 51testing软件测试圈微信