为什么要用嵌套类,内部类?

为什么Java支持类嵌套?
  Java并非一定要支持类嵌套。实际上,如果你看过内部类规范文档,你将会发现类嵌套的应用在哪里。不过,Java支持类嵌套至少有两个好处:
  .令源代码更加清晰
  .减少命名的冲突
  通过类嵌套可以令源代码变得更为清楚,因为你声明了一个包含有一些对象的类,它必须可以管理和允许类中的方法来直接访问对象的字段,以及调用对象的方法--即使是外部类中的私有字段和方法。要理解这个好处,可用以下的一个例子说明,在这个程序中要循环遍历一个Employee对象中的Job对象:
  Listing 1. JobIterator1.java
  // JobIterator1.java
  class Job
  {
  private String jobTitle;
  Job (String jobTitle)
  {
  this.jobTitle = jobTitle;
  }
  public String toString ()
  {
  return jobTitle;
  }
  }
  class Employee
  {
  private String name;
  private Job [] jobs;
  private int jobIndex = 0;
  Employee (String name, Job [] jobs)
  {
  this.name = name;
  this.jobs = jobs;
  }
  String getName ()
  {
  return name;
  }
  boolean hasMoreJobs ()
  {
  return jobIndex < jobs.length;
  }
  Job nextJob ()
  {
  return !hasMoreJobs () ? null : jobs [jobIndex++];
  }
  }
  class JobIterator1
  {
  public static void main (String [] args)
  {
  Job [] jobs = { new Job ("Janitor"), new Job ("Delivery Person") };
  Employee e = new Employee ("John Doe", jobs);
  System.out.println (e.getName () + " works the following jobs:/n");
  while (e.hasMoreJobs ())
  System.out.println (e.nextJob ());
  }
  }
  运行后, JobIterator1产生如下的输出:
  John Doe works the following jobs:
  
     Janitor
     Delivery Person
  JobIterator1包含有Job, Employee和JobIterator1类。Job封装了一个job的title,而Employee封装了一个员工的名字以及该员工所做工作的一个数组。JobIterator1包含有一个main()的方法用来创建Job和Employee对象,并且打印出该员工的名字和工作。
  Employee类中包含有方法hasMoreJobs()和nextJob()。这两个方法构成了一个循环。当一个Employee对象初始化时,在私有jobs数组中的一个内部索引被设置为0。如果该索引的值比jobs数组的长度少,hasMoreJobs()方法返回一个布尔值true。nextJob()使用该索引值由数组中返回一个Job对象--并且增加该索引的值,以便下一次调用nextJob()时返回下一个job对象的一个引用。
  JobIterator1类在设计上有一些问题。首先,在循环结束后你不能重新启动一个循环。当然,你可以很容易地解决这个问题,只要Employee类加入一个reset()方法就可以了,该方法将jobIndex设置为0。第二个问题更加严重,你不能为一个Employee创建多个循环。这是由于hasMoreJobs()和nextJob()方法已经写死在Employee中了。要解决这两个问题,开发者可以声明一个循环类,它的对象可以循环jobs数组。在循环结束后,程序可以通过创建一个新的循环对象来启动一个新的循环。同样,通过创建多个循环对象,一个程序可以在同一个Employee对象的jobs数组上执行多个循环。列表2的程序为我们展示了一个名字为JobIterator的循环类:
  Listing 2. JobIterator2.java
  // JobIterator2.java
  class Job
  {
  private String jobTitle;
  
  Job (String jobTitle)
  {
  this.jobTitle = jobTitle;
  }
  public String toString ()
  {
  return jobTitle;
  }
  }
  class Employee
  {
  private String name;
  private Job [] jobs;
  Employee (String name, Job [] jobs)
  {
  this.name = name;
  this.jobs = jobs;
  }
  String getName ()
  {
  return name;
  }
  JobIterator getJobIterator ()
  {
  return new JobIterator (jobs);
  }
  }
  class JobIterator
  {
  private Job [] jobs;
  private int jobIndex = 0;
  JobIterator (Job [] jobs)
  {
  this.jobs = jobs;
  }
  boolean hasMoreJobs ()
  {
  return jobIndex < jobs.length;
  }
  Job nextJob ()
  {
  return !hasMoreJobs () ? null : jobs [jobIndex++];
  }
  }
  class JobIterator2
  {
  public static void main (String [] args)
  {
  Job [] jobs = { new Job ("Janitor"), new Job ("Delivery Person") };
  Employee e = new Employee ("John Doe", jobs);
  System.out.println (e.getName () + " works the following jobs:/n");
  JobIterator ji = e.getJobIterator ();
  while (ji.hasMoreJobs ())
  System.out.println (ji.nextJob ());
  }
  }
  JobIterator2的输出和JobIterator1一样,但有所不同的是,JobIterator2将循环的代码由Employee转移到JobIterator。还有,Employee声明了一个getJobIterator()的方法,可返回一个新的JobIterator对象引用。要注意到JobIterator和 Employee是紧耦合的类:JobIterator的构造器需要一个Employee私有jobs数组的引用。我们要记下这个耦合关系,因为它为理解类嵌套内部的工作提供了一个线索。
  虽然JobIterator2很方便地解决了JobIterator1的问题,但这个新的程序又引入了一个新问题:由于新加入了一个和Employee类处在同一级的JobIterator类,这样将来就不能在同一级的源文件中加入一个普遍的JobIterator接口。因为在源文件的同一级中,你不同拥有同样名字的两个类/接口。虽然在我们的例子中这不是一个严重的问题,但是在重要的程序中,有些情况下必须在同一个源文件中使用相同名字的类/接口。要令那些名字共存,你必须认识到一些类完全依赖其它的类。你应该在其依赖的类中声明这些类。列表3展示了如何在一个Employee类中声明一个JobIterator类--JobIterator依赖Employee类。
  Listing 3. JobIterator3.java
  
  // JobIterator3.java
  
  class Job
  {
  private String jobTitle;
  
  Job (String jobTitle)
  {
  this.jobTitle = jobTitle;
  }
  
  public String toString ()
  {
  return jobTitle;
  }
  }
  
  class Employee
  {
  private String name;
  private Job [] jobs;
  
  Employee (String name, Job [] jobs)
  {
  this.name = name;
  this.jobs = jobs;
  }
  
  String getName ()
  {
  return name;
  }
  
  JobIterator getJobIterator ()
  {
  return new JobIterator ();
  }
  
  class JobIterator
  {
  private int jobIndex = 0;
  
  public boolean hasMoreJobs ()
  {
  return jobIndex < jobs.length;
  }
  
  public Object nextJob ()
  {
  return !hasMoreJobs () ? null : jobs [jobIndex++];
  }
  }
  }
  class JobIterator3
  {
  public static void main (String [] args)
  {
  Job [] jobs = { new Job ("Janitor"), new Job ("Delivery Person") };
  Employee e = new Employee ("John Doe", jobs);
  System.out.println (e.getName () + " works the following jobs:/n");
  Employee.JobIterator eji = e.getJobIterator ();
  while (eji.hasMoreJobs ())
  System.out.println (eji.nextJob ());
  }
  }
  JobIterator3的输出和JobIterator1、JobIterator2的一样,它使用了类嵌套:Employee  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值