概述
Note that the argument to filter() must be final.This is required by the anonymous inner class so that it can use an object from outside its scope.(注意,传向filter()的参数必须是final的。这在匿名内部类中是必需的,这样它才能够使用来自该类范围之外的对象)
It's possible to place a class definition within another class definition.This is called an inner class.
The inner class is a valuable feature because it allows you to group classes that logically belong together and to control the visibility of one within the other.(内部类是一种非常有用的特性,因为它允许你把一些逻辑相关的类组织在一起,并控制位于内部的类的可视性。)However,it's important to understand that inner classes are distinctly different from composition.(然而必须要了解,内部类和组合是完全不同的概念,这一点很重要。)
At first,inner classes look like a simple code-hiding mechanism:You place classes inside other classes.(在最初,内部类看起来就像是一种代码隐藏机制:将类置于其他类的内部。)You'll learn,however,that the inner class does more than that——it knows about and can communicate with the surrounding class——and the kind of code you can write with inner classes is more elegant and clear,although there's certainly no guarantee of this.(但是,你将会了解到,内部类远不止此,它了解外围类,并能与之通信;而且你用内部类写出来的代码更加优雅而清晰,尽管并不总是这样。)
Initially,inner classes may seem odd,and it will take some time to become comfortable using them in your designs.(最初,内部类可能看起来有些奇怪,而且要花时间才能在设计中轻松地使用它们。)The need for inner classes isn't always obvious,but after the basic syntax and semantics of inner classes have been described,the section "Why inner classes?" should begin to make clear the benefits of inner classes.(对内部类的需求并非总是很明显的,但是在描述完内部类的基本语法与语义之后,10.8节就应该使得内部类的益处明确显见了。)
After that section,the remainder of the chapter contains more detailed explorations of the syntax of inner classes.These features are provided for language completeness,but you might not need to use them,at least not at first.So the initial parts of the chapter might be all you need for now,and you can leave the more detailed explorations as reference material.(你可以将更详尽的探索当作参考资料)
Creating inner classes
- public class Parcel1 {
- class Contents{
- private int i = 11;
- public int value(){ return i; }
- }
- class Destination{
- private String label;
- Destination(String whereTo){
- label = whereTo;
- }
- String readLabel(){ return label; }
- }
- //Using inner classes looks just like
- //using any other class,within Parcel1
- public void ship(String dest){
- Contents c = new Contents();
- Destination d = new Destination(dest);
- System.out.println(d.readLabel());
- }
- public static void main(String[] args){
- Parcel1 p = new Parcel1();
- p.ship("Tasmania");
- }
- }
The inner classes used inside ship() look just like ordinary classes.Here,the only practical difference is that the names are nested within Parcel1.You'll see in a while that this isn't the only difference.
- public class Parcel2 {
- class Contents{
- private int i = 11;
- public int value(){ return i; }
- }
- class Destination{
- private String label;
- Destination(String whereTo){
- label = whereTo;
- }
- String readLabel(){ return label; }
- }
- public Destination to(String s){
- return new Destination(s);
- }
- public Contents contents(){
- return new Contents();
- }
- public void ship(String dest){
- Contents c = contents();
- Destination d = to(dest);
- System.out.println(d.readLabel());
- }
- public static void main(String[] args){
- Parcel2 p = new Parcel2();
- p.ship("Tasmania");
- Parcel2 q = new Parcel2();
- //Defining references to inner classes:
- Parcel2.Contents c = q.contents();
- Parcel2.Destination d = q.to("Borneo");
- }
- }
If you want to make an object of the inner class anywhere except from within a non-static method of the out class,you must specify the type of what object as OuterClassName.InnerClassName,as seen in main().(如果想从外部类的非静态方法之外的任意位置创建某个内部类的对象,那么必须像在main()方法中那样,具体地指明这个对象的类型:OuterClassName.InnerClassName。)(意思是说在外部类的静态方法中或者其他的非静态方法中)
The link to the outer class
So far,it appears that inner classes are just a name-hiding and code organization scheme(代码组织方案),which is helpful but not totally compelling(引人注目的;强制的;激发兴趣的).(到目前为止,内部类似乎还只是一种名字隐藏和组织代码的模式。这些是有用的,但还不是最引人注目的,)However,there's another twist.(它还有其他的用途。)When you create an inner class,an object of that inner class has a link to the enclosing object that made it,and so it can access the members of that enclosing object( 宿主类的对象)——without any special qualifications(当生成一个内部类的对象时,此对象与制造它的外围对象之间就有了一种联系,所以它能访问其外围对象的所有成员,而不需要任何特殊条件。).In addition,inner classes have access rights to all the elements in the enclosing class.The following example demonstrates this:
- interface Selector{
- boolean end();
- Object current();
- void next();
- }
- public class Sequence {
- private Object[] items;
- private int next = 0;
- public Sequence(int size){ items = new Object[size]; }
- public void add(Object x){
- if(next < items.length){
- items[next++] = x;
- }
- }
- private class SequenceSelector implements Selector{
- private int i = 0;
- @Override
- public boolean end() {
- return i == items.length;
- }
- @Override
- public Object current() {
- return items[i];
- }
- @Override
- public void next() {
- if(i<items.length) i++;
- }
- }
- public Selector selector(){
- return new SequenceSelector();
- }
- public static void main(String[] args){
- Sequence sequence = new Sequence(10);
- for(int i = 0; i < 10; i++){
- sequence.add(Integer.toString(i));
- }
- Selector selector = sequence.selector();
- while(!selector.end()){
- System.out.print(selector.current() + " ");
- selector.next();
- }
- }
- }
The Sequence is simply a fixed-sized array of Object with a class wrapped around it.(Sequence只是一个固定大小的Object的数组,以类的形式包装了起来。)You call add() to add a new Object to the end of the sequence(if there's room left.)(可以使用add()在序列末增加新的Object(只要还有空间)。)To fetch each of the objects in a Sequence,there's an interface called Selector.(要获取Sequence中的每一个对象,可以使用Selector接口。)This is an example of the Iterator design pattern that you shall learn more about later in the book.(这是“迭代器”设计模式的一个例子,在本书稍后的部分将更多地学习它。)A Selector allows you to see if you're at the end(),to access the current() Object,and to move to the next() Object in the Sequence.(Selector允许你检查序列是否到末尾了(end()),访问当前对象(current()),以及移到序列中的下一个对象(next())。)Because Selector is an interface,other classes can implement the interface in their own ways,and other methods can take the interface as an argument,in order to create more general-purpose(一般用途的) code.(因为Selector是一个接口,所以别的类可以按它们自己的方式来实现这个接口,并且别的方法能以此接口为参数,来生成更加通用的代码。)
Here,the SequenceSelector is a private class that provides Selector functionality.(这里,SequenceSelector是提供Selector功能的private类)In main(),you can see the creation of a Sequence,followed by the addition of a number of String objects.(可以看到,在main()中创建了一个Sequence,并向其中添加了一些String对象。)Then,a Selector is produced with a call to selector(),and this is used to move throuth the Sequence and select each item.(然后通过调用selector()获取一个Selector,并用它在Sequence中移动和选择每一个元素。)
At first,the creation of SequenceSelector looks like just another inner class.(最初看到SequenceSelector,可能会觉得它只不过是一个内部类罢了。)But examine it closely.Note that each of the methods——end(),current(),and next()——refers to items,which is a reference that isn't part of SequenceSelector,but is instead a private field in the enclosing class.(请仔细观察它,注意方法end()、current()和next()都用到了item,这是一个引用,它并不是SequenceSelector的一部分,而是外围类中的一个private字段。)However,the inner class can access methods and fields from the enclosing class as if it owned them.This turns out to be very convenient,as you can see in the preceding example.
So an inner class has automatic access to the members of the enclosing class.(所以内部类自动拥有对其外围类所有成员的访问权。)How can this happen?The inner class secretly captures a reference to the particular object of the enclosing class that was responsible for creating it.(当某个外围类的对象创建了一个内部类对象时,此内部类对象必定会秘密地捕获一个指向那个外围类对象的引用。) Then,when you refer to a member of the enclosing class,that reference is used to select that member.(然后,在你访问此外围类的成员时,就是用那个引用来选择外围类的成员。)Fortunately,the compiler takes care of all these details for you,but now you can see that an object of an inner class can be created only in association with an object of the enclosing class(when,as you shall see,the inner class is non-static.)(幸运的是,编译器会帮你处理所有的细节,但你现在可以看到:内部类的对象只能在与其外围类的对象相关联的情况下才能被创建(就像你应该看到的,在内部类是非static类时)。)Construction of the inner-class object requires the reference to the object of the enclosing class,and the compiler will complain if it cannot access the reference.Most of the time this occurs without any intervention(介入;调停) on the part of the programmer.(构建内部类对象时需要一个指向其外围类对象的引用,如果编译器访问不到这个引用就会报错。不过绝大多数时候这都无需程序员操心。)
Using .this and .new
If you need to produce the reference to the outer-class object,you name the outer class followed by a dot and this.(如果你需要生成对外部类对象的引用,可以使用外部类的名字后面紧跟圆点和this。)The resulting reference is automatically the correct type,which is known and checked at compile time,so there is no runtime overhead.(这样产生的引用自动地具有正确的类型,这一点在编译器就被知晓并受到检查,因此没有任何运行时开销。)Here's an example that shows how to use .this:
- public class DotThis {
- void f(){
- System.out.println("DotThis.f()");
- }
- public class Inner{
- public DotThis outer(){
- return DotThis.this;
- //A plain "this" wouble be Inner's "this"
- }
- }
- public Inner inner(){ return new Inner(); }
- public static void main(String[] args){
- DotThis dt = new DotThis();
- DotThis.Inner dti = dt.inner();
- dti.outer().f();
- }
- }
运行结果:
DotThis.f() |
Sometimes you want to tell some other object to create an object of one of its inner classes.(有时你可能想要告知某些其他对象,去创建其某个内部类的对象。)To do this you must provide a reference to the other outer-class object in the new expression,using the .new syntax,like this:(要实现此目的,你必须在new表达式中提供对其外部类对象的引用,这是需要使用.new语法的,就像下面这样:)
- //Creating an inner class directly using the .new syntax.
- public class DotNew {
- public class Inner{}
- public static void main(String[] args){
- DotNew dn = new DotNew();
- DotNew.Inner dni = dn.new Inner();
- /*
- No enclosing instance of type DotNew is accessible. Must qualify the allocation with
- an enclosing instance of type DotNew (e.g. x.new A() where x is an instance of
- DotNew).
- */
- //new DotNew.Inner(); //报错
- }
- }
To create an object of the inner class directly,you don't follow the same form and refer to the outer class name DotNew as you might expect,but instead you must use an object of the outer class to make an object of the inner class, as you can see above.(要想直接创建内部类的对象,你不能按照你想象的方式,去引用外部类的名字DotNew,而是必须使用外部类的对象来创建该内部类对象,就像在上面的程序中所看到的那样。)This also resolves the name scoping issues for the inner class,so you don't say(indeed,you can't say) dn.new DotNew.Inner();(这也解决了内部类名字作用域的问题,因此你不必声明(实际上你不能声明)dn.new DotNew.Inner())
It's not possible to create an object of the inner class unless you already have an object of the outer class.(在拥有外部类对象之前是不可能创建内部类对象的。)This is because the object of the inner class is quietly connected to the object of the outer class that it was made from.(这是因为内部类对象会暗暗地连接到创建它的外部类对象上。)However,if you make a nested class(a static inner class),then it doesn't need a reference to the outer-class object.(但是,如果你创建的是嵌套类(静态内部类),那么它就不需要对外部类对象的引用。)
Here,you see the use of .new applied to the "Parcel" example:(下面你可以看到将.new应用于Parcel的示例:)
- //Using .new to create instances of inner classes.
- public class Parcel3 {
- class Contents{
- private int i = 11;
- public int value(){ return i; }
- }
- class Destination{
- private String label;
- Destination(String whereTo){ label = whereTo; }
- String readLabel(){ return label; }
- }
- public static void main(String[] args){
- Parcel3 p = new Parcel3();
- Parcel3.Contents c = p.new Contents();
- Parcel3.Destination d = p.new Destination("Tasmania");
- }
- }
111
转载于:https://blog.51cto.com/gzzjsoft/1036906
最后
以上就是冷傲小鸭子为你收集整理的Inner Classes的全部内容,希望文章能够帮你解决Inner Classes所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复