赛迪网 > IT技术 Java > 技术动态
  IT资讯搜索
 
IT产品搜索
[程序开发][网管世界][网络安全][数据库技术]
[操作系统][嘉宾聊天·在线访谈][活动集锦]
[精彩专题][Symantec专区][订阅IT技术周刊]
[开发论坛][网管论坛][安全论坛][数据库论坛]
[操作系统论坛][Sybase专区][IBM dW技术专区]
[病毒求助][病毒与漏洞播报][文档·源码下载]

Java语言深入:关于java数组的深度思考

发布时间:2008.02.25 04:41     来源:赛迪网    作者:lingwen20

刚刚开始接触java数组的人都会听到一句类似的话:java是纯面向对象的语言,他的数组也是一个对象。
    于是乎,我就按照一个对象的方式来使用数组,心安理得。直到我接触到C的数组后,才发现将数组作为一个类来使用在实现上是多么的“不自然”。
    首先我们看一下表面现象,数组创建的时候采用的是如下语句:
    MyClass[] arr = new MyClass[9];
    而普通类采用的是如下语句:
   MyClass obj = new MyClass();
   就是说,创建数组的时候不使用小括号传参。使得数组和普通类看起来就有很多不同,因为小括号里的参数是传递给构造方法的,进而让人感觉数组类是没有构造方法的。
    再往深了想,还有很多让人感觉不自然的东西。可以肯定的是,java确实将数组作为了一个类来处理。还是用上面的例子说明:
    可以通过以下方法得到MyClass[]的Class实例:
    arr.getClass()或MyClass[].class.这样,我就可以向数组类里面“窥探”了。
    Class clazz = MyClass[].class;
    System.out.println(clazz.getConstructors()。length);
    打印出来的结果是0;证明数组类确实没有构造方法。
    如果强行执行clazz.newInstance();就会得到下面的错误。
    java.lang.InstantiationException: [Larraytest.MyClass;
    证明数组类不能够通过普通的反射方式来创建一个实例。
    再看看数组类的“庐山真面目”:
    System.out.println(clazz);
    输出是:
    [Larraytest.MyClass
    对Java Class文件结构稍有了结就知道,这个字符串的意思就是一个元素类型为arraytest.MyClass的一维数组。也就是说,数组类型不是和普通类一样,以一个全限定路径名+类名来作为自己的唯一标示的,而是以[+一个或者多个L+数组元素类全限定路径+类来最为唯一标示的。这个()也是数组和普通类的区别。而这个区别似乎在某种程度上说明数组和普通java类在实现上有很大区别。因为java虚拟机(java指令集)在处理数组类和普通类的时候,肯定会做出区分。我猜想,可能会有专门的java虚拟机指令来处理数组。
    既然我们可以得到数组的Class类实例,就说明肯定需要调用ClassLoader的defineClass(不一定非要是loadClass方法)方法,来构造一个Class实例。java虚拟机规范规定,任何一个可以被加载的类,如果其类文件存储在文件系统上,那么一个*.class文件只能存储一个类信息,也就是说,数组类的信息不可能以类文件的形式存储在本地磁盘上(否则任意一个类都要配有255个数组类了……),既然这样,那就说明java虚拟机肯定内置了一块用来声明数组类的数据(不管是几级数组)。这是符合java虚拟机规范的,规范规定class类数据可以来自任意介质,包括本地磁盘、网络、数据库、内存等等。
    分析到这里,我基本上可以肯定:java对数组对象化的操作的支持是指令级的,也就是说java虚拟机有专门针对数组的指令。数组的Class类实例是java虚拟机动态创建动态加载的,其结构与普通java类的Class实例有一些不同。
    JDK API中有一个java.lang.reflect.Array类,这个类提供了很多方法(绝大多数是native方法,这在另一个方面证明了java对数组的支持是专用指令支持的,否则用本地方法干嘛^_^),用来弥补我们对数组操作的局限性。
    下面这句话用来创建一个一维的、长度为10的、类型为arraytest.MyClass的数组:
    arraytest.MyClass[] arr = (arraytest.MyClass[]) Array.newInstance(arraytest.MyClass, 10);
    下面这句话用来创建一个二维的、3乘5的、类型为arraytest.MyClass的数组:
    int[] arrModel = new int[]{3,5};
    Object arrObj = Array.newInstance(Sub.class, arrModel);
    当然你可以用一个数组的引用指向上面的二维数组,这里我们用一个Object的引用指向他。
    使用的时候,我们也是可以利用Array类提供的方法来实现:
    System.out.println(Array.getLength(arrObj);//第一维长度为3 
    System.out.println(Array.getLength(Array.get(arrObj, 2)));//第二维长度为5,这里如果写3,就会得到你意想之中的
    java.lang.ArrayIndexOutOfBoundsException
    打印结果是如我所想的:
    3 
    5
    对于数组的Class类实例,还有一些奇怪的现象:在运行代码java.lang.reflect.Field fieldarr = clazz.getField("length");的时候,会抛出异常:java.lang.NoSuchFieldException: length,这似乎在说数组类没有length这个域,而这个域其实是我们用的最多的一个(也就是说这个域是肯定存在的)。我想关于数组的Class类实例、数组的实现等,还有很多“猫腻”在里面。
    顺便说一句,java数组最多只能是255维的。这个让人看到了C的影子,嘿嘿。
    “Java把数组当作一个java类来处理”说起来容易,用起来自然,但是细细想来,还是有很多不简单的地方呀。 
          (责任编辑:包春林)


[ 发表评论 ] 字体[  ] [ 打印 ] [ 进入博客 ] [ 进入论坛 ]  [ 推荐给朋友 ]
  相关文章
· J2SE综合:讨论关于Java占用内存的研究 (02-22) · 基础:深入理解Java对象实例生成的例子 (02-22)
· JAVA基础:Java变量类型之间的相互转换 (02-22) · Java入门--深入的理解嵌套类和内部类 (02-22)
· Java入门--对java两个IO基础类的理解 (02-22) · Java入门:如何在内部类中返回外部类对象 (02-22)
· 高级:编写多线程Java应用程序常见问题 (02-22) · Java语言深入:Java类完整构造执行顺序 (02-22)
· 基础:理解构造器--构造器和方法的区别 (02-22) · 用例子来说明面向对象和面向过程的区别 (02-22)
  客户需求反馈表
* 姓  名:
更多资料  了解方案  认识厂商
* 单位名称:
* 联系电话:
* 电子邮件:
  赛迪推荐  
  手机·资费 ·新品·导购·评测·手机资费·宽带
手机搜索  诺基亚 N73 MOTO Z6
  IT产品 ·笔记本·台式机·服务器·打印·投影
IT产品搜索 
  IT技术 ·开发·网管·安全·数据库·操作系统
  信息化 ·热点·专题·访谈·周刊·方案案例
· 工信部“三定”公布 总编制731名设24司局
· 北京发电子商务监管意见 营利性网店须办照
· 直播 08中国城市信息化高峰论坛 案例点评
· 烽火网络校园解决方案 移民安置信息管理系统
  IT博客 ·曾剑秋·项立刚·Java学习·网管
  IT技术论坛 ·开发·网管·安全·数据库·系统