电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

Java中LinkedList真的是查找慢增刪快

瀏覽:2日期:2022-08-22 13:27:03

測試結果

廢話不多說,先上測試結果。作者分別在ArrayList和LinkedList的頭部、尾部和中間三個位置插入與查找100000個元素所消耗的時間來進行對比測試,下面是測試結果

(感謝@Hosalo的指正,在這里說明一下測試的環境,尾部插入是在空表的基礎上測試的,頭部和中間位置插入是在已存在100000個元素的表上進行測試的)

插入 查找 ArrayList尾部 26ms 4ms ArrayList頭部 2887ms 3ms ArrayList中間 1936ms 4ms LinkedList尾部 28ms 9ms LinkedList頭部 15ms 11ms LinkedList中間 12310ms 11387ms

測試結論

ArrayList的查找性能絕對是一流的,無論查詢的是哪個位置的元素 ArrayList除了尾部插入的性能較好外(位置越靠后性能越好),其他位置性能就不如人意了 LinkedList在頭尾查找、插入性能都是很棒的,但是在中間位置進行操作的話,性能就差很遠了,而且跟ArrayList完全不是一個量級的

源碼分析

我們把Java中的ArrayList和LinkedList就是分別對順序表和雙向鏈表的一種實現,所以在進行源碼分析之前,我們先來簡單回顧一下數據結構中的順序表與雙向鏈表中的關鍵概念

順序表:需要申請連續的內存空間保存元素,可以通過內存中的物理位置直接找到元素的邏輯位置。在順序表中間插入or刪除元素需要把該元素之后的所有元素向前or向后移動。 雙向鏈表:不需要申請連續的內存空間保存元素,需要通過元素的頭尾指針找到前繼與后繼元素(查找元素的時候需要從頭or尾開始遍歷整個鏈表,直到找到目標元素)。在雙向鏈表中插入or刪除元素不需要移動元素,只需要改變相關元素的頭尾指針即可。

所以我們潛意識會認為:ArrayList查找快,增刪慢。LinkedList查找慢,增刪快。但實際上真的是這樣的嗎?我們一起來看看吧。

測試程序

測試程序代碼基本沒有什么營養,這里就不貼出來了,但是得把程序的運行結果貼出來,方便逐個分析。

運行結果

ArrayList尾部插入100000個元素耗時:26msLinkedList尾部插入100000個元素耗時:28msArrayList頭部插入100000個元素耗時:859msLinkedList頭部插入100000個元素耗時:15msArrayList中間插入100000個元素耗時:1848msLinkedList中間插入100000個元素耗時:15981msArrayList頭部讀取100000個元素耗時:7msLinkedList頭部讀取100000個元素耗時:11msArrayList尾部讀取100000個元素耗時:12msLinkedList尾部讀取100000個元素耗時:9msArrayList中間讀取100000個元素耗時:13msLinkedList中間讀取100000個元素耗時:11387ms

ArrayList尾部插入

源碼

