成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Java? 教程(Map接口)

Kahn / 2862人閱讀

Map接口

Map是將鍵映射到值的對(duì)象,map不能包含重復(fù)的鍵:每個(gè)鍵最多可以映射一個(gè)值,它模擬數(shù)學(xué)函數(shù)抽象。Map接口包括基本操作的方法(如put、get、remove、containsKeycontainsValuesizeempty),批量操作(如putAllclear)和集合視圖(如keySet、entrySetvalues)。

Java平臺(tái)包含三個(gè)通用Map實(shí)現(xiàn):HashMap、TreeMap和LinkedHashMap,它們的行為和性能完全類似于HashSet、TreeSetLinkedHashSet,如Set接口部分所述。

本頁(yè)的其余部分詳細(xì)討論了Map接口,但首先,這里有一些使用JDK 8聚合操作收集到Map的示例,對(duì)現(xiàn)實(shí)世界對(duì)象進(jìn)行建模是面向?qū)ο缶幊讨械某R娙蝿?wù),因此可以合理地認(rèn)為某些程序可能會(huì)按部門對(duì)員工進(jìn)行分組:

// Group employees by department
Map> byDept = employees.stream()
.collect(Collectors.groupingBy(Employee::getDepartment));

或者按部門計(jì)算所有工資的總和:

// Compute sum of salaries by department
Map totalByDept = employees.stream()
.collect(Collectors.groupingBy(Employee::getDepartment,
Collectors.summingInt(Employee::getSalary)));

或者通過成績(jī)及格或成績(jī)不及格分組學(xué)生:

// Partition students into passing and failing
Map> passingFailing = students.stream()
.collect(Collectors.partitioningBy(s -> s.getGrade()>= PASS_THRESHOLD));

你還可以按城市分組:

// Classify Person objects by city
Map> peopleByCity
         = personStream.collect(Collectors.groupingBy(Person::getCity));

或者甚至級(jí)聯(lián)兩個(gè)收集器按州和城市對(duì)人進(jìn)行分類:

// Cascade Collectors 
Map>> peopleByStateAndCity
  = personStream.collect(Collectors.groupingBy(Person::getState,
  Collectors.groupingBy(Person::getCity)))

同樣,這些只是如何使用新JDK 8 API的幾個(gè)示例,有關(guān)lambda表達(dá)式和聚合操作的深入介紹,請(qǐng)參閱標(biāo)題為聚合操作的課程。

Map接口基本操作

Mapput、get、containsKey、containsValue、sizeisEmpty)的基本操作與Hashtable中的對(duì)應(yīng)操作完全相同,以下程序生成其參數(shù)列表中找到的單詞的頻率表,頻率表將每個(gè)單詞映射到它在參數(shù)列表中出現(xiàn)的次數(shù)。

import java.util.*;

public class Freq {
    public static void main(String[] args) {
        Map m = new HashMap();

        // Initialize frequency table from command line
        for (String a : args) {
            Integer freq = m.get(a);
            m.put(a, (freq == null) ? 1 : freq + 1);
        }

        System.out.println(m.size() + " distinct words:");
        System.out.println(m);
    }
}

關(guān)于這個(gè)程序唯一棘手的問題是put語(yǔ)句的第二個(gè)參數(shù),該參數(shù)是一個(gè)條件表達(dá)式,如果單詞之前從未出現(xiàn)過,則其頻率設(shè)置為1,如果單詞已經(jīng)出現(xiàn),則其頻率設(shè)置為當(dāng)前值加1,嘗試使用以下命令運(yùn)行此程序:

java Freq if it is to be it is up to me to delegate

該程序產(chǎn)生以下輸出。

8 distinct words:
{to=3, delegate=1, be=1, it=2, up=1, if=1, me=1, is=2}

假設(shè)你希望按字母順序查看頻率表,你所要做的就是將Map的實(shí)現(xiàn)類型從HashMap更改為TreeMap,進(jìn)行這種更改會(huì)導(dǎo)致程序從同一命令行生成以下輸出。

8 distinct words:
{be=1, delegate=1, if=1, is=2, it=2, me=1, to=3, up=1}

類似地,你可以通過將map的實(shí)現(xiàn)類型更改為LinkedHashMap,使程序按照單詞首次出現(xiàn)在命令行上的順序打印頻率表,這樣做會(huì)產(chǎn)生以下輸出。

