`
sdustyongz
  • 浏览: 89561 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java编程思想final的理解

 
阅读更多
根据上下文环境,java的关键字final也存在着细微的区别,但通常指的是“这是无法改变的。”不想改变的理由由两种:一种是效率,另一种是设计。由于两个原因相差很远,所以关键子final可能被吴用。

   接下来介绍一下使用到fianl的三中情况:数据,方法,类。

 

   final数据

   许多编程语言都有某种方法,来向编译器告知一块数据是恒定不变的。有时数据的恒定不变是很有用的,例如:

1,一个编译时恒定不变的常量

2,一个在运行时初始化,而你不希望它被改变。

   对于编译期常量的这种情况,编译器可以将该常量值代入任何可能用到它的计算式中,也就是说,可以在编译期就执行计算式,这减轻了一些运行时的负担。在java中,这类常量必须是基本类型,并且以final表示。在对这个常量定义时,必须进行赋值。

   一个即是static又是fianl的域只占一段不能改变的存储空间。

   当final应用于对象引用时,而不是基本类型时,其含义有些让人疑惑。对基本类型使用fianl不能改变的是他的数值。而对于对象引用,不能改变的是他的引用,而对象本身是可以修改的。一旦一个final引用被初始化指向一个对象,这个引用将不能在指向其他对象。java并未提供对任何对象恒定不变的支持。这一限制也通用适用于数组,它也是对象。

   下面的事例示范fianl域的情况。注意,根据惯例,即是static又是fianl的域(即编译器常量)将用大写表示,并用下划分割个单词:

[java] view plaincopy
package reusing; 
 
//: reusing/FinalData.java 
// The effect of final on fields. 
import java.util.*; 
import static net.mindview.util.Print.*; 
 
class Value { 
  int i; // Package access 
  public Value(int i) { this.i = i; } 

 
 
 
public class FinalData { 
  private static Random rand = new Random(47); 
  private String id; 
  public FinalData(String id) { this.id = id; } 
  // Can be compile-time constants: 
  private final int valueOne = 9; 
  private static final int VALUE_TWO = 99; 
  // Typical public constant: 
  public static final int VALUE_THREE = 39; 
  // Cannot be compile-time constants: 
  private final int i4 = rand.nextInt(20); 
  static final int INT_5 = rand.nextInt(20); 
  private Value v1 = new Value(11); 
  private final Value v2 = new Value(22); 
  private static final Value VAL_3 = new Value(33); 
  // Arrays: 
  private final int[] a = { 1, 2, 3, 4, 5, 6 }; 
  public String toString() { 
    return id + ": " + "i4 = " + i4 + ", INT_5 = " + INT_5; 
  } 
 
 
  public static void main(String[] args) { 
    FinalData fd1 = new FinalData("fd1"); 
    //! fd1.valueOne++; // Error: can't change value 
    fd1.v2.i++; // Object isn't constant! 
    fd1.v1 = new Value(9); // OK -- not final 
    for(int i = 0; i < fd1.a.length; i++) 
      fd1.a[i]++; // Object isn't constant! 
    //! fd1.v2 = new Value(0); // Error: Can't 
    //! fd1.VAL_3 = new Value(1); // change reference 
    //! fd1.a = new int[3]; 
    print(fd1); 
    print("Creating new FinalData"); 
    FinalData fd2 = new FinalData("fd2"); 
    print(fd1); 
    print(fd2); 
  } 
}  
 
/* Output:
fd1: i4 = 15, INT_5 = 18
Creating new FinalData
fd1: i4 = 15, INT_5 = 18
fd2: i4 = 13, INT_5 = 18
*/ 
     由于valueOne和VALUE_TWO都是带有编译时数值的fianl基本类型,所以它们二者均可以用作编译期常量,并且没有重大区别。VALUE_THREE是一种更加典型的对常量进行定义的方式:定义为public,可以被任何人访问;定义为static,则强调只有一份;定义为fianl,这说明它是个常量。请注意带有恒定初始值(即,编译期常量)的final static基本类型全用大写字母命名,并且字母与字母之间用下划线隔开。

   我们不能因为某些数据是fianl的就认为在编译时可以知道它的值。在运行时使用随机数来初始化i4和INT_5的值叫说明了这一点。事例部分也展示了将fianl数据定义为static和非static的区别。此区别只有当数值在运行时内被初始化时才会显现,这是因为在编译器对编译时的数值一视同仁(并且他们可能因为优化而消失)。当运行时会看见这个区别。请注意,在此fd1和fd2中i4的值是唯一的,每次都会被初始化为15,13。INT_5的值是不可以通过创建第二个FinalData对象加以改变的。这是因为他是static的,在装载类时(也就是第一次创建这个类对象时)已经被初始化,而不是每次创建都初始化。

  


如果看上面的事例来理解我标记颜色的的部分有点困难的话,请看下面的事例:

  


[java] view plaincopy
public class B3 { 
    static Random r =new Random(12); 
    final int int1= r.nextInt(100);//产生0-99的随机数 
    static final int INT_2= r.nextInt(100); 
     
 
    public static void main(String[] args) { 
        B3 b1=new B3(); 
        System.out.println("int1:"+b1.int1+"    INT_2:"+b1.INT_2); 
        B3 b2=new B3(); 
        //b2.INT_2=100;//错误的赋值 
        System.out.println("int1:"+b2.int1+"    INT_2:"+b2.INT_2); 
 
    } 
 

启动main()先执行的是B3 b1=new B3();,创建B3的第一个对象,这将会先初始化static final int INT_2= r.nextInt(100);,然后是初始化final int int1= r.nextInt(100);,所以第一条输出语句的结果是int1:12    INT_2:66。接下来创建B3的第二个对象,这也会导致B3类中成员的初始化,但static final int INT_2= r.nextInt(100);不会在被初始化,为什么前面已经提过。输出的结果是int1:56    INT_2:66。两次的输出INT_2的值都是一样的。

   在说回我们的第一个事例,V1到VAL_3说明final引用的意义。正如在main()方法中看见的,可以改变对象数组a的值,但不能将a的引用指向另一个对象。看起来使基本类型成为fianl比引用类型成为final的用处大。

    java也许生成"空白final",所谓空白final是指被声明为final但又未给初值的域。无论什么情况下编译器都会保证final域在使用前初始化。但空白final在fianl的使用上提供了很大的灵活性,为此,一个fianl域可以根据某些对象有所不同,却又保持恒定不变的特性。下面的事例说明了一点。


[java] view plaincopy
class Poppet { 
  private int i; 
  Poppet(int ii) { i = ii; } 

 
public class BlankFinal { 
  private final int i = 0; // Initialized final 
  private final int j; // Blank final 
  private final Poppet p; // Blank final reference 
  // Blank finals MUST be initialized in the constructor: 
  public BlankFinal() { 
    j = 1; // Initialize blank final 
    p = new Poppet(1); // Initialize blank final reference 
  } 
  public BlankFinal(int x) { 
    j = x; // Initialize blank final 
    p = new Poppet(x); // Initialize blank final reference 
  } 
  public static void main(String[] args) { 
    new BlankFinal(); 
    new BlankFinal(47); 
  } 
} // 


final 参数

      java中也许将参数列表中的参数以声明的方式声指明为final。这意味着你无发改变参数所指向的对象。

[java] view plaincopy
class Gizmo { 
  public void spin() {} 

 
public class FinalArguments { 
  void with(final Gizmo g) { 
    //! g = new Gizmo(); // Illegal -- g is final 
  } 
  void without(Gizmo g) { 
    g = new Gizmo(); // OK -- g not final 
    g.spin(); 
  } 
  // void f(final int i) { i++; } // Can't change 
  // You can only read from a final primitive: 
  int g(final int i) { return i + 1; } 
  public static void main(String[] args) { 
    FinalArguments bf = new FinalArguments(); 
    bf.without(null); 
    bf.with(null); 
  } 
} // 
方法f()g()展示了基本类型的参数被指定为final是所出现的结果:你可以读参数,但不能修改参数。这一特性只要用来向匿名内部类传递数据。

final 方法

   使用final方法有两个原因。第一个原因是把方法锁定,以防止任何继承它的类修改它的含义。这是出于设计的考虑:想要确保在继承中使用的方法保持不变,并且不会被覆盖。

   过去建议使用final方法的第二个原因是效率。在java的早期实现中,如果将一个方法指明为fianl,就是同意编译器将针对该方法的所有调用都转为内嵌调用。当编译器发现一个final方法调用命令时,它会根据自己的谨慎判断,跳过插入程序代码这种正常的调用方式而执行方法调用机制(将参数压入栈,跳至方法代码处执行,然后跳回并清理栈中的参数,处理返回值),并且以方法体中的实际代码的副本来代替方法调用。这将消除方法调用的开销。当然,如果一个方法很大,你的程序代码会膨胀,因而可能看不到内嵌所带来的性能上的提高,因为所带来的性能会花费于方法内的时间量而被缩减。

    上面标颜色的地方不太懂。不知道那位看过Java编程思想和知道的高人给解释解释。

    在最进的java版本中,虚拟机(特别是hotspot技术)可以探测到这些情况,并优化去掉这些效率反而降低的额外的内嵌调用,因此不再需要使用final方法来进行优化了。事实上,这种做法正逐渐受到劝阻。在使用java se5/6时,应该让编译器和JVM去处理效率问题,只有在想明确禁止覆盖式,才将方法设置为fianl的。

    final和private关键字

   类中的所有private方法都是隐式的制定为final的。由于你无法访问private方法你也就无法覆盖它。可以对private方法添加final修饰词,但这毫无意义。

[java] view plaincopy
class WithFinals { 
  // Identical to "private" alone: 
  private final void f() { print("WithFinals.f()"); } 
  // Also automatically "final": 
  private void g() { print("WithFinals.g()"); } 

 
class OverridingPrivate extends WithFinals { 
  private final void f() { 
    print("OverridingPrivate.f()"); 
  } 
  private void g() { 
    print("OverridingPrivate.g()"); 
  } 

 
class OverridingPrivate2 extends OverridingPrivate { 
  public final void f() { 
    print("OverridingPrivate2.f()"); 
  } 
  public void g() { 
    print("OverridingPrivate2.g()"); 
  } 


     "覆盖"只有在某方法是基类接口的一部分时才会发生。即,必须将一个对象向上转型为它的基类并条用相同的方法。如果某方法是private的,它就不是基类接口的一部分。它仅是一些隐藏于类中的程序代码,如果一个基类中存在某个private方法,在派生类中以相同的名称创建一个public,protected或包访问权限方法的话,该方法只不过是与基类中的方法有相同的名称而已,并没有覆盖基类方法。由于private方法无法触及且有很好的隐藏性,所以把它看成是因为他所属类的组织结的原因而存在外,其他任何事物都不用考虑。

    final 类

    当将类定义为final时,就表明了你不打算继承该类,而且也不也许别人这样做。换句话说,出于某种考虑,你对该类的设计永不需要做任何变动,或者出于安全的考虑,你不希望他有子类。

[java] view plaincopy
class SmallBrain {} 
 
final class Dinosaur { 
  int i = 7; 
  int j = 1; 
  SmallBrain x = new SmallBrain(); 
  void f() {} 

 
//! class Further extends Dinosaur {} 
// error: Cannot extend final class 'Dinosaur' 
 
public class Jurassic { 
  public static void main(String[] args) { 
    Dinosaur n = new Dinosaur(); 
    n.f(); 
    n.i = 40; 
    n.j++; 
  } 
}  

    请注意,final类的域可以根据个人的意愿选择是或不是final。不论类是否被定义为final,相同的规则同样适用于定义为final的域。然而,由于final是无法继承的,所以被final修饰的类中的方法都隐式的制定为fianl,因为你无法覆盖他们。在fianl类中可以给方法添加final,但这不会产生任何意义。
分享到:
评论

相关推荐

    JAVA编程思想中文版.zip

    《JAVA编程思想》是 Bruce Eckel 的经典著作,中文版为国内Java开发者提供了深入理解Java语言的宝贵资源。这本书全面而深入地介绍了Java编程的核心概念和技术,是学习和提升Java编程技能的重要参考资料。 本书主要...

    java编程思想练习题答案

    《Java编程思想》是Bruce Eckel的经典之作,它深入浅出地介绍了Java语言的核心概念和技术。这本书中的练习题是学习者提升技能的重要途径,而这个压缩包提供的就是这些练习题的解答,以源码的形式保存为.java文件。...

    java编程思想读书笔记

    ### Java编程思想读书笔记 #### 一、Java与C++的区别及内存管理 在学习Java的过程中,我们常常会拿它与C++进行比较。这两门语言虽然有着相似之处,但也有许多不同点。 1. **内存管理:** - C++提供了更为底层的...

    Java编程思想重点笔记(Java开发必看)

    ### Java编程思想重点知识点解析 #### 一、Java中的多态性理解 多态性是面向对象编程的一个核心特性,允许子类对象替代父类对象。在Java中,多态性的实现主要通过以下几种方式: 1. **动态绑定**:在Java中,除了...

    Java编程思想重点笔记(Java开发必看)

    Java 编程思想是每一位Java开发者都需要深入了解的重要领域。这篇笔记涵盖了多个关键知识点,包括多态性、final方法、静态方法、构造函数以及构造过程中的多态行为。 首先,Java中的多态性是面向对象编程的核心特性...

    Java编程思想重点笔记(Java开发必看).pdf

    Java编程思想是每个Java开发者必备的知识体系,涵盖了Java语言的核心概念和高级特性。这篇笔记将聚焦于Java中的多态性,这是面向对象编程的关键概念,同时也涉及到构造函数、静态方法和final关键字的理解。 1. **...

    JAVA编程思想

    《JAVA编程思想》这本书深入探讨了JAVA这门编程语言的核心概念和实践技巧。以下是其中一些关键知识点...以上是《JAVA编程思想》一书中部分核心概念的概览,实际学习过程中应深入理解并结合实践,以掌握Java编程的精髓。

    java编程基础的例子

    通过学习这些基本概念及其实际应用,初学者可以更好地理解和掌握Java编程的核心思想,为进一步深入学习和开发复杂的Java应用程序打下坚实的基础。这些例子将帮助理解如何有效地利用抽象类和接口来设计可扩展和模块化...

    Java编程思想之接口Java中实现多重继承.doc

    ### Java编程思想之接口Java中实现多重继承 #### 一、引言 在面向对象编程中,多重继承一直是讨论的热点之一。C++等语言支持直接的多重继承,而Java则通过接口机制来实现类似的多重继承特性。本文将详细介绍如何在...

    Java编程艺术 PDF

    Herbert Schildt 和James Holmes两位编程大师通过将Java应用于某些有趣、实用的计算机问题和编程任务中,全面展示了Java语言的强大功能、敏捷性、多样性和艺术性。本书各章内容分别涉及到Java精髓、递归下降的表达式...

    Thinking in Java 中文第四版+习题答案

    附录C Java编程规则 附录D 性能 D.1 基本方法 D.2 寻找瓶颈 D.2.1 安插自己的测试代码 D.2.2 JDK性能评测 D.2.3 特殊工具 D.2.4 性能评测的技巧 D.3 提速方法 D.3.1 常规手段 D.3.2 依赖语言的方法 D.3.3 特殊情况 D...

    java编程思想

    ### Java编程思想——抽象类与接口的理解 #### 抽象类的概念 在Java编程中,抽象类是一个重要的概念,它代表了一种特定的设计模式。抽象类通常用来定义一个类族的公共属性和行为,但并不提供一个具体的实现。也...

    java clone

    在Java编程语言中,`clone`是一个非常重要的概念,它涉及到对象复制和对象克隆。本文将深入探讨Java中的`clone`方法,包括其工作原理、使用场景、注意事项以及一些个人实践心得。 首先,让我们理解什么是`clone`。...

    java编程规范

    Java编程规范是软件开发过程中的重要指南,尤其对于大型企业如阿里巴巴来说,一套严谨的编码规范可以确保代码的可读性、可维护性和团队协作效率。以下是对"阿里Java开发规范"的一些关键知识点的详细说明: 1. **...

    面向对象编程(Java).pdf

    封装是面向对象编程的核心思想。它隐藏了对象的内部细节,仅向外界提供必要的接口。在Java中,类是封装的基本单位,通过私有(private)、受保护(protected)、公有(public)等访问修饰符来控制对类成员(字段和...

    JAVA类与对象及数组习题及答案.pdf

    在本段内容中,我们将会讨论Java编程语言的基础知识点,包括类和对象的使用、基本数据类型、数组、方法的定义和调用等。 首先,我们来看看关于Java的类、对象以及数组的概念。Java是一种面向对象的编程语言,它的...

    阿里巴巴Java编程规范.pdf

    面向对象编程(OOP)是Java开发的核心思想之一,阿里巴巴Java编程规范对此有明确的要求: 1. **封装**:保护数据的私有性,通过提供公共的getter和setter方法来访问属性。 2. **继承**:合理使用继承,避免过度继承,...

    JAVA编程常用英文单词汇总.doc

    这是一种编程思想,通过模拟现实世界中的对象及其交互来进行软件设计。 - **OOP**: Object-Oriented Programming,面向对象编程。这是一种编程范式,强调的是将数据和处理这些数据的方法绑定在一起,并且能够通过...

    Java面向对象初步理解

    7. **关键字**:Java中有许多关键字,如`final`用于声明不可变的变量或类,`abstract`用于声明抽象类或方法,`interface`用于定义接口。 8. **访问权限修饰符**:`public`、`private`、`protected`和默认(包访问)...

    Thinking-in-Java-4th-Edition习题答案

    首先,关于“Java编程思想”,这是Java学习的基础。Java是一种面向对象的语言,它的核心理念是封装、继承和多态。封装意味着隐藏实现细节,提供简洁的接口供外部调用;继承则允许子类继承父类的属性和方法,实现代码...

Global site tag (gtag.js) - Google Analytics