add(E e)方法 public boolean add(E e) { // 檢查是否需要擴容 ensureCapacityInternal(size + 1); // Increments modCount!! // 直接在尾部添加元素 elementData[size++] = e; return true; }

可以看出,對ArrayList的尾部插入,直接插入即可,無須額外的操作。

LinkedList尾部插入

源碼

LinkedList中定義了頭尾節點 /** * Pointer to first node. */ transient Node<E> first; /** * Pointer to last node. */ transient Node<E> last;

add(E e)方法,該方法中調用了linkLast(E e)方法

public boolean add(E e) { linkLast(e); return true; }

linkLast(E e)方法,可以看出,在尾部插入的時候,并不需要從頭開始遍歷整個鏈表,因為已經事先保存了尾結點,所以可以直接在尾結點后面插入元素

/** * Links e as last element. */ void linkLast(E e) { // 先把原來的尾結點保存下來 final Node<E> l = last; // 創建一個新的結點,其頭結點指向last final Node<E> newNode = new Node<>(l, e, null); // 尾結點置為newNode last = newNode; if (l == null) first = newNode; else // 修改原先的尾結點的尾結點,使其指向新的尾結點 l.next = newNode; size++; modCount++; }

總結

對于尾部插入而言,ArrayList與LinkedList的性能幾乎是一致的

ArrayList頭部插入

源碼

add(int index, E element)方法,可以看到通過調用系統的數組復制方法來實現了元素的移動。所以,插入的位置越靠前,需要移動的元素就會越多

public void add(int index, E element) { rangeCheckForAdd(index); ensureCapacityInternal(size + 1); // Increments modCount!! // 把原來數組中的index位置開始的元素全部復制到index+1開始的位置(其實就是index后面的元素向后移動一位) System.arraycopy(elementData, index, elementData, index + 1, size - index); // 插入元素 elementData[index] = element; size++; }

LinkedList頭部插入

源碼

add(int index, E element)方法,該方法先判斷是否是在尾部插入,如果是調用linkLast()方法,否則調用linkBefore(),那么是否真的就是需要重頭開始遍歷呢?我們一起來看看

public void add(int index, E element) { checkPositionIndex(index); if (index == size) linkLast(element); else linkBefore(element, node(index)); }

在頭尾以外的位置插入元素當然得找出這個位置在哪里,這里面的node()方法就是關鍵所在,這個函數的作用就是根據索引查找元素,但是它會先判斷index的位置,如果index比size的一半(size >> 1,右移運算,相當于除以2)要小,就從頭開始遍歷。否則,從尾部開始遍歷。從而可以知道,對于LinkedList來說,操作的元素的位置越往中間靠攏,效率就越低

Node<E> node(int index) { // assert isElementIndex(index); if (index < (size >> 1)) { Node<E> x = first; for (int i = 0; i < index; i++)x = x.next; return x; } else { Node<E> x = last; for (int i = size - 1; i > index; i--)x = x.prev; return x; } }

這個函數的工作就只是負責把元素插入到相應的位置而已,關鍵的工作在node()方法中已經完成了

void linkBefore(E e, Node<E> succ) { // assert succ != null; final Node<E> pred = succ.prev; final Node<E> newNode = new Node<>(pred, e, succ); succ.prev = newNode; if (pred == null) first = newNode; else pred.next = newNode; size++; modCount++; }

總結

對于LinkedList來說,頭部插入和尾部插入時間復雜度都是O(1) 但是對于ArrayList來說,頭部的每一次插入都需要移動size-1個元素,效率可想而知 但是如果都是在最中間的位置插入的話,ArrayList速度比LinkedList的速度快將近10倍

ArrayList、LinkedList查找

這就沒啥好說的了,對于ArrayList,無論什么位置,都是直接通過索引定位到元素,時間復雜度O(1) 而對于LinkedList查找,其核心方法就是上面所說的node()方法,所以頭尾查找速度極快,越往中間靠攏效率越低

到此這篇關于Java中LinkedList真的是查找慢增刪快 的文章就介紹到這了,更多相關Java LinkedList查找慢增刪快 內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
主站蜘蛛池模板: 汝成内控-行政事业单位内部控制管理服务商 | 振动筛-交叉筛-螺旋筛-滚轴筛-正弦筛-方形摇摆筛「新乡振动筛厂家」 | 地图标注|微信高德百度地图标注|地图标记-做地图[ZuoMap.com] | 袋式过滤器,自清洗过滤器,保安过滤器,篮式过滤器,气体过滤器,全自动过滤器,反冲洗过滤器,管道过滤器,无锡驰业环保科技有限公司 | EFM 022静电场测试仪-套帽式风量计-静电平板监测器-上海民仪电子有限公司 | 辐射色度计-字符亮度测试-反射式膜厚仪-苏州瑞格谱光电科技有限公司 | 必胜高考网_全国高考备考和志愿填报信息平台| 铸铝门厂家,别墅大门庭院大门,别墅铸铝门铜门[十大品牌厂家]军强门业 | 钢化玻璃膜|手机钢化膜|钢化膜厂家|手机保护膜-【东莞市大象电子科技有限公司】 | 牛奶检测仪-乳成分分析仪-北京海谊| 兰州UPS电源,兰州山特UPS-兰州万胜商贸 | 市政路灯_厂家-淄博信达电力科技有限公司 | 通信天线厂家_室分八木天线_对数周期天线_天线加工厂_林创天线源头厂家 | 混合气体腐蚀试验箱_盐雾/硫化氢/气体腐蚀试验箱厂家-北京中科博达 | 地源热泵一体机,地源热泵厂家-淄博汇能环保设备有限公司 | 浙江上沪阀门有限公司 | 设定时间记录电子秤-自动累计储存电子秤-昆山巨天仪器设备有限公司 | 陶瓷加热器,履带式加热器-吴江市兴达电热设备厂 | 【孔氏陶粒】建筑回填陶粒-南京/合肥/武汉/郑州/重庆/成都/杭州陶粒厂家 | 小型铜米机-干式铜米机-杂线全自动铜米机-河南鑫世昌机械制造有限公司 | 蓄电池在线监测系统|SF6在线监控泄露报警系统-武汉中电通电力设备有限公司 | 土壤水分自动监测站-SM150便携式土壤水分仪-铭奥仪器 | 工业设计,人工智能,体验式3D展示的智能技术交流服务平台-纳金网 J.S.Bach 圣巴赫_高端背景音乐系统_官网 | 植筋胶-粘钢胶-碳纤维布-碳纤维板-环氧砂浆-加固材料生产厂家-上海巧力建筑科技有限公司 | 防爆暖风机_防爆电暖器_防爆电暖风机_防爆电热油汀_南阳市中通智能科技集团有限公司 | 最新范文网_实用的精品范文美文网 | 重庆网站建设,重庆网站设计,重庆网站制作,重庆seo,重庆做网站,重庆seo,重庆公众号运营,重庆小程序开发 | 动库网动库商城-体育用品专卖店:羽毛球,乒乓球拍,网球,户外装备,运动鞋,运动包,运动服饰专卖店-正品运动品网上商城动库商城网 - 动库商城 | 创客匠人-让IP变现不走弯路| 【北京写字楼出租_写字楼租赁_办公室出租网/出售】-远行地产官网 | 东莞喷砂机-喷砂机-喷砂机配件-喷砂器材-喷砂加工-东莞市协帆喷砂机械设备有限公司 | 耐酸泵,耐酸泵厂家-淄博华舜耐腐蚀真空泵 | 分光色差仪,测色仪,反透射灯箱,爱色丽分光光度仪,美能达色差仪维修_苏州欣美和仪器有限公司 | 恒温振荡混匀器-微孔板振荡器厂家-多管涡旋混匀器厂家-合肥艾本森(www.17world.net) | 冰晶石|碱性嫩黄闪蒸干燥机-有机垃圾烘干设备-草酸钙盘式干燥机-常州市宝康干燥 | 真空泵维修保养,普发,阿尔卡特,荏原,卡西亚玛,莱宝,爱德华干式螺杆真空泵维修-东莞比其尔真空机电设备有限公司 | 北京康百特科技有限公司-分子蒸馏-短程分子蒸馏设备-实验室分子蒸馏设备 | 东莞动力锂电池保护板_BMS智能软件保护板_锂电池主动均衡保护板-东莞市倡芯电子科技有限公司 | crm客户关系管理系统,销售管理系统,crm系统,在线crm,移动crm系统 - 爱客crm | 乙炔气体报警装置|固定式氯化氢检测仪|河南驰诚电气百科 | 沈阳网站建设_沈阳网站制作_沈阳网页设计-做网站就找示剑新零售 沈阳缠绕膜价格_沈阳拉伸膜厂家_沈阳缠绕膜厂家直销 |