Java 如何同時返回多個不同類型
Java 同時返回多個不同類型
前言
雖然對於這種需求不常用,且比較冷門,但是還是有其存在的價值,再次做一下整理。我們常用的return語句隻允許返回單個對象,相對的解決辦法就是創建一個對象,用它來持有想要返回的多個對象。
實現這種功能,還要歸功於Java1.5的新特性-泛型,我們利用泛型,可以一次性地解決該問題,以後再也不用在這個問題上浪費時間瞭,並且,我們可以再編譯期就能夠確保類型安全。
你也許已經想到使用集合可以實現我們的需求,但是雖然一次可以返回多個值,但是其類型都是相同的,並不完全符合我們的需求。
我們需要引入一個概念:元組(tuple),元組也稱為數據傳送對象或信使。元組是將一組對象直接打包存儲於其中的一個單一對象,這個容器對象允許讀取其中元素,但是不允許向其中存放新的對象。
通常,元組可以具有任意長度,同時元組中的對象可以是任意不同的類型。我們能夠為每一個對象指明其類型,並且可以正確讀取到數據,這就是元組可以提供的功能。我們要處理不同長度的問題,需要創建多個不同的元組。
首先我們創建一個2維元組
//: net/mindview/util/TwoTuple.java(Java編程思想_代碼_目錄) package net.mindview.util; public class TwoTuple<A, B> { public final A first; public final B second; public TwoTuple(A a, B b) { first = a; second = b; } public String toString() { return "(" + first + ", " + second + ")"; } }
構造器捕獲瞭要存儲的對象,而toString()方法是一個便利函數,用來顯示列表中的值。
註意:元組隱含地保持瞭其中元素的次序。
閱讀上面的代碼,以及根據元組的定義,你一定會感到詫異,設計思路不是應該將first和second聲明為private,然後生成這兩個變量的get方法嗎?
以上是我們大多數人的思維,但是我們仔細分析上面的代碼,可以發現完全符合我們的要求。首先我們可以讀取first和second,並且因為使用瞭final聲明,我們就無法在修改其值。我們對比這兩種寫法,很顯然,上面給出的代碼更加合理,更加簡潔明瞭。
還有另一種設計考慮,即你確實希望允許客戶端程序員改變first或second所引用的對象。然而,采用上面的形式無疑是更安全的做法,這樣的話,如果程序員想要使用具有不同元素的元組,就強制要求他們另外創建一個新的TwoTuple對象。
我們可以利用繼承機制實現長度更長的元組
//: net/mindview/util/ThreeTuple.java package net.mindview.util; public class ThreeTuple<A,B,C> extends TwoTuple<A,B> { public final C third; public ThreeTuple(A a, B b, C c) { super(a, b); third = c; } public String toString() { return "(" + first + ", " + second + ", " + third +")"; } }
//: net/mindview/util/FourTuple.java package net.mindview.util; public class FourTuple<A,B,C,D> extends ThreeTuple<A,B,C> { public final D fourth; public FourTuple(A a, B b, C c, D d) { super(a, b, c); fourth = d; } public String toString() { return "(" + first + ", " + second + ", " + third + ", " + fourth + ")"; } }
//: net/mindview/util/FiveTuple.java package net.mindview.util; public class FiveTuple<A,B,C,D,E> extends FourTuple<A,B,C,D> { public final E fifth; public FiveTuple(A a, B b, C c, D d, E e) { super(a, b, c, d); fifth = e; } public String toString() { return "(" + first + ", " + second + ", " + third + ", " + fourth + ", " + fifth + ")"; } }
為瞭使用元組,你隻需定義一個長度適合的元組,將其作為方法的返回值,然後在return語句中創建該元組,並返回即可。
實例
//: generics/TupleTest.java import net.mindview.util.*; class Amphibian { } class Vehicle { } public class TupleTest { static TwoTuple<String, Integer> f() { // Autoboxing converts the int to Integer: return new TwoTuple<String, Integer>("hi", 47); } static ThreeTuple<Amphibian, String, Integer> g() { return new ThreeTuple<Amphibian, String, Integer>(new Amphibian(), "hi", 47); } static FourTuple<Vehicle, Amphibian, String, Integer> h() { return new FourTuple<Vehicle, Amphibian, String, Integer>(new Vehicle(), new Amphibian(), "hi", 47); } static FiveTuple<Vehicle, Amphibian, String, Integer, Double> k() { return new FiveTuple<Vehicle, Amphibian, String, Integer, Double>( new Vehicle(), new Amphibian(), "hi", 47, 11.1); } public static void main(String[] args) { TwoTuple<String, Integer> ttsi = f(); System.out.println(ttsi); // ttsi.first = "there"; // Compile error: final System.out.println(g()); System.out.println(h()); System.out.println(k()); } }
輸出結果:
(hi, 47)
(Amphibian@15db9742, hi, 47)
(Vehicle@6d06d69c, Amphibian@7852e922, hi, 47)
(Vehicle@4e25154f, Amphibian@70dea4e, hi, 47, 11.1)
由於有瞭泛型,你可以很容易地創建元組,令其返回一組任意類型的對象。而你所要做的,隻是編寫表達式而已。
通過ttsi.first = “there”;語句的錯誤,我們可以看出,final聲明確實能夠保護public元素,在對象被構造出來之後,聲明為final的元素便不能被再賦予其他值瞭。
java return返回多個值或多種類型的值方法(list;Map)
1、在寫方法的時候
有時候需要返回多個值,有時候返回的多個值的類型還不同,依據不同情況以下提供瞭三種方法返回多個值。
2、具體思路都在代碼註釋裡瞭
如果返回的多個值類型相同,可以用方法一和方法二;如果返回的多個值類型不同,可以用方法三。
import java.util.*; public class Demo { //方法1:返回list public static List<int[]> returnList(){ List<int[]> list=new ArrayList<>(); list.add(new int[]{1}); list.add(new int[]{1, 2}); list.add(new int[]{1, 2, 3}); return list; } //方法2:返回Map,一個Map隻能有一種數據類型 public static Map<String, Integer> returnMap(){ Map<String,Integer> map = new HashMap<>(); map.put("age",1); //”age“是key,類似於索引,1是索引對應的int值 map.put("high",30); //System.out.println(map.get("age")); Map<String,int[]> map1 = new HashMap<>(); map1.put("array", new int[]{1, 2, 3}); //System.out.println(Arrays.toString( map1.get("array") )); return map; } //方法3:一次性返回兩種類型的數據,結合瞭Map和list public static List<Map> returnMapList(){ Map<String,Integer> map = new HashMap<>(); map.put("age",1); map.put("high",30); Map<String,int[]> map1 = new HashMap<>(); map1.put("array", new int[]{1, 2}); List<Map> listMap = new ArrayList<Map>(); listMap.add(map); listMap.add(map1); return listMap; } //測試代碼 public static void main(String[] args){ //返回list List<int[]> list1 = returnList(); System.out.println(Arrays.toString(list1.get(0)) + Arrays.toString(list1.get(1)) + Arrays.toString(list1.get(2)) + "\nreturnlist結束\n"); //返回Map,一個Map隻能有一種數據類型 Map<String,Integer> mapTest = returnMap(); System.out.println("age = " + mapTest.get("age") +", high = " + mapTest.get("high") + "\nreturnMap結束\n"); //一次性返回兩種類型的數據,結合瞭Map和list List<Map> list2 = returnMapList(); System.out.println(list2.get(0) +" , " + list2.get(1) + "\nreturnMapList結束");//list2.get(1)是數組的地址 System.out.print("調用返回的int和int[]:"); Map map0 = list2.get(0); Map map1 = list2.get(1); System.out.println( "age = " + map0.get("age") ); System.out.println("array = " + Arrays.toString((int[]) map1.get("array"))); // System.out.println(Arrays.toString((int[]) list2.get(1).get("array"))); //調用過程也可以這樣寫 } }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- Java中List.of()和Arrays.asList()的區別及原因分析
- java Array和Arrays的區別總結
- Java 基礎–Arrays工具類詳解
- 聊聊Arrays.deepToString()和Arrays.toString()的區別
- Java Collections.shuffle()方法案例詳解