黄视频网站在线免费观看-黄视频网站在线看-黄视频网站在线观看-黄视频网站免费看-黄视频网站免费观看-黄视频网站免费

千鋒教育-做有情懷、有良心、有品質(zhì)的職業(yè)教育機(jī)構(gòu)

手機(jī)站
千鋒教育

千鋒學(xué)習(xí)站 | 隨時(shí)隨地免費(fèi)學(xué)

千鋒教育

掃一掃進(jìn)入千鋒手機(jī)站

領(lǐng)取全套視頻
千鋒教育

關(guān)注千鋒學(xué)習(xí)站小程序
隨時(shí)隨地免費(fèi)學(xué)習(xí)課程

當(dāng)前位置:首頁  >  技術(shù)干貨  > 記一次OOM問題的解決連載一 【流式查詢】

記一次OOM問題的解決連載一 【流式查詢】

來源:千鋒教育
發(fā)布人:qyf
時(shí)間: 2023-03-02 20:44:00 1677761040

  一. 問題概述

  老師的一個(gè)學(xué)生入職了杭州中通全球創(chuàng)研中心,最近他給老師分享一個(gè)他們公司解決OOM問題的案例,老師覺得十分有趣,特意把這個(gè)案例記錄下來,日后我會(huì)做成教學(xué)案例分享給學(xué)生。這個(gè)問題發(fā)生的背景如下:

  【在物流領(lǐng)域,針對(duì)各個(gè)下級(jí)網(wǎng)點(diǎn)而言,每月1日~9日是進(jìn)行財(cái)務(wù)月結(jié)的重要時(shí)間節(jié)點(diǎn)。在這個(gè)關(guān)鍵節(jié)點(diǎn)上,各個(gè)網(wǎng)點(diǎn)需要使用導(dǎo)出功能輸出寄派件、費(fèi)用客戶信息等多種信息進(jìn)行匯總結(jié)算】。也就是說,在月初的時(shí)候,每個(gè)網(wǎng)點(diǎn)都要統(tǒng)計(jì)一個(gè)月的各種流水(寄件,收件等),最后再以excel表格的形式下載給客戶。

  那么在這個(gè)業(yè)務(wù)中為什么很容易發(fā)生OOM異常呢?這是因?yàn)槠骄?個(gè)網(wǎng)點(diǎn)1個(gè)月的流水?dāng)?shù)據(jù)大約在30w行左右,根據(jù)計(jì)算得出大約500行數(shù)據(jù)就會(huì)占用1M內(nèi)存,而1個(gè)站點(diǎn)把30萬行一股腦地讀到內(nèi)存中,就會(huì)占用600M內(nèi)存。試想一下,如果全國的網(wǎng)點(diǎn)都在月初集中下載報(bào)表的話,JVM是很容出現(xiàn)內(nèi)存溢出的問題的!

  二. 解決方案

  那么這樣一個(gè)棘手的問題,如果我們只用一個(gè)單一的解決方案是不夠的,老師根據(jù)學(xué)生的描述,建議該學(xué)生主要采取以下幾種解決方案。

  2.1 用硬盤空間置換內(nèi)存空間

  如果我們在接到統(tǒng)計(jì)數(shù)據(jù)請求的時(shí)候,一次性把30w條數(shù)據(jù)從數(shù)據(jù)庫讀取到一個(gè)List集合中,這顯然是不合理的,因?yàn)檫@樣一個(gè)List集合就會(huì)占用600M內(nèi)存。所以我們可以進(jìn)行分頁查詢,每次查詢1000條數(shù)據(jù),然后往硬盤里寫,多讀取幾次,一點(diǎn)一點(diǎn)的把所有的數(shù)據(jù)都讀出來,再一點(diǎn)一點(diǎn)的往硬盤中寫。這樣在這個(gè)過程中,占用的內(nèi)存就會(huì)少很多,主要變成了對(duì)硬盤空間的占用。而我們操作excel的技術(shù),可以選擇阿里巴巴的easyexcel。

  2.2 使用Mybatis的流式查詢

  我們可以使用Mybatis的【流式查詢】查詢技術(shù),在查詢成功后返回的是一個(gè)迭代器而不是一個(gè)集合,應(yīng)用每次都從迭代器中獲取一條查詢結(jié)果,能夠降低內(nèi)存的使用。試想一下,如果我們不使用流式查詢,而想要一次性從數(shù)據(jù)庫中讀取30萬條數(shù)據(jù),內(nèi)存是根本不夠用的!這時(shí)我們只能選擇分頁查詢,而分頁查詢的性能又取決于表設(shè)計(jì)以及索引的設(shè)計(jì),大量數(shù)據(jù)分頁查詢的性能是很低的。老師對(duì)比了使用流式查詢和分頁查詢的兩種方案,得到的結(jié)論是取30萬條數(shù)據(jù)時(shí),流式查詢的速度大約是分頁查詢的4~5倍左右。

  2.3 使用redission信號(hào)量限流

  生成一個(gè)月的流水報(bào)表是一個(gè)非常耗時(shí)的操作,用戶也不可能馬上就要結(jié)果,所以我這個(gè)學(xué)生的公司對(duì)同時(shí)生成報(bào)表的請求數(shù)量做了限制,同時(shí)只能處理10個(gè)報(bào)表的生成。在這期間如果再有生成報(bào)表的請求,我們將會(huì)讓這些請求排隊(duì),等到前面的報(bào)表生成完畢后,再處理后面的請求。報(bào)表生成成功后,再通知客戶主動(dòng)去下載,老師建議這里使用redisson分布式鎖的信號(hào)量來限制同時(shí)創(chuàng)建報(bào)表的線程數(shù)量。

  2.4 MQ解耦+微服務(wù)拆分

  本次業(yè)務(wù)中,讀數(shù)據(jù)庫,編寫excel文件,上傳到文件服務(wù)器這三個(gè)操作都非常耗時(shí),學(xué)生的公司使用了MQ解耦,并把這次請求拆分成3個(gè)微服務(wù),這樣讀、寫、上傳就不會(huì)相互影響了。

  三. 流式查詢

  在這篇文章中,老師只給大家分享一下Mybatis流式查詢的實(shí)現(xiàn)方法,其他的解決方案以后會(huì)在其他的文章中給大家呈現(xiàn)。

  3.1 概念

  流式查詢就是查詢成功后返回的是一個(gè)迭代器而不是一個(gè)集合,應(yīng)用每次都從迭代器中獲取一條查詢結(jié)果,這樣能夠降低內(nèi)存的使用。

  3.2 Mybatis實(shí)現(xiàn)流式查詢

  接下來就是實(shí)現(xiàn)流失查詢的具體過程。

  在mapper映射文件中,編寫流式查詢的邏輯。