8 distinct words:
{if=1, it=2, is=2, to=3, be=1, up=1, me=1, delegate=1}

這種靈活性提供了基于接口的框架功能的有力說明。

與Set和List接口一樣,Map強(qiáng)化了對(duì)equalshashCode方法的要求,因此可以比較兩個(gè)Map對(duì)象的邏輯相等性,而不考慮它們的實(shí)現(xiàn)類型,如果兩個(gè)Map實(shí)例表示相同的鍵值映射,則它們是相等的。

按照慣例,所有通用Map實(shí)現(xiàn)都提供構(gòu)造函數(shù),這些構(gòu)造函數(shù)接受Map對(duì)象并初始化新Map以包含指定Map中的所有鍵值映射。這個(gè)標(biāo)準(zhǔn)的Map轉(zhuǎn)換構(gòu)造函數(shù)完全類似于標(biāo)準(zhǔn)的Collection構(gòu)造函數(shù):它允許調(diào)用者創(chuàng)建一個(gè)所需實(shí)現(xiàn)類型的Map,該Map最初包含另一個(gè)Map中的所有映射,而不管其他Map的實(shí)現(xiàn)類型如何。例如,假設(shè)你有一個(gè)名為mMap,以下單行創(chuàng)建一個(gè)新的HashMap,最初包含與m相同的所有鍵值映射。

Map copy = new HashMap(m);
Map接口批量操作

clear的操作完全符合你的想法:它從Map中刪除所有映射。putAll操作是Collection接口的addAll操作的Map模擬,除了明顯使用將一個(gè)Map轉(zhuǎn)儲(chǔ)到另一個(gè)Map之外,它還有第二個(gè)更微妙的用途,假設(shè)Map用于表示屬性—值對(duì)的集合,putAll操作與Map轉(zhuǎn)換構(gòu)造函數(shù)結(jié)合使用,提供了一種使用默認(rèn)值實(shí)現(xiàn)屬性映射創(chuàng)建的簡(jiǎn)潔方法。以下是演示此技術(shù)的靜態(tài)工廠方法。

static  Map newAttributeMap(Mapdefaults, Map overrides) {
    Map result = new HashMap(defaults);
    result.putAll(overrides);
    return result;
}
集合視圖

Collection視圖方法允許以這三種方式將Map視為Collection

keySetMap中包含鍵的Set。

valuesMap中包含值的Collection,此Collection不是Set,因?yàn)槎鄠€(gè)鍵可以映射到相同的值。

entrySetMap中包含的鍵值對(duì)的Set,Map接口提供了一個(gè)名為Map.Entry的小型嵌套接口,該接口是此Set中元素的類型。

Collection視圖提供迭代Map的唯一方法,此示例說明了使用for-each構(gòu)造迭代Map中的鍵的標(biāo)準(zhǔn)語(yǔ)法:

for (KeyType key : m.keySet())
    System.out.println(key);

使用迭代器:

// Filter a map based on some 
// property of its keys.
for (Iterator it = m.keySet().iterator(); it.hasNext(); )
    if (it.next().isBogus())
        it.remove();

迭代值的語(yǔ)法是類似的,以下是迭代鍵值對(duì)的語(yǔ)法。

for (Map.Entry e : m.entrySet())
    System.out.println(e.getKey() + ": " + e.getValue());

起初,許多人擔(dān)心這些語(yǔ)法可能會(huì)很慢,因?yàn)槊看握{(diào)用Collection視圖操作時(shí)Map都必須創(chuàng)建一個(gè)新的Collection實(shí)例,放松:每次要求給定的Collection視圖時(shí),Map都沒有理由不能總是返回相同的對(duì)象,這正是java.util中所有Map實(shí)現(xiàn)的功能。

對(duì)于所有這三個(gè)Collection視圖,調(diào)用Iteratorremove操作將從支持Map中刪除相關(guān)條目,假設(shè)支持Map一開始就支持元素刪除,這由前面的過濾語(yǔ)法說明。

使用entrySet視圖,還可以通過在迭代期間調(diào)用Map.EntrysetValue方法來更改與鍵關(guān)聯(lián)的值(同樣,假設(shè)Map一開始就支持值修改)。請(qǐng)注意,這些是在迭代期間修改Map的唯一安全方法,如果在迭代進(jìn)行過程中以任何其他方式修改基礎(chǔ)Map,則行為是未指定的。

Collection視圖支持以多種形式刪除元素 — removeremoveAll、retainAllclear操作,以及Iterator.remove操作(同樣,這假設(shè)支持Map支持元素刪除)。

Collection視圖在任何情況下都不支持元素添加,對(duì)于keySetvalues視圖沒有任何意義,并且對(duì)于entrySet視圖沒有必要,因?yàn)橹С?b>Map的putputAll方法提供相同的功能。

