java——方法重载与重写、构造方法、this关键字、static关键字、strictfp关键字、类的导入

  Java SE5新增加@Override注解,它并不是关键字,但是可以把它当作关键字使用。当你想要覆写(重写)某个方法时,可以选择添加这个注解,在你不留心重载而并非覆写了该方法时,编译器就会生成一条错误信息。

  方法重载:重载的时候,方法名要一样,但是参数类型或个数不一样,返回值类型可以相同也可以不相同。无法以返回型别作为重载函数的区分标准。

  方法覆写(重写):

    若子类中的方法与父类中的某一方法具有相同的方法名、返回类型和参数表,则新方法将覆盖原有的方法。

    方法重载是编译时处理的,而方法覆盖是在运行时处理的。  

    在子类的某个方法覆盖父类的方法时,这个方法的访问权限可以被扩大,但是不能缩小,否则报错。

    父类的静态方法不能被子类的非静态方法覆盖,同样父类的非静态方法也不能被子类的静态方法覆盖。

    子类的方法覆盖父类的方法时,子类不能抛出父类没有抛出的异常。

 

构造方法:类中的一种方法,类似Python的__init__,但是有一些要求:

  1.方法名要和类名相同。

  2.在方法名的前面没有返回值类型的声明。

  3.不能用return返回一个值,但是可以单独写return作为方法的结束。

  4.java中每个类都至少有一个构造方法,如果没有定义构造方法,系统也会自动创建一个空的构造方法。

  5.构造方法一般用public来修饰。

  6.在初始化含参的构造方法时必须检查参数的合法性,这样可以提高代码的健壮性。

拷贝构造方法:

  用该类的一个对象来初始化这个类:

//Example:
//1.Clock类:
public class Clock {
 private int hour;
 private int minute;
 private int second;
 
 public Clock(){
  setTime(0,0,0);
 }
 
 public Clock(int h,int m,int s){
  setTime(h,m,s);
 }
 
 /* 拷贝构造函数 */
 public Clock(Clock clock){
  this.hour=clock.hour;
  this.minute=clock.minute;
  this.second=clock.second;
 }
 
 public int getHour() {
  return hour;
 }
 public int getMinute() {
  return minute;
 }
 public int getSecond() {
  return second;
 }
 public void setTime(int h,int m,int s){
  if(h>=0 && h<24)
   this.hour=h;
  else
   this.hour=0;
  
  if(m>=0 && m<60)
   this.minute=m;
  else
   this.minute=0;
  
  if(s>=0 && s<60)
   this.second=s;
  else
   this.second=0;
 }
 
 public void printTime(){
  if(this.hour<10)
   System.out.print("0");
  System.out.print(this.hour+":");
  
  if(this.minute<10)
   System.out.print("0");
  System.out.print(this.minute+":");
  
  if(this.second<10)
   System.out.print("0");
  System.out.println(this.second);
 }
}
//2.main函数:
public class Program {
 public static void main(String[] args) {
  Clock c1=new Clock(6,43,23);
  Clock c2=new Clock(c1);//调用拷贝构造函数
  c1.printTime();
  c2.printTime();
 }
}
//3.运行结果:
06:43:23
06:43:23

 用 this 调用构造方法

  调用这个类的另一个构造方法

public Card(String i, String n, char s, double m, double initial){
    this(i, n, s, m);
    money += initial;

 构造方法继承

  java子类可以继承父类所拥有的成员变量和成员方法,但父类的构造方法不能继承。(正常子类可以直接用父类的方法和变量,但是构造方法必须使用super()调用)

  要在子类中的构方法中使用super();来调用父类的构造方法,该方法必须放置于子类构造方法的第一句:

public son(){
    super(); // 调用父类无参构造方法
    a = 1; 
}

  如果子类的构造方法没有调用父类的构造方法,那系统会自动调用无参父类的构造方法。此时若父类没有无参构造函数会在编译时报错。

  一个构造方法中不能同时包括this调用和super调用,如果想同时包含两个调用,可以在this调用的构造方法中首先调用super()。

this关键字:

  1.通过this关键字可以明确的去访问一个类的成员变量,解决与局部变量名称冲突的问题。

  2.通过this关键字调用成员方法。

  3.构造方法是在实例化对象时被java虚拟机自动调用的,在程序中不能像调用其他方法一样调用构造方法,但可以在一个构造方法中使用“this([par1, par2...])”的形式来调用其他的构造方法,但是在成员方法中不可行,也不能在一个类中的两个构造方法中使用this相互调用。

 

static关键字:

  静态变量

    在类加载之后创建和初始化,而不是对象创建后

    在一个java类中,可以使用static关键字来修饰成员变量,该变量被称为静态变量,静态变量被所有实例共享。要注意,static不能用来修饰局部变量和内部类中的变量。

    (为什么static不能修饰局部变量?答:局部变量之所以叫局部变量,其作用域也只限于声明它的方法体内。在方法被调用时,这些局部变量获得内存空间,到方法执行结束时,他们所占据的内存空间就被释放。而static定义的变量叫做静态变量,静态变量在类加载的时候创建,随着类的消失而消失。二者生命周期不同。)

    (为什么static不能修饰内部类中的变量?答:静态变量是要占用内存的,在编译时只要是定义为静态变量了,系统就会自动分配内存给他,而内部类是在宿主类编译完编译的,也就是说,必须有宿主类存在后才能有内部类,这也就和编译时就为静态变量分配内存产生了冲突,因为系统执行:运行宿主类->静态变量内存分配->内部类,而此时内部类的静态变量先于内部类生成,这显然是不可能的,所以不能定义静态变量)

  静态方法

    跟Python中的静态方法@staticmethod类似,这种方法在不创建对象的情况下也能被调用:类名.方法。

    实例化后的对象同样可以调用静态方法:对象名.方法。

    静态方法在访问本类的成员时,只允许访问静态成员(即静态成员变量和静态方法),而不允许访问实例成员变量和实例方法;实例方法则无此限制。

    静态方法的覆盖符合非静态方法覆盖的一些规则,二者区别在于java虚拟机将静态方法绑定在类上,而将非静态方法绑定于对象实例。

  静态代码块

    static {...}

    当类被加载时静态代码块会被执行,由于类只加载一次,静态代码块也只执行一次,因此通常会使用静态代码块来对类的成员进行初始化。 

  静态类:

   java允许我们在一个类里面定义静态类。比如内部类(nested class)。把nested class封闭起来的类叫外部类。

  在java中,我们不能用static修饰顶级类(top level class)。只有内部类可以为static。 

strictfp关键字:

  在main方法中使用strictfp将使用严格的浮点计算,在类中使用strictfp关键字则这个类中的所有方法都要使用严格的浮点计算。

类的导入:

  如果同时要使用java.sql.Date和java.util.Date,需要在类的前面指出完整的类名:

    java.util.Date uDate = new java.util.Date();

    java.sql.Date sDate = new java.sql.Date();

  main函数为什么一定要用static修饰?

    main函数是程序的入口,而存在于类中的main函数如果没有static修饰,只是一个普通的实例方法,只有通过类对象才能调用。加上static之后,在类对象定义之前,该方法就已经构建完成,使得该方法独立于类对象之外,可以成为所有程序的入口。

静态导入:  

//导入System的静态方法和静态域
import static java.lang.System.*;
public class exp {
    public static void main(String args[]) {
        // 不需要加类的前缀
        out.println("No main()");        
    }
}

 

 

  

posted @ 2018-09-02 10:44  高圈圈  阅读(556)  评论(0编辑  收藏  举报