溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何在Java中使用HashMap改進查找性能

發布時間:2021-02-07 18:08:39 來源:億速云 閱讀:269 作者:Leah 欄目:開發技術

如何在Java中使用HashMap改進查找性能?很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

Java中,HashMap,其實就是鍵值對。一個Key,對應一個值;寫數據時,指定Key寫對應值;讀取時憑Key找到相應值。感覺就跟Redis差不多。

// 創建 HashMap 對象 Sites
HashMap<Integer, String> Sites = new HashMap<Integer, String>();
// 添加鍵值對
Sites.put(1, "Google");
Sites.put(2, "Runoob");
Sites.put(3, "Taobao");
Sites.put(4, "Zhihu");
//讀取
String val = Sites.get(1);//得到Google

為什么說可以用HashMap來改進性能呢?原因不是說HashMap這種數據結構存儲性能就比其他的,比如數組,集合先進多少。我主要看中的,是在知道Key的情況下,找到相應值得速度非???。如果是用數組,最簡單的,用循環;講究一點,排好序,用折半查找(二分查找)。都比不上用Key在HashMap里直接讀取。不知道為什么HashMap在查找方面為啥這么快,估計是存儲結構,使用了啥樹,并為Key建立了索引。這是另外一個課題,以后再了解。昨天,我只是利用了這個特性,將運行幾個小時都沒結束的問題,只耗費了十幾秒。

問題如下:
有25萬條記錄,每條記錄含經緯度;存在不同記錄坐標相同情況?,F在想將坐標相同的記錄歸并在一起。

如果數據是保存在數據庫里,那么用SQL進行坐標分組,應該能解決問題。然而并沒有數據庫,數據是從gdb文件里讀出來的。

好吧,將數據保存到數組里,再新建一個集合;然后循環數組,與新集合中的記錄逐個比較,坐標相同就歸并到新集合,不同就插入新集合。最簡單了。結果2個小時過去了,還沒有結束的跡象。

想想也對,新集合越來越大,比較的次數也越來越多,仿佛棋盤里的大米一樣,每格的大米數量是前一格的兩倍;最后即使是整個國家糧庫的大米都放進去,都填不滿整個棋盤。

將25萬條記錄先排好序再處理?單是排序就忙死了,不行吧。

將25萬條記錄先保存到數據庫里,再分組?應該也可以,但總覺得笨了一些,而且速度應該也是以分鐘算的。

最后決定用HashMap來做這個新集合。
如上所述,HashMap按照Key來寫入或讀取值。關鍵是這個Key怎么得來。上面的例子,是寫代碼的人自己給出了一些字符作為Key。而在我們項目中,可以用經緯度之和的哈希值來作為Key。哈希值相同的,就認為是經緯度相同,只需要判斷新集合中,是否存在這個Key對應的元素就可以了,根本無須循環比較。

由于存在兩個不同的經緯度加起來,結果是一樣的可能性,因此先將經度 乘以1000,再加緯度,這樣基本杜絕沖突的機會。

代碼如下:

private HashMap<Long,SimpleItem> recGeo(HashMap<Long, SimpleItem> map,String geo,int j){
  /*
    將相同坐標的記錄合成一條
    HashMap<Long, SimpleItem> map, 新集合
    String geo, 坐標字符串
    int j 記錄ID
   */

  try {
    Point p = (Point)reader.read(geo);
    /*
      計算哈希值
      因為如果采用循環來比較,數據量太大,速度太慢了
      為避免不同坐標出現經度+緯度結果相同的情況,將經度 * 1000再相加
     */
    //計算Key
    long k = Long.valueOf(Double.doubleToLongBits(p.getX() * 1000 + p.getY())).hashCode();
    
    SimpleItem si = map.get(k);
    if(si != null){//新集合中該Key對應元素已存在,應該是相同坐標的記錄
      si.getPointers().add(j);//歸并
    } else {//否則插入
      si = new SimpleItem();
      si.setGeo(geo);
      List<Integer> pointers = new ArrayList();
      pointers.add(j);
      si.setPointers(pointers);
      map.put(k,si);
    }
  } catch (ParseException e) {
    e.printStackTrace();
  }

  return map;
}

private static GeometryFactory geometryFactory = JTSFactoryFinder.getGeometryFactory( null );
private static WKTReader reader = new WKTReader( geometryFactory );
class SimpleItem{
  private Point geo;
  private List<Integer> pointers;

  public Point getGeo() {
    return geo;
  }

  public void setGeo(String geo) {
    try {
      this.geo = (Point)reader.read(geo);
    } catch (ParseException e) {
      e.printStackTrace();
    }
  }

  public List<Integer> getPointers() {
    return pointers;
  }

  public void setPointers(List<Integer> pointers) {
    this.pointers = pointers;
  }
}

看完上述內容是否對您有幫助呢?如果還想對相關知識有進一步的了解或閱讀更多相關文章,請關注億速云行業資訊頻道,感謝您對億速云的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女