Collection視圖的花哨用途:Map代數(shù)

應(yīng)用于Collection視圖時(shí),批量操作(containsAll、removeAllretainAll)是令人驚訝的強(qiáng)大工具。對(duì)于初學(xué)者,假設(shè)你想知道一個(gè)Map是否是另一個(gè)Map的子圖 — 也就是說,第一個(gè)Map是否包含第二個(gè)Map中的所有鍵值映射,以下語(yǔ)法可以解決這個(gè)問題。

if (m1.entrySet().containsAll(m2.entrySet())) {
    ...
}

沿著類似的路線,假設(shè)你想知道兩個(gè)Map對(duì)象是否包含所有相同鍵的映射。

if (m1.keySet().equals(m2.keySet())) {
    ...
}

假設(shè)你有一個(gè)表示屬性—值對(duì)集合的Map,以及兩個(gè)表示所需屬性和允許屬性的Set(允許的屬性包括必需的屬性),以下代碼段確定屬性映射是否符合這些約束,如果不符合則打印詳細(xì)的錯(cuò)誤消息。

static  boolean validate(Map attrMap, Set requiredAttrs, SetpermittedAttrs) {
    boolean valid = true;
    Set attrs = attrMap.keySet();

    if (! attrs.containsAll(requiredAttrs)) {
        Set missing = new HashSet(requiredAttrs);
        missing.removeAll(attrs);
        System.out.println("Missing attributes: " + missing);
        valid = false;
    }
    if (! permittedAttrs.containsAll(attrs)) {
        Set illegal = new HashSet(attrs);
        illegal.removeAll(permittedAttrs);
        System.out.println("Illegal attributes: " + illegal);
        valid = false;
    }
    return valid;
}

假設(shè)你想知道兩個(gè)Map對(duì)象共有的所有鍵。

SetcommonKeys = new HashSet(m1.keySet());
commonKeys.retainAll(m2.keySet());

類似的語(yǔ)法可以為你提供共同的值。

到目前為止提出的所有語(yǔ)法都是非破壞性的,也就是說,它們不會(huì)修改支持Map,這里有一些,假設(shè)你要?jiǎng)h除一個(gè)Map與另一個(gè)Map共有的所有鍵值對(duì)。

m1.entrySet().removeAll(m2.entrySet());

假設(shè)你要從一個(gè)Map中刪除在另一個(gè)Map中具有映射的所有鍵。

m1.keySet().removeAll(m2.keySet());

在同一個(gè)批量操作中開始混合鍵和值時(shí)會(huì)發(fā)生什么?假設(shè)你有一個(gè)Map,managers,將公司中的每個(gè)員工映射到員工的經(jīng)理,我們會(huì)故意模糊鍵和值對(duì)象的類型,沒關(guān)系,只要它們是相同的,現(xiàn)在假設(shè)你想知道所有“個(gè)人貢獻(xiàn)者”(或非管理者)是誰(shuí),以下代碼段將準(zhǔn)確告訴你你想要了解的內(nèi)容。

Set individualContributors = new HashSet(managers.keySet());
individualContributors.removeAll(managers.values());

假設(shè)你要解雇所有直接向某位經(jīng)理Simon報(bào)告的員工。

Employee simon = ... ;
managers.values().removeAll(Collections.singleton(simon));

請(qǐng)注意,這個(gè)語(yǔ)法是使用Collections.singleton,這是一個(gè)靜態(tài)工廠方法,它返回一個(gè)帶有指定元素的不可變Set。

