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编程思想》
Java 同时返回多个不同类型的方法的更多相关文章
- Java连载76-基础数据类型包装类型及其方法简介
一.java中八种基本数据类型对应的包装类型 基本数据类型 包装类型 byte java.lang.Byte short ...
- java将long数据转为int类型的方法
二.调用intValue()方法 [java] long ll = 300000; int ii= new Long(ll).intValue(); 三.先把long转换成字符串String,然后在转 ...
- java反射三种获得类类型的方法
public class Test { public static void main(String[] args) { Test t=new Test();//所有的类都是Class类的实例(类类型 ...
- Java中返回参数值的几种状态
Java 中无参无返回值方法的使用 第一步,定义方法 例如:下面代码定义了一个方法名为 show ,没有参数,且没有返回值的方法,执行的操作为输出 " welcome to imooc. & ...
- Java协变返回类型
今天看到句话:“支持重写方法时返回协变类型”. 那么什么事协变类型?在网上找了找资料,大体上明白了. Java 5.0添加了对协变返回类型的支持,即子类覆盖(即重写)基类方法时,返回的类型可以是基类方 ...
- “全栈2019”Java第五十八章:多态中方法返回类型可以是子类类型
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- Java中返回值定义为int类型的 方法return 1返回的是int还是Integer&&finally中return问题
在Java中返回值定义为int类型的 方法return 1:中返回的是Integer值,在返回的时候基本类型值1被封装为Integer类型. 定义一个Test类,在异常处理try中和finally中分 ...
- Java常见面试题02-方法重写和方法重载的区别?方法重载能改变返回值类型吗?
方法重写和方法重载的区别?方法重载能改变返回值类型吗? A:Override方法重写和Overload方法重载的区别? Overload是否可以改变返回值类型?可以 方法重写 • 子类中 ...
- java统一返回标准类型
一.前言.背景 在如今前后端分离的时代,后端已经由传统的返回view视图转变为返回json数据,此json数据可能包括返回状态.数据.信息等......因为程序猿的习惯不同所以返回json数据的格式也 ...
随机推荐
- hashCode方法的作用?
(1)前言,想要明白hashCode的作用,你必须要先知道Java中的集合. Java中的集合(Collection)有两类,一类是List,再有一类是Set. 前者集合内的元素是有序的,元素可以重复 ...
- js 停止事件冒泡 阻止浏览器的默认行为(比如阻止超连接 # )
在前端开发工作中,由于浏览器兼容性等问题,我们会经常用到“停止事件冒泡”和“阻止浏览器默认行为”. 1..停止事件冒泡 JavaScript代码 //如果提供了事件对象,则这是一个非IE浏览器if ...
- HDU.3571.N-dimensional Sphere(高斯消元 模线性方程组)
题目链接 高斯消元详解 /* $Description$ 在n维空间中给定n+1个点,求一个点使得这个点到所有点的距离都为R(R不给出).点的任一坐标|xi|<=1e17. $Solution$ ...
- swoole深入学习 4. process
swoole-1.7.2增加了一个进程管理模块,用来替代PHP的pcntl扩展.pcntl是php新增的一个多进程扩展,用来实现多进程,但是有很多不完善的地方,swoole 就完善了这些地方,而且使得 ...
- memcached 数据库
安装 [root@node2 ~]# yum install memcached 启动 [root@node2 ~]# systemctl start memcached 连接 [root@node2 ...
- Docker 停止容器
查看运行的容器 [root@node3 ~]# docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES f070c6924656 ...
- QThreadPool&QRunnable&类函数的使用
QThreadPool+QRunnable线程池与QThread线程两种方式使用的场景不同,QThreadPool+QRunnable线程池主要用于那种不需要一直运行的任务,而QThread主要用于长 ...
- uva 10816 Travel in Desert(简单的好题~两种方法)
题意: 给出 一个图 点与点之间的路径上有两个权值 路径长度和温度 要求在所走路径中的温度的最大值最小的前提下 走最短路径 解题思路1: 首先用 最小生成树 的方法走出 最小瓶颈路 .把在这期间用到的 ...
- boost.property_tree解析xml的帮助类以及中文解析问题的解决(转)
boost.property_tree可以用来解析xml和json文件,我主要用它来解析xml文件,它内部封装了号称最快的xml解析器rapid_xml,其解析效率还是很好的.但是在使用过程中却发现各 ...
- SQL Server 2012安装step by step
安装光盘介质问题,报错,换盘 Overall summary: Final result: Passed Exit code (Decimal): ...