SparseArray解析

来源:http://www.cnblogs.com/ganchuanpu/archive/2017/04/17/6724487.html
-Advertisement-
Play Games

HashMap是java里比較常用的一個集合類,我們一般用來緩存一些處理後的結果。但當你做一個Android項目時,在代碼中定義這樣一個變數,實例化時,Eclipse卻給出了一個 performance 警告。 意思是說Map已經不用了,使用SparseArray<Object>代替,以獲取更好性能 ...


 HashMap是java里比較常用的一個集合類,我們一般用來緩存一些處理後的結果。但當你做一個Android項目時,在代碼中定義這樣一個變數,實例化時,Eclipse卻給出了一個 performance 警告。

 

   意思是說Map已經不用了,使用SparseArray<Object>代替,以獲取更好性能。為什麼用SparseArray呢,單從字面意思,SparseArray就是稀疏數組

 所謂稀疏數組就是數組中大部分的內容值都未被使用(或都為零),在數組中僅有少部分的空間使用。因此造成記憶體空間的浪費,為了節省記憶體空間,並且不影響數組中原有的內容值,我們可以採用一種壓縮的方式來表示稀疏數組的內容。

  假設有一個9*7的數組,其內容如下:

       圖 1 二維數組示例                                                                                                                                                                            

  在此數組中,共有63個空間,但卻只使用了5個元素,造成58個元素空間的浪費。以下我們就使用稀疏數組重新來定義這個數組:

 

      圖 2 使用稀疏數組進行壓縮

                                                                                                                                                                                  

其中在稀疏數組中第一部分所記錄的是原數組的列數和行數以及元素使用的個數、第二部分所記錄的是原數組中元素的位置和內容。經過壓縮之後,原來需要聲明大小為63的數組,而使用壓縮後,只需要聲明大小為6*3的數組,僅需18個存儲空間。

 

我們再來看看源碼中SparseArray到底做了哪些事情:

類結構

 

 


繼續閱讀SparseArray的源碼,從構造方法我們可以看出,它和一般的List一樣,可以預先設置容器大小,預設的大小是10:

 

 

也可以自定義容量

 

再來看看它對數據的“增刪改查”。

它有兩個方法可以添加鍵值對:

 

有四個方法可以執行刪除操作:

 

   修改數據起初以為只有setValueAt(int index, E value)可以修改數據,但後來發現put(int key, E value)也可以修改數據,我們查看put(int key, E value)的源碼可知,在put數據之前,會先查找要put的數據是否已經存在,如果存在就是修改,不存在就添加。

 

 

所以,修改數據實際也有兩種方法:

 

最後再來看看如何查找數據。有兩個方法可以查詢取值:

 

其中get(int key)也只是調用了 get(int key,E valueIfKeyNotFound),最後一個從傳參的變數名就能看出,傳入的是找不到的時候返回的值.get(int key)當找不到的時候,預設返回null。

查看第幾個位置的鍵:

 

有一點需要註意的是,查看鍵所在位置,由於是採用二分法查找鍵的位置,所以找不到時返回小於0的數值,而不是返回-1。返回的負值是表示它在找不到時所在的位置。

查看第幾個位置的值:

 

查看值所在位置,沒有的話返回-1:

 

最後,發現其核心就是折半查找函數(binarySearch),演算法設計的很不錯。

 

相應的也有SparseBooleanArray,用來取代HashMap<Integer, Boolean>,SparseIntArray用來取代HashMap<Integer, Integer>,大家有興趣的可以研究。

 

總結:SparseArray是Android里為<Interger,Object>這樣的Hashmap而專門寫的類,目的是提高記憶體效率,其核心是折半查找函數(binarySearch)。註意記憶體二字很重要,因為它僅僅提高記憶體效率,而不是提高執行效率,所以也決定它只適用於Android系統(記憶體對android項目有多重要,地球人都知道)。SparseArray有兩個優點:1.避免了自動裝箱(auto-boxing),2.數據結構不會依賴於外部對象映射。我們知道HashMap 採用一種所謂的“Hash 演算法”來決定每個元素的存儲位置,存放的都是數組元素的引用,通過每個對象的hash值來映射對象。而SparseArray則是用數組數據結構來保存映射,然後通過折半查找來找到對象。但其實一般來說,SparseArray執行效率比HashMap要慢一點,因為查找需要折半查找,而添加刪除則需要在數組中執行,而HashMap都是通過外部映射。但相對來說影響不大,最主要是SparseArray不需要開闢記憶體空間來額外存儲外部映射,從而節省記憶體。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • git:https://github.com/reg21st/vue2 management platform 訪問:https://reg21st.github.io/vue2 management platform 概述: 最近學習vue2.0和elementUI的使用,在各種文檔的幫助下,嘗試 ...
  • 語句 (建議)每條語句放在不同的行上並加上分號 first cnblogs;second cnblogs; 註釋 (建議)用"//" 來註釋單行,用"/*"註釋多行 //有註釋是好事 /*有註釋是好事 有註釋是好事 有註釋是好事 有註釋是好事*/ 變數 JS語法不允許變數名中包含空格或標點符號(美元 ...
  • 本文也同步發表在我的公眾號“我的天空” 上一期我們已經介紹了閉包,由於閉包可以延長函數內部的變數的生存周期,因此我們可以將不需要暴露在全局的變數封裝成函數的內部變數,從而避免代碼污染。 譬如要實現一個簡單的累加器,為了保存每次累加的結果,因此聲明瞭一個全局變數total,代碼如下: var tota ...
  • 文字溢出隱藏 如果你觀察過浮動元素,你會發現這樣一個事實,當前一個元素將寬度占滿以後,後一個元素就會往下掉,如下所示 代碼如下 也許在你眼裡這是個再正常不過的現象,不過有人卻將這個現象用在了實現文本溢出隱藏上,我們來看看他們是怎麼實現的。 先來看看它的效果,如下 實現這個效果的原理就是先將省略號通過 ...
  • 一、進程 進程可以理解為一個應用程式;比如說打開 QQ,那麼在記憶體中就會為 QQ 分配一塊進程;打開微信,那麼在記憶體中就會為 微信 分配一塊進程;並且進程之間是相互獨立的 二、線程 1. 基本概念 線程是進程的基本單元(可以理解為一個進程中的各個操作);比如說 QQ,其中的很多操作,比如說發送文件, ...
  • 一,效果圖。 二,工程圖。 三,代碼。 RootViewController.h #import <UIKit/UIKit.h> @interface RootViewController : UIViewController @end RootViewController.m #import "R ...
  • 怕有些人不知道怎麼進入微信的新消息提示音功能,我這裡說下操作步驟: 打開微信 我 設置 新消息提醒 新消息提示音。 經過以上的步驟就進入了這樣的界面 具體實現的步驟。 難點之一:獲取到手機系統的提示音,並將它們顯示在一個listview裡面。 參考如下代碼: 將獲取到的消息提示音的名字,加入到arr ...
  • 首先在看這個博客之前, 你可以先看下這個博客,http://blog.csdn.net/harryweasley/article/details/50057707 裡面介紹了兩種方式來獲取應用程式的信息,一種是packageInfo,一種是ResolveInfo,通過packageInfo來獲取應用 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...