一旦你完成了這項(xiàng)工作,你可能會(huì)有一群?jiǎn)T工,他們的經(jīng)理不再為公司工作(如果任何Simon的直接報(bào)告本身就是經(jīng)理),以下代碼將告訴你哪些員工擁有不再為公司工作的經(jīng)理。

Map m = new HashMap(managers);
m.values().removeAll(managers.keySet());
Set slackers = m.keySet();

這個(gè)例子有點(diǎn)棘手,首先,它創(chuàng)建Map的臨時(shí)副本,并從臨時(shí)副本中刪除其(manager)值是原始Map中的鍵的所有條目,請(qǐng)記住,原始Map為每個(gè)員工都有一個(gè)條目。因此,臨時(shí)Map中的其余條目包括來自原始Map的其(經(jīng)理)值不再是雇員的所有條目,因此,臨時(shí)副本中的鍵恰好代表了我們正在尋找的員工。

多重映射

多重映射就像Map,但它可以將每個(gè)鍵映射到多個(gè)值,Java集合框架不包含多重映射的接口,因?yàn)樗鼈儾⒉怀S?。使?b>Map值為List實(shí)例作為多重映射的Map是一件相當(dāng)簡(jiǎn)單的事情。下一個(gè)代碼示例演示了此技術(shù),該示例讀取每行包含一個(gè)單詞(全部小寫)的單詞列表,并打印出符合大小標(biāo)準(zhǔn)的所有變位詞組。變位詞組是一堆單詞,所有單詞都包含完全相同的字母,但順序不同,該程序在命令行上有兩個(gè)參數(shù):(1)字典文件的名稱,(2)要打印出的變位詞組的最小尺寸,不打印包含少于指定最小值的單詞組的變位詞組。

找到變位詞組有一個(gè)標(biāo)準(zhǔn)技巧:對(duì)于字典中的每個(gè)單詞,按字母順序排列單詞中的字母(即,將單詞的字母重新排序?yàn)樽帜疙樞颍┎l目放入多重映射,將字母順序排列的單詞映射到原始單詞。例如,單詞bad導(dǎo)致將abd條目映射為bad以將其放入多重映射中,稍作思考就會(huì)發(fā)現(xiàn),任何給定鍵映射到的所有單詞都構(gòu)成一個(gè)變位詞組。迭代多重映射中的鍵,打印出符合大小約束的每個(gè)變位詞組是一件簡(jiǎn)單的事情。

以下程序是該技術(shù)的直接實(shí)現(xiàn)。

import java.util.*;
import java.io.*;

public class Anagrams {
    public static void main(String[] args) {
        int minGroupSize = Integer.parseInt(args[1]);

        // Read words from file and put into a simulated multimap
        Map> m = new HashMap>();

        try {
            Scanner s = new Scanner(new File(args[0]));
            while (s.hasNext()) {
                String word = s.next();
                String alpha = alphabetize(word);
                List l = m.get(alpha);
                if (l == null)
                    m.put(alpha, l=new ArrayList());
                l.add(word);
            }
        } catch (IOException e) {
            System.err.println(e);
            System.exit(1);
        }

        // Print all permutation groups above size threshold
        for (List l : m.values())
            if (l.size() >= minGroupSize)
                System.out.println(l.size() + ": " + l);
    }

    private static String alphabetize(String s) {
        char[] a = s.toCharArray();
        Arrays.sort(a);
        return new String(a);
    }
}

在173,000字的字典文件上運(yùn)行此程序,最小變位詞組大小為8會(huì)產(chǎn)生以下輸出。

9: [estrin, inerts, insert, inters, niters, nitres, sinter,
     triens, trines]
8: [lapse, leaps, pales, peals, pleas, salep, sepal, spale]
8: [aspers, parses, passer, prases, repass, spares, sparse,
     spears]
10: [least, setal, slate, stale, steal, stela, taels, tales,
      teals, tesla]
8: [enters, nester, renest, rentes, resent, tenser, ternes,
     treens]
8: [arles, earls, lares, laser, lears, rales, reals, seral]
8: [earings, erasing, gainers, reagins, regains, reginas,
     searing, seringa]
8: [peris, piers, pries, prise, ripes, speir, spier, spire]
12: [apers, apres, asper, pares, parse, pears, prase, presa,
      rapes, reaps, spare, spear]
11: [alerts, alters, artels, estral, laster, ratels, salter,
      slater, staler, stelar, talers]
