讲解Java中如何构造内部类对象以及访问对象

通过反射构造内部类对象
首先在 javalang 包下写一个包含内部类的类:

package javalang;

public class Outer {
  public static class Inner1{}
}

注意这个类是 public static,后面我们慢慢把这些修饰符去掉。

要想通过反射来创建 Inner1 对象,首先要获得 Inner1 的 Class 对象。我们在 Outer 中写上 main 方法:

public class Outer {
  public static class Inner1{}

  public static void main(String[] args) {
    System.out.println(Inner1.class);
  }
}

输出结果:

class javalang.Outer$Inner1

然后我们试一下这个类名对不对:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1"));
}

运行一下,没错。然后就是用它来创建对象。创建对象要靠构造方法。这个类有没有构造方法呢?我们可以这么写:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1").getConstructors().length);
}

运行一下,输出 1。看来有。然后看看这个构造方法是什么样子的:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1").getConstructors()[0]);
}

输出结果:public javalang.Outer$Inner1()。这就是缺省构造方法嘛。所以我们可以这样写:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1")
      .getConstructors()[0].newInstance();
}

输出结果:javalang.Outer$Inner1@ca0b6。这说明执行成功了。

接下来我们把 Inner 的 public 关键字去掉,然后再运行。结果报错了:

Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 0

这说明没有找到构造方法。真的没有吗?我们把 main 方法改回来:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1")
      .getConstructors().length);
}

输出结果:0。真的没有构造方法吗?其实不是,只是构造方法不是公开的。这时我们必须用 getDeclaredConstructors() 来获得:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1")
      .getDeclaredConstructors().length);
}

输出结果:1。这就把构造方法找到了。然后我们继续调用这个构造方法:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1")
      .getDeclaredConstructors()[0].newInstance());
}

输出结果:javalang.Outer$Inner1@ca0b6。现在我们可以用反射来构造非公开内部类的对象了。

接下来,我们再把 static 关键字去掉。这时候报错了:
Exception in thread "main" java.lang.IllegalArgumentException: wrong number of arguments

这说明什么呢?我们调用的时候没有传参数,而错误内容就是说参数数量不正确。那么这个构造方法有什么参数呢?我们改一下代码看看:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1")
      .getDeclaredConstructors()[0]);
}

输出结果:javalang.Outer$Inner1(javalang.Outer)

原来构造方法里面需要一个 Outer 类型的参数。这好办:

public static void main(String[] args) throws Exception {
  System.out.println(Class.forName("javalang.Outer$Inner1")
      .getDeclaredConstructors()[0].newInstance(new Outer()));
}

输出结果:

javalang.Outer$Inner1@ca0b6

OK,原来如此。看来非静态的内部类没有缺省的构造方法,构造时需要传一个外部类的实例作为参数。

Java: 如何访问一个对象
对 Java 初学者来说一个头疼的问题是,如何决定把一个对象是定义为方法变量,还是定义为成员变量?
 
最开始初学者还不会关心这点。但是当写出来的程序越来越大,类越来越多时,这种苦恼也应运而生。
 
但我这里要写的是:如何随心所欲的安排一个对象,让你随时可以访问。掌握了这点,你就可以自由的决定将一个对象放在什么地方了。
 
下面举一个简单的例子:

public class AccessingObject { 

  public static void main(String[] args) {
    Date date = new Date();
  } 

  // 获得 date 对象一小时后的时间
  private static void anHourLater() {
    // 这里如何获得 main() 方法中的 date 变量?
  }
}

正如 anHourLater() 方法中描述的,想要获得 date 一小时后的时间。怎么办呢?有下面几种方法。
 
(1)参数传递

public class AccessingObject { 

  public static void main(String[] args) {
    Date date = new Date();
    anHourLater(date);
  } 

  // 获得 date 对象一小时后的时间
  private static void anHourLater(Date d) {
    Date anHourLater = new Date(d.getTime() + 3600000);
  }
}

(2)定义为成员。成员可以由所有方法访问,成员的初始化可以放在定义的地方,也可以放在任何一个方法里。

public class AccessingObject { 

  private static Date date; 

  public static void main(String[] args) {
    date = new Date();
    anHourLater();
  } 

  // 获得 date 对象一小时后的时间
  private static void anHourLater() {
    Date anHourLater = new Date(date.getTime() + 3600000);
  }
}

(3)放到另外一个类当中去。在下面的例子中,DateHolder.date 可以被同一个包中的所有类访问,而不仅限于 AccessingObject 类。

public class AccessingObject { 

  public static void main(String[] args) {
    DateHolder.date = new Date();
  } 

  // 获得 date 对象一小时后的时间
  private static void anHourLater() {
    Date anHourLater = new Date(DateHolder.date.getTime() + 3600000);
  }
} 