<!--
1: fetchSize: 官方文檔建議設(shè)置成Integer.MIN_VALUE
2: resultSetType="FORWARD_ONLY" 返回一個(gè)只向前的游標(biāo)
3:注意我把表一次性查出,并沒有使用分頁邏輯,依靠流式查詢一行一行得到結(jié)果
-->
<select id="selectFetchSize" fetchSize="-2147483648" resultSetType="FORWARD_ONLY" resultType="com.qf.shop.cms.entity.TContent">
select * from t_content
</select>

  在mapper接口文件中添加selectFetchSize方法。

  // 參數(shù) ResultHandler 是一個(gè)回調(diào)接口,也就是從游標(biāo)中獲得一條數(shù)據(jù)就會(huì)回調(diào)接口中的方法

  void selectFetchSize(ResultHandlerhandler);

  自己編寫一個(gè)類實(shí)現(xiàn)ResultHandler接口,在該接口中定義從游標(biāo)獲得一條數(shù)據(jù)后的回調(diào)邏輯。

  /**

  * 通過流式查詢每獲得一條數(shù)據(jù)的回調(diào)類

  */

  public class TContentResultHandler implements ResultHandler{

  /**

  * 這里每集滿1000條數(shù)據(jù) 往硬盤的excel文件中追加一次數(shù)據(jù)

  */

  private final static int BATCH_SIZE = 1000;

  /**

  * 計(jì)數(shù)器

  */

  private int size=0;

  /**

  * 存儲(chǔ)每批數(shù)據(jù)的臨時(shí)容器

  */

  private ListtContents = new ArrayList<>();

  /**

  * 每從流式查詢中獲得一行結(jié)果,就會(huì)調(diào)用一次這個(gè)方法

  * @param resultContext

  */

  @Override

  public void handleResult(ResultContext resultContext){

  // 這里獲取流式查詢每次返回的單條結(jié)果

  TContent resultObject = resultContext.getResultObject();

  // 你可以看自己的項(xiàng)目需要分批進(jìn)行處理或者單個(gè)處理,這里以分批處理為例

  tContents.add(resultObject);

  size++;

  if (size == BATCH_SIZE) {

  // 如果集滿1000條就往文件中寫一次

  handle();

  }

  }

  /**

  * 集滿1000條 執(zhí)行一次的邏輯

  */

  private void handle() {

  try {

  // 在這里可以對(duì)你獲取到的批量結(jié)果數(shù)據(jù)進(jìn)行需要的業(yè)務(wù)處理

  // 這里的業(yè)務(wù)是 往文件中寫一次

  } finally {

  // 處理完每批數(shù)據(jù)后后將臨時(shí)清空

  size = 0;

  tContents.clear();

  }

  }

  /**

  * 這個(gè)方法給外面調(diào)用,用來完成最后一批數(shù)據(jù)處理

  */

  public void end(){

  handle();// 處理最后一批不到BATCH_SIZE的數(shù)據(jù)

  }

  }

  在業(yè)務(wù)邏輯(service)層調(diào)用流式查詢方法。

  @Autowired

  private TContentMapper contentMapper;

  public void streamQuery(){

  // 生成流式查詢的回調(diào)對(duì)象

  TContentResultHandler tContentResultHandler = new TContentResultHandler();

  // 調(diào)用流式查詢

  contentMapper.selectFetchSize(tContentResultHandler);

  // 執(zhí)行完最后一批數(shù)據(jù)的邏輯

  tContentResultHandler.end();

  }

  四. 后話

  老師前面已經(jīng)說到,為了解決本次產(chǎn)生的OOM問題,老師給大家列舉了非常多的解決方案,但本篇文章介紹的流式查詢只是其中的方案之一。至于其他的解決方案,老師將在后續(xù)的文章中為大家一一揭曉,敬請各位繼續(xù)關(guān)注本公眾號(hào),如有問題,可以在評(píng)論區(qū)給我們留言哦。

