0 0

JAVA 泛型5

public class GenericMethods {
	public <T> void f(T... x) {
		System.out.println(x.getClass().getName());
	}

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		GenericMethods gm = new GenericMethods();
		gm.f("123", "234");
		gm.f(123, 2132);
		gm.f(gm, gm);
		gm.f(23, "1232", gm);
		gm.f("123", 12);
	}
}


运行的结果如下:
[Ljava.lang.String;
[Ljava.lang.Integer;
[Lcom.test.generic.Method.GenericMethods;
[Ljava.lang.Object;
[Ljava.io.Serializable;//此行为什么不是Object?


为啥最后一行不是Object,变成了Serializable??

问题补充:看倒数第二行代码
gm.f(23, "1232", gm);  //autoboxing,对应(Integer,String,GenericMethods),这时返回的是[Ljava.lang.Object; 因为所有类都extends Object
再看倒数第一行
gm.f("123", 12);  //autoboxing,对应(String,Integer),他们都extends Object,同时都实现了Serializable,Comparable接口,打印的是[Ljava.lang.Serializable,为何不是[Ljava.lang.Object,或[Ljava.lang.Comparable

又再运行以下语句
gm.f(12, "123"); //打印是[Ljava.lang.Comparable;
gm.f("123", 12); //打印是[Ljava.io.Serializable;
2012年6月26日 11:59

8个答案 按时间排序 按投票排序

1 0

注意:void f(T... x) 的含义是 后面的参数必须是统一的一个类型,比如int 必须全int String 必须全string,你的类型不统一,虚拟机就只能更具自己的判断来处理。说不定你用jdk7就会是object

2012年6月27日 08:28
1 0

java.lang.Number (implements java.io.Serializable)
         java.lang.Byte (implements java.lang.Comparable<T>)
         java.lang.Double (implements java.lang.Comparable<T>)
         java.lang.Float (implements java.lang.Comparable<T>)
         java.lang.Integer (implements java.lang.Comparable<T>)
         java.lang.Long (implements java.lang.Comparable<T>)
         java.lang.Short (implements java.lang.Comparable<T>)
java.lang.String (java.io.Serializable, implements java.lang.CharSequence, java.lang.Comparable<T>)

我猜应该和实现的接口的顺序有关系吧。。。

估计是按照顺序匹配的(我猜的),如果是integer在前,先把integer和string比(不匹配再依次把string当Serializable,CharSequence、Comparable),再把integer当做Comparable和string(依次xxx xxx xxx) 比较,如果类型相同,就当做是Comparable传进去当参数

如果是string在前,估计先和string和integer(依次Comparable,Number,Serializable)比,再把string当Serializable和integer比。。。

总归是这么比较出来的,所以应该是和比较的顺序有关吧。。。。

2012年6月26日 16:21
1 0

对于编译器来说,数字 会被打包为Integer类型
下面是 String  Integer Number 三个对象
Integer extends Number

对于最后一行来看,gm.f("123", 12)

编译器 看做 gm.f(String ,Integer); 再变为 T[] 首先拿到的是String的 Serializable 接口,在去找Integer有没有Serializable接口,如果有那么就用Serializable

其实我们把 顺寻反过来看就知道了
gm.f(12,123") 这里输出的就是 Comparable

先去找Integer的接口

public final class String implements java.io.Serializable, Comparable<String>, CharSequence

public final class Integer extends Number implements Comparable<Integer>

public abstract class Number implements java.io.Serializable

2012年6月26日 16:14
0 0

System.out.println(new Serializable[]{"123", 12});
"123" 是String,String实现了Serializable接口
12 会被装箱为Integer,Integer继承于Number类,Number实现了Serializable接口

	public class GenericMethods {
		public <T> void f(T... x) {
			System.out.println(x.getClass().getName());
		}

		/**
		 * @param args
		 */
		public static void main(String[] args) {
			// TODO Auto-generated method stub
			GenericMethods gm = new GenericMethods();
			gm.f("123", "234");
			gm.f(123, 2132);
			gm.f(gm, gm);
			gm.f(23, "1232", gm);
			gm.f("123", 12);
			
			// 楼主看这里,和泛型没有关系!
			System.out.println(new String[]{"123", "234"});
			System.out.println(new Integer[]{123, 2132});
			System.out.println(new GenericMethods[]{gm, gm});
			System.out.println(new Object[]{23, "1232", gm});
			System.out.println(new Serializable[]{"123", 12});
		}
	}


JDK 中 String 的部分代码
public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence
{
    /** The value is used for character storage. */
    private final char value[];


JDK 中 Integer 的部分代码
public final class Integer extends Number implements Comparable<Integer> {
    /**
     * A constant holding the minimum value an <code>int</code> can
     * have, -2<sup>31</sup>.
     */
    public static final int   MIN_VALUE = 0x80000000;

public abstract class Number implements java.io.Serializable {
    /**
     * Returns the value of the specified number as an <code>int</code>.
     * This may involve rounding or truncation.
     *
     * @return  the numeric value represented by this object after conversion
     *          to type <code>int</code>.
     */
    public abstract int intValue();

2012年6月27日 13:22
0 0

这个是跟jdk的编译有关的,其实编译后的字节码已经确定了类型。反编译的代码如下:

	public GenericMethods()
	{
	}

	public transient void f(Object x[])
	{
		System.out.println(((Object) (x)).getClass().getName());
	}

	public static void main(String args[])
	{
		GenericMethods gm = new GenericMethods();
		gm.f(new String[] {
			"123", "234"
		});
		gm.f(new Integer[] {
			Integer.valueOf(123), Integer.valueOf(2132)
		});
		gm.f(new GenericMethods[] {
			gm, gm
		});
		gm.f(new Object[] {
			Integer.valueOf(23), "1232", gm
		});
		gm.f(new Serializable[] {
			"123", Integer.valueOf(12)
		});
		gm.f(new Comparable[] {
			Integer.valueOf(12), "123"
		});
	}

至于为什么是这个样子,我们用下边的代码来看一下:
System.out.println(Integer.class.getInterfaces()[0].getName());
System.out.println(String.class.getInterfaces()[0].getName());

2012年6月27日 13:06
0 0

记得在张老师的java基础提高视频里曾听到过,这个涉及到参数类型推断了,
http://docs.oracle.com/javase/specs/jls/se7/html/jls-15.html#jls-15.12.2.7
怎么根据多个参数类型推断出一个类型,里面有一堆数学公式。
我看我们一般人就别去搞这种数学的东西了吧。
你有兴趣可以研究下上面链接的15.12.2.7. Inferring Type Arguments Based on Actual Arguments

2012年6月26日 16:06
0 0

查了资料,我最后发现,其实不是泛型的问题

Interge和String等对象都实现了Serializable接口,所以这两个传进去,就把T自动匹配成Serializable对象了!!

2012年6月26日 14:28
0 0

你试试这个

public class Test {
	public <T> void f(T... x) {
		for(int i = 0; i < x.length; i++)
			System.out.println(x[i].getClass().getName());
	}

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Test gm = new Test();
		gm.f("123", "234");
		gm.f(123, 2132);
		gm.f(gm, gm);
		gm.f(23, "1232", gm);
		gm.f("123", 12);
	}
}

T...x 表示的是类型为T的数组序列,只有一个类,你参数都是一样的时候当然正确,不一样的时候,把string和int封装成一个类(Serializable)里的成员吧估计


2012年6月26日 12:50

相关推荐

    Java泛型编程指南.pdf

    ### Java泛型编程指南知识点详解 #### 一、绪论:理解Java泛型的重要性与背景 **1.1 泛型的基本概念** 泛型是一种在编程语言中支持编写类型安全的通用函数或类的能力。在Java中引入泛型的主要目的是为了提供更...

    Java泛型的用法及T.class的获取过程解析

    Java泛型的用法及T.class的获取过程解析 Java泛型是Java编程语言中的一种重要特性,它允许开发者在编写代码时指定类型参数,从而提高代码的灵活性和可读性。本文将详细介绍Java泛型的用法 及T.class的获取过程解析...

    Java泛型三篇文章,让你彻底理解泛型(super ,extend等区别)

    Java 泛型详解 Java 泛型是 Java SE 5.0 中引入的一项特征,它允许程序员在编译时检查类型安全,从而减少了 runtime 错误的可能性。泛型的主要优点是可以Reusable Code,让程序员编写更加灵活和可维护的代码。 ...

    Java泛型应用实例

    Java泛型是Java编程语言中的一个强大特性,它允许我们在定义类、接口和方法时指定类型参数,从而实现代码的重用和类型安全。在Java泛型应用实例中,我们可以看到泛型如何帮助我们提高代码的灵活性和效率,减少运行时...

    很好的Java泛型的总结

    Java泛型机制详解 Java泛型是Java语言中的一种机制,用于在编译期检查类型安全。Java泛型的出现解决了Java早期版本中类型安全检查的缺陷。Java泛型的好处是可以在编译期检查类型安全,避免了运行时的...

    java 泛型类的类型识别示例

    综上所述,虽然Java泛型在编译后会进行类型擦除,但通过上述技巧,我们仍然能够在运行时获得关于泛型类实例化类型的一些信息。在实际开发中,这些方法可以帮助我们编写更加灵活和安全的代码。在示例文件`GenericRTTI...

    java泛型技术之发展

    Java泛型是Java编程语言中的一个关键特性,它在2004年随着Java SE 5.0的发布而引入,极大地增强了代码的类型安全性和重用性。本篇文章将深入探讨Java泛型的发展历程、核心概念以及其在实际开发中的应用。 1. **发展...

    SUN公司Java泛型编程文档

    Java泛型是Java编程语言中的一个关键特性,它在2004年随着JDK 5.0的发布被引入。这个特性极大地提高了代码的类型安全性和可读性,减少了在运行时出现ClassCastException的可能性。SUN公司的Java泛型编程文档,包括...

    java 泛型接口示例

    下面我们将详细探讨Java泛型接口的相关知识点。 1. **泛型接口的定义** 泛型接口的定义方式与普通接口类似,只是在接口名之后添加了尖括号`&lt;T&gt;`,其中`T`是一个类型参数,代表某种未知的数据类型。例如: ```java...

    java 泛型方法使用示例

    下面我们将深入探讨Java泛型方法的概念、语法以及使用示例。 **一、泛型方法概念** 泛型方法是一种具有类型参数的方法,这些类型参数可以在方法声明时指定,并在方法体内部使用。与类的泛型类似,它们提供了编译时...

    Java 泛型擦除后的三种补救方法

    Java 泛型是一种强大的工具,它允许我们在编程时指定变量的类型,提供了编译时的类型安全。然而,Java 的泛型在运行时是被擦除的,这意味着在运行时刻,所有的泛型类型信息都会丢失,无法直接用来创建对象或进行类型...

    java泛型的内部原理及更深应用

    Java泛型是Java编程语言中的一个强大特性,它允许在定义类、接口和方法时使用类型参数,从而实现参数化类型。这使得代码更加安全、可读性更强,并且能够减少类型转换的必要。在“java泛型的内部原理及更深应用”这个...

    JAVA泛型加减乘除

    这是一个使用JAVA实现的泛型编程,分为两部分,第一部分创建泛型类,并实例化泛型对象,得出相加结果。 第二部分用户自行输入0--4,选择要进行的加减乘除运算或退出,再输入要进行运算的两个数,并返回运算结果及...

    java泛型总结

    ### Java泛型总结 #### 一、Java泛型概述 Java泛型是在JDK5之后引入的一个特性,它提供了一种类型安全的机制,用于指定集合或其他数据结构中的元素类型。通过使用泛型,程序员可以在编译阶段检测类型错误,避免了...

    java泛型学习ppt

    "Java 泛型学习" Java 泛型是 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类。泛型的主要目标是提高 Java 程序的类型安全。通过知道使用泛型定义的变量的类型限制,编译器可以在一个高得多的...

    Java泛型使用详细分析.pdf

    Java 泛型使用详细分析 Java 泛型是 Java 语言中的一种类型系统特性,允许开发者在编译期检查类型安全,以避免在运行时出现类型相关的错误。在本文中,我们将详细介绍 Java 泛型的使用方法和实现原理。 一、泛型的...

    Java泛型技术之发展.pdf

    Java泛型是Java编程语言中的一个关键特性,它在2004年随着Java SE 5.0的发布而引入,极大地增强了代码的类型安全性和重用性。本篇文章将深入探讨Java泛型的发展历程、核心概念以及其在实际开发中的应用。 1. **发展...

    Java泛型类型擦除后的补偿

    本文将深入探讨Java泛型类型擦除的概念,并介绍在类型擦除后,为了保持泛型的安全性和便利性,Java设计者所采取的一些补偿机制。 1. **类型擦除**: - 在编译期间,所有的泛型类型信息都会被替换为它们的实际类型...

Global site tag (gtag.js) - Google Analytics