亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

為什么我不能在需要 List<X> 的地方使用實現 List<X> 的類?

為什么我不能在需要 List<X> 的地方使用實現 List<X> 的類?

神不在的星期二 2023-12-13 14:33:29
首先是簡單的情況(A):public class PsList implements List<Ps> { ... }別處private void doSomething(List<Ps> list) { ... }// compilesList<Ps> arrayList = new ArrayList<Ps>();doSomething(arrayList);// does not compilePsList psList = new PsList();doSomething(psList);好的。我知道我可以通過添加 ? 將其更改為“工作” 延伸如下:private void doSomething(? extends List<Ps> list) { ... }// compilesList<Ps> arrayList = new ArrayList<Ps>();doSomething(arrayList);// compilesPsList psList = new PsList();doSomething(psList);我的問題是為什么我需要這樣做?對我來說完全是無稽之談。我正在實現所期望的確切界面。我可以傳遞除 ArrayList 之外的其他列表類型,為什么我的不行?生活總是比這更復雜,所以我真正的編碼問題是(B):public class PsList implements List<Ps> { ... }private void doSomething(Map<String, ? extends List<Ps>> map, Boolean custom) {...// need to create a new List<Ps> of either an ArrayList<Ps> or PsListmap.put("stringValue", custom ? new PsList() : new ArrayList<Ps>());...}那么,無論哪種情況,Java都會抱怨map正在等待?擴展 List 作為值。即使我將其更改為:List<Ps> list = new ArrayList<>();List<Ps> psList = new PsList();map.put("string", custom ? psList : list);當然這不能編譯:? extends List<Ps> list = new ArrayList<>();? extends List<Ps> psList = new PsList();map.put("string", custom ? psList : list);那么我應該怎么做才能讓這樣的事情發揮作用呢?編輯1:好的,最小的復制品:Ps.javapackage com.foo;public class Ps{}第一個參數類型錯誤。找到:'java.lang.String,com.foo.PsList>',必需:'java.util.Map>'
查看完整描述

2 回答

?
慕標琳琳

TA貢獻1830條經驗 獲得超9個贊

正如您在問題中途似乎意識到的那樣,您的問題不在于List<Ps>PsList.

問題是你不能添加到Map<String, ? extends List<Ps>>.

讓我們考慮一個更簡單的例子:

void doSomething(Map<String, ? extends Number> map) {
    map.put(String, Integer.valueOf(0));    // Not allowed.}

問題在于,這Map<String, ? extends Number>并不意味著“值可以是 Number 或 Number 的任何子類”。

每個通用類型對象都有一個特定的非通配符類型。意思是,不存在類型為 的 Map Map<String, ? extends Number>。然而,以下情況可能存在:

  • Map<String, Integer>(僅允許整數值)

  • Map<String, Double>(僅允許 Double 值)

  • Map<String, Number>(允許任何 Number 子類的值)

Map<String, ? extends Number>指的是可能是上述任何一個的Map (或者,當然,任何其他特定的 Number 子類)。編譯器不知道 Map 的值是什么特定類型,但 Map 的值仍然具有特定類型,但該類型不會以任何方式使用?。

因此,再次查看示例方法:

void doSomething(Map<String, ? extends Number> map) {

    // Not allowed.  The caller might have passed a Map<String, Double>.

    map.put(String, Integer.valueOf(0));


    // Not allowed.  The caller might have passed a Map<String, Integer>.

    map.put(String, Double.valueOf(0));


    // Not allowed.  The caller might have passed a Map<String, Integer>

    // or Map<String, Double>.  This method has no way of knowing what the

    // actual restriction is.

    Number someNumber = generateNewNumber();

    map.put(String, someNumber);

}

事實上,您無法向類型為上限通配符的 Map 或 Collection 添加任何內容,因為無法知道這樣做是否正確且安全。


對于您的情況,最簡單的解決方案是刪除通配符:


private void doSomething(Map<String, List<Ps>> map, boolean custom) {

    // ...

    map.put("stringValue", custom ? new PsList() : new ArrayList<Ps>());

}

如果您確實有具有不同值類型的映射,則需要告訴該方法正在使用的特定類型:


private <L extends List<Ps>> void doSomething(Map<String, L> map,

                                              Supplier<L> listCreator) {

    // ...


    map.put("stringValue", listCreator.get());

}

然后你可以像這樣調用該方法:


if (custom) {

    doSomething(psMap, PsList::new);

} else {

    doSomething(listMap, ArrayList::new);

}


查看完整回答
反對 回復 2023-12-13
?
ABOUTYOU

TA貢獻1812條經驗 獲得超5個贊

你絕對可以做到。這是一個例子:


public class MyClass {


    static interface FooList<T> {}

    static class Ps {}


    static void doSomething(FooList<Ps> list) { }


    static class PsList implements FooList<Ps> { }


    public static void main(String[] args)

    {

        FooList psList = new PsList();


        doSomething(psList);


        System.out.println("HelloWorld!");

    }

}

請參閱此處的演示。



查看完整回答
反對 回復 2023-12-13
  • 2 回答
  • 0 關注
  • 185 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號