tags:
聲明:本站稿件版權(quán)均屬千鋒教育所有,未經(jīng)許可不得擅自轉(zhuǎn)載。
10年以上業(yè)內(nèi)強(qiáng)師集結(jié),手把手帶你蛻變精英
請您保持通訊暢通,專屬學(xué)習(xí)老師24小時(shí)內(nèi)將與您1V1溝通
免費(fèi)領(lǐng)取
今日已有369人領(lǐng)取成功
劉同學(xué) 138****2860 剛剛成功領(lǐng)取
王同學(xué) 131****2015 剛剛成功領(lǐng)取
張同學(xué) 133****4652 剛剛成功領(lǐng)取
李同學(xué) 135****8607 剛剛成功領(lǐng)取
楊同學(xué) 132****5667 剛剛成功領(lǐng)取
岳同學(xué) 134****6652 剛剛成功領(lǐng)取
梁同學(xué) 157****2950 剛剛成功領(lǐng)取
劉同學(xué) 189****1015 剛剛成功領(lǐng)取
張同學(xué) 155****4678 剛剛成功領(lǐng)取
鄒同學(xué) 139****2907 剛剛成功領(lǐng)取
董同學(xué) 138****2867 剛剛成功領(lǐng)取
周同學(xué) 136****3602 剛剛成功領(lǐng)取
相關(guān)推薦HOT
python字符串截取?