class DateHolder {
  public static Date date;
}

这三个例子比较起来,前两个只能在类的内部使用,相对比较安全。如果你不希望这个对象被别的类直接修改,就不要用第三种方式。
 
第一种方式和第二种方式的区别在于:如果一个对象只在方法中使用,那么当方法执行完后,这个对象能够很容易的被回收(注意,不是马上被回收)。如果定义为类的成员,那么只有当它所在的类被回收之后,这个对象才会被回收。显然,第一种方式是最节约资源的,我们应该尽量使用第一种方式。
 
回头再看看这三个例子,如果 main() 方法要获得 anHourLater() 方法中得出的一小时后时间,它也有几种对应的方式。后两个例子就不用改了,date 对象是可以直接访问的;第一个例子,有两种修改方式:
 
(1)作为返回值

public class AccessingObject { 

  public static void main(String[] args) {
    Date date = new Date();
    Date anHourLater = anHourLater(date);
  } 

  // 获得 date 对象一小时后的时间
  private static Date anHourLater(Date d) {
    return new Date(d.getTime() + 3600000);
  }
}

(2)直接修改参数的内容

public class AccessingObject { 

  public static void main(String[] args) {
    Date date = new Date();
    anHourLater(date);
  } 

  // 获得 date 对象一小时后的时间
  private static void anHourLater(Date d) {
    d.setTime(d.getTime() + 3600000);
  }
}

其中第二种方法要慎用,因为随便动人家的东西是不对的,你不知道方法的调用者喜不喜欢你这么做。

(0)