9: [capers, crapes, escarp, pacers, parsec, recaps, scrape,
     secpar, spacer]
9: [palest, palets, pastel, petals, plates, pleats, septal,
     staple, tepals]
9: [anestri, antsier, nastier, ratines, retains, retinas,
     retsina, stainer, stearin]
8: [ates, east, eats, etas, sate, seat, seta, teas]
8: [carets, cartes, caster, caters, crates, reacts, recast,
     traces]

許多這些詞似乎有點(diǎn)虛偽,但這不是程序的錯(cuò);它們?cè)谧值湮募?,這是使用的字典文件,它源自Public Domain ENABLE基準(zhǔn)參考詞列表。

上一篇:Deque接口 下一篇:對(duì)象排序

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/73946.html

相關(guān)文章

  • Java? 教程(集合接口

    集合接口 核心集合接口封裝了不同類型的集合,如下圖所示,這些接口允許獨(dú)立于其表示的細(xì)節(jié)來操縱集合,核心集合接口是Java集合框架的基礎(chǔ),如下圖所示,核心集合接口形成層次結(jié)構(gòu)。 showImg(https://segmentfault.com/img/bVbntJW?w=402&h=146); Set是一種特殊的Collection,SortedSet是一種特殊的Set,依此類推,另請(qǐng)注意,層次結(jié)構(gòu)...

    elisa.yang 評(píng)論0 收藏0
  • Java 8 簡(jiǎn)明教程

    摘要:簡(jiǎn)明教程原文譯者黃小非來源簡(jiǎn)明教程并沒有沒落,人們很快就會(huì)發(fā)現(xiàn)這一點(diǎn)歡迎閱讀我編寫的介紹。編譯器會(huì)自動(dòng)地選擇合適的構(gòu)造函數(shù)來匹配函數(shù)的簽名,并選擇正確的構(gòu)造函數(shù)形式。 Java 8 簡(jiǎn)明教程 原文:Java 8 Tutorial 譯者:ImportNew.com - 黃小非 來源:Java 8簡(jiǎn)明教程 ? Java并沒有沒落,人們很快就會(huì)發(fā)現(xiàn)這一點(diǎn) 歡迎閱讀我編寫的Java ...

    testHs 評(píng)論0 收藏0
  • Java 8 并發(fā)教程:原子變量和 ConcurrentMa

    摘要:并發(fā)教程原子變量和原文譯者飛龍協(xié)議歡迎閱讀我的多線程編程系列教程的第三部分。如果你能夠在多線程中同時(shí)且安全地執(zhí)行某個(gè)操作,而不需要關(guān)鍵字或上一章中的鎖,那么這個(gè)操作就是原子的。當(dāng)多線程的更新比讀取更頻繁時(shí),這個(gè)類通常比原子數(shù)值類性能更好。 Java 8 并發(fā)教程:原子變量和 ConcurrentMap 原文:Java 8 Concurrency Tutorial: Synchroni...

    bitkylin 評(píng)論0 收藏0
  • Java? 教程(SortedMap接口

    SortedMap接口 SortedMap是一個(gè)按升序維護(hù)其條目的Map,根據(jù)鍵的自然順序或在創(chuàng)建SortedMap時(shí)提供的Comparator進(jìn)行排序,SortedMap接口提供常規(guī)Map操作和以下操作的操作: 范圍視圖 — 對(duì)排序后的map執(zhí)行任意范圍操作 端點(diǎn) — 返回已排序map中的第一個(gè)或最后一個(gè)鍵 比較器訪問 — 返回用于排序map的Comparator(如果有的話) 下面的接口是...

    JessYanCoding 評(píng)論0 收藏0
  • Java? 教程(抽象方法和類)

    抽象方法和類 抽象類是一個(gè)聲明為abstract的類 — 它可能包括也可能不包括抽象方法,抽象類無法實(shí)例化,但可以進(jìn)行子類化。 抽象方法是在沒有實(shí)現(xiàn)的情況下聲明的方法(沒有大括號(hào),后跟分號(hào)),如下所示: abstract void moveTo(double deltaX, double deltaY); 如果一個(gè)類包含抽象方法,那么該類本身必須被聲明為abstract,如: public abs...

    Amio 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<