在Python中,字符串是一種非常常見的數(shù)據(jù)類型,它可以用來表示文本、數(shù)字、符號(hào)等內(nèi)容。在實(shí)際應(yīng)用中,我們經(jīng)常需要對(duì)字符串進(jìn)行截取,以便獲取...詳情>>

2023-11-02 17:56:27
Python socket C/S結(jié)構(gòu)的聊天室應(yīng)用實(shí)現(xiàn)?

隨著互聯(lián)網(wǎng)的發(fā)展,聊天室應(yīng)用成為人們?nèi)粘I钪惺殖R姷囊环N社交方式。Python語言的Socket模塊是實(shí)現(xiàn)網(wǎng)絡(luò)通信的重要工具,可以輕松地實(shí)現(xiàn)C/...詳情>>

2023-11-02 17:53:38
用while求1到100的奇數(shù)和?

在計(jì)算機(jī)編程中,循環(huán)語句是非常重要的一部分。而while語句是其中最基本也是最常用的一種。它的作用是在滿足一定條件的情況下,重復(fù)執(zhí)行一段代...詳情>>

2023-11-02 17:50:57
python創(chuàng)建一個(gè)集合?

在Python中,集合是一種無序且不重復(fù)的數(shù)據(jù)類型,可以用于存儲(chǔ)一組元素。創(chuàng)建一個(gè)集合非常簡單,只需要使用大括號(hào){}或者set()函數(shù)即可。使用大...詳情>>

2023-11-02 17:34:02
linux改文件屬主命令?

Linux文件相關(guān)命令1、命令一:cat cat命令應(yīng)該是在Linux中查看文件內(nèi)容最常見的命令了。使用cat命令會(huì)打印指定文件的所有內(nèi)容到標(biāo)準(zhǔn)輸出上,比...詳情>>