相关推荐

  • Java编程中静态内部类与同步类的写法示例

    java静态内部类 将某个内部类定义为静态类,跟将其他类定义为静态类的方法基本相同,引用规则也基本一致.不过其细节方面仍然有很大的不同.具体来说,主要有如下几个地方要引起各位程序开发人员的注意.     (一)一般情况下,如果一个内部类不是被定义成静态内部类,那么在定义成员变量或者成员方法的时候,是不能够被定义成静态成员变量与静态成员方法的.也就是说,在非静态内部类中不可以声明静态成员.     (二)一般非静态外部类可以随意访问其外部类的成员变量以及方法(包括声明为private的方法),但是

  • 简单谈谈java中匿名内部类构造函数

    先看看下面的代码能不能编译通过: public static void main(String[] args) { List l1 = new ArrayList(); List l2 = new ArrayList(){}; List l3 = new ArrayList(){{}}; System.out.println(l1.getClass() == l2.getClass() ); System.out.println(l2.getClass() == l3.getClass() );

  • Java内部类_动力节点Java学院整理

    内部类是指在一个外部类的内部再定义一个类.类名不需要和文件夹相同.内部类可以是静态static的,也可用public,default,protected和private修饰.(而外部顶级类即类名和文件名相同的只能使用public和default). 注意:内部类是一个编译时的概念,一旦编译成功,就会成为完全不同的两类.对于一个名为outer的外部类和其内部定义的名为inner的内部类.编译完成后出现outer.class和outer$inner.class两类.所以内部类的成员变量/方法名可以和

  • Java中的匿名内部类小结

    java内部类分为: 成员内部类.静态嵌套类.方法内部类.匿名内部类 . 在java的世界里,提供了匿名内部类语法,用于帮助大家简化代码,本文简要从接口,抽象类以及常规类以代码的形式描述其常用模式. 1. 接口模式 public interface IWriter { void write(); } public static void main(String[] args) { IWriter writer = new IWriter() { @Override public void wri

  • 深入浅析Java中Static Class及静态内部类和非静态内部类的不同

    java中的类可以是static吗?答案是可以.在java中我们可以有静态实例变量.静态方法.静态块.类也可以是静态的. java允许我们在一个类里面定义静态类.比如内部类(nested class).把nested class封闭起来的类叫外部类.在java中,我们不能用static修饰顶级类(top level class).只有内部类可以为static.      静态内部类和非静态内部类之间到底有什么不同呢?下面是两者间主要的不同. (1)内部静态类不需要有指向外部类的引用.但非静态内部

  • 讲解Java中如何构造内部类对象以及访问对象

    通过反射构造内部类对象 首先在 javalang 包下写一个包含内部类的类: package javalang; public class Outer { public static class Inner1{} } 注意这个类是 public static,后面我们慢慢把这些修饰符去掉. 要想通过反射来创建 Inner1 对象,首先要获得 Inner1 的 Class 对象.我们在 Outer 中写上 main 方法: public class Outer { public static cl

  • 基于Java中对域和静态方法的访问不具有多态性(实例讲解)

    1.将方法调用同方法主体关联起来被称为 2.编译期绑定(静态)是在程序编译阶段就确定了引用对象的类型 3.运行期绑定(动态绑定)是指在执行期间判断所引用对象的实际类型,根据其实际的类型调用其相应的方法 4.除了static方法和final方法(private方法属于final方法),其他所有方法都是后期绑定,Java中所有的方法都是通过动态绑定来实现多态 5.访问某个域的行为不具有多态性 package polymorphism; class SuperField { public int fi

  • 简单讲解java中throws与throw的区别

    Java中throws和throw的区别讲解 当然,你需要明白异常在Java中式以一个对象来看待. 并且所有系统定义的编译和运行异常都可以由系统自动抛出,称为标准异常,但是一般情况下Java 强烈地要求应用程序进行完整的异常处理,给用户友好的提示,或者修正后使程序继续执行. 直接进入正题哈: 1.用户程序自定义的异常和应用程序特定的异常,必须借助于 throws 和 throw 语句来定义抛出异常. 1.1   throw是语句抛出一个异常. 语法:throw (异常对象);         

  • 深入浅出讲解Java中的枚举类

    目录 一.枚举类的使用 二.如何定义枚举类 背景:类的对象只有有限个,确定的.举例如下: > 星期: Monday (星期一).-.. Sunday (星期天) > 性别: Man (男). Woman (女) > 季节: Spring (春节).--.. Winter (冬天) > 支付方式: Cash (现金). WeChatPay (微信). Alipay (支付宝) BankCard (银 行卡). CreditCard (信用卡) > 就职状态: Busy . Fr

  • Java中的Kotlin 内部类原理

    目录 Java 中的内部类 OutterJava.class InnJava.class Kotlin 中的内部类 总结 Java 中的内部类 这是一个 Java 内部类的简单实现: public class OutterJava {    private void printOut() {        System.out.println("AAA");   } ​    class InnJava {        public void printInn() {        

  • 实例讲解Java中的synchronized

    一.使用场景 在负责后台开发的时候,很多时候都是提供接口给前端开发人员去调用,会遇到这样的场景: 需要提供一个领奖接口,每个用户名只能领取一次,我们可以将成功领取的用户在数据库用个标记保存起来.如果这个用户再来领取的时候,查询数据库看该用户是否领取过. 但是问题来了,假设用户手速很快,极短时间内点了两次领奖按钮(前端没有进行控制,我们也不能依赖前端去控制).那么可能掉了两次领奖接口,而且有可能第二次调用的时候查询数据库的时候,第一次领奖还没有执行完成更新领奖标记. 这种场景就可以使用到synch

  • 关于JAVA中stream流的基础处理(获取对象字段和对象批量处理等)

    目录 Stream流程处理情况 1:按指定字段获取集合对象中的字段: 2:按指定字段对集合对象进行排序: 3: 按指定字段对集合对象去重处理 4: 对集合对象批量处理 5: 将集合对象中指定字段转数组 Stream流程处理情况 1:按指定字段获取集合对象中的字段: List<String> idList = initUserList.stream().map(User::getId).collect(Collectors.toList()); 2:按指定字段对集合对象进行排序: List<

  • 举例讲解Java中Piped管道输入输出流的线程通信控制

    PipedOutputStream和PipedInputStream 在java中,PipedOutputStream和PipedInputStream分别是管道输出流和管道输入流. 它们的作用是让多线程可以通过管道进行线程间的通讯.在使用管道通信时,必须将PipedOutputStream和PipedInputStream配套使用. 使用管道通信时,大致的流程是:我们在线程A中向PipedOutputStream中写入数据,这些数据会自动的发送到与PipedOutputStream对应的Pip

  • 讲解Java中的基础类库和语言包的使用

    Java基础类库 Java 的类库是 Java 语言提供的已经实现的标准类的集合,是 Java 编程的 API(Application Program Interface),它可以帮助开发者方便.快捷地开发 Java 程序.这些类根据实现的功能不同,可以划分为不同的集合,每个集合组成一个包,称为类库.Java 类库中大部分都是由Sun 公司提供的,这些类库称为基础类库. Java 语言中提供了大量的类库共程序开发者来使用,了解类库的结构可以帮助开发者节省大量的编程时间,而且能够使编写的程序更简单

  • 举例讲解Java中的Stream流概念

    1.基本的输入流和输出流 流是 Java 中最重要的基本概念之一.文件读写.网络收发.进程通信,几乎所有需要输入输出的地方,都要用到流. 流是做什么用的呢?就是做输入输出用的.为什么输入输出要用"流"这种方式呢?因为程序输入输出的基本单位是字节,输入就是获取一串字节,输出就是发送一串字节.但是很多情况下,程序不可能接收所有的字节之后再进行处理,而是接收一点处理一点.比方你下载魔兽世界,不可能全部下载到内存里再保存到硬盘上,而是下载一点就保存一点.这时,流这种方式就非常适合. 在 Jav

随机推荐