2023-10-31 19:58:15
欧美国产日韩在线| 成人免费网站久久久| 国产不卡在线看| 麻豆污视频| 国产不卡在线看| a级毛片免费观看网站| 欧美激情中文字幕一区二区| 国产视频久久久| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 九九热国产视频| 欧美激情一区二区三区视频| 97视频免费在线| 精品久久久久久影院免费| 精品久久久久久中文字幕一区| 黄色免费三级| 欧美激情伊人| 99久久精品国产片| 欧美激情一区二区三区在线| 成人高清免费| 国产一区免费观看| 日韩免费在线| 高清一级淫片a级中文字幕 | 欧美国产日韩久久久| 成人免费观看男女羞羞视频| 麻豆系列 在线视频| 超级乱淫黄漫画免费| 99久久精品国产免费| 欧美大片一区| 日本伦理黄色大片在线观看网站| 国产伦久视频免费观看视频| 午夜在线亚洲| 成人高清视频在线观看| 日韩一级黄色| 天天做人人爱夜夜爽2020毛片| 精品视频免费在线| 久久精品大片| 国产高清在线精品一区a| 久久99中文字幕| 成人影院久久久久久影院| 亚欧成人毛片一区二区三区四区| 日本在线不卡免费视频一区| 美国一区二区三区| 日韩av东京社区男人的天堂| 99色精品| 一级女性大黄生活片免费| 日本特黄特色aaa大片免费| 国产成人精品综合在线| 国产视频网站在线观看| 久久久成人网| 国产麻豆精品免费视频| 欧美日本国产| 国产成人女人在线视频观看| 国产精品12| 亚州视频一区二区| 九九九网站| 日日爽天天| 青草国产在线观看| 韩国毛片| 国产a网| 国产综合成人观看在线| 香蕉视频久久| 精品在线观看一区| 精品久久久久久中文字幕一区| 免费一级片在线| 久久国产影院| 欧美a级片视频| 成人影院久久久久久影院| 99色视频在线观看| 日韩在线观看免费完整版视频| 国产美女在线观看| 成人影院一区二区三区| 99色吧| 天天做日日爱| 精品视频在线看 | 欧美爱爱动态| 国产亚洲精品成人a在线| 日韩免费片| 色综合久久手机在线| 国产极品精频在线观看| 国产美女在线观看| 日本免费看视频| 成人免费高清视频| 中文字幕一区二区三区 精品| 精品国产一区二区三区精东影业| 黄视频网站在线观看| 香蕉视频三级| 一a一级片| 青青青草视频在线观看| 久久成人性色生活片| 免费一级片在线| 国产精品1024永久免费视频| 欧美大片aaaa一级毛片| 精品国产香蕉在线播出| 韩国三级视频在线观看| 精品久久久久久中文| 一a一级片| 日韩中文字幕在线观看视频| 午夜在线亚洲| 亚洲第一色在线| 九九久久国产精品大片| 国产一区免费在线观看| 日韩在线观看免费完整版视频| 欧美大片a一级毛片视频| 国产高清视频免费观看| 日韩欧美一二三区| 久久成人亚洲| 国产高清在线精品一区a| 国产高清在线精品一区a| 国产不卡精品一区二区三区| 欧美国产日韩精品| 欧美大片a一级毛片视频| 精品视频在线观看免费| 九九精品在线播放| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 欧美18性精品| 欧美a级片免费看| 国产伦精品一区二区三区在线观看 | 日本免费看视频| 午夜激情视频在线观看| 黄视频网站在线免费观看| 可以免费看污视频的网站| 深夜做爰性大片中文| 日韩在线观看免费| 九九久久99| 国产国产人免费视频成69堂| 日本在线不卡视频| 亚洲第一色在线| 精品国产一区二区三区久久久蜜臀| 黄视频网站在线免费观看| 国产成人精品综合| 黄色短视屏| 日韩男人天堂| 午夜欧美成人久久久久久| 欧美国产日韩在线| 99色视频在线| 免费国产在线视频| 色综合久久天天综合| 亚州视频一区二区| 一本伊大人香蕉高清在线观看| 国产麻豆精品视频| 91麻豆精品国产自产在线观看一区 | 亚洲天堂一区二区三区四区| 香蕉视频久久| 国产视频一区二区在线播放| 国产视频久久久久| 成人免费网站久久久| 欧美日本韩国| 天堂网中文在线| 麻豆网站在线看| 国产高清在线精品一区二区| 久久精品大片| 国产91素人搭讪系列天堂| 韩国毛片免费| 日韩欧美一二三区| 国产一区二区精品在线观看| 国产不卡在线播放| 国产麻豆精品免费密入口| 韩国毛片免费大片| 久久99这里只有精品国产| 国产综合91天堂亚洲国产| 国产成人精品综合在线| 亚洲精品久久玖玖玖玖| 国产精品12| 99色吧| 成人影院久久久久久影院| a级毛片免费观看网站| 久久精品大片| a级黄色毛片免费播放视频| 欧美一级视频免费观看| 99久久精品国产国产毛片 | 国产高清在线精品一区a| 国产成人精品影视| 久久精品免视看国产明星| 国产麻豆精品hdvideoss| 精品国产亚一区二区三区| 九九久久国产精品| 一级毛片视频在线观看| 天天做日日爱| 国产精品自拍在线观看| 国产原创视频在线| 欧美另类videosbestsex视频 | 精品视频在线观看视频免费视频| 国产视频一区二区在线播放| 国产网站免费观看| 你懂的日韩| 沈樵在线观看福利| 久久精品欧美一区二区| 精品久久久久久中文字幕2017| 欧美a级v片不卡在线观看| 韩国三级视频在线观看| 九九九在线视频| 91麻豆精品国产自产在线| 免费一级片网站| 欧美a免费| 国产麻豆精品免费密入口| 国产一区精品| 一级毛片视频播放| 午夜欧美成人久久久久久| 国产一区二区精品| 毛片的网站|