理解使用static import 机制(转)

简介: J2SE 1.5里引入了“Static Import”机制,借助这一机制,可以用略掉所在的类或接口名的方式,来使用静态成员。本文介绍这一机制的使用方法,以及使用过程中的注意事项。     在Java程序中,是不允许定义独立的函数和常量(当然,准确的说,只是被final修饰、只能赋值一次的变量)的。
J2SE 1.5里引入了“Static Import”机制,借助这一机制,可以用略掉所在的类或接口名的方式,来使用静态成员。本文介绍这一机制的使用方法,以及使用过程中的注意事项。

    在Java程序中,是不允许定义独立的函数和常量(当然,准确的说,只是被final修饰、只能赋值一次的变量)的。即使从它们本身的功能来看,完全不需要依附于什么东西,也要找个类或接口作为挂靠单位才行(在类里可以挂靠各种成员,而接口里则只能挂靠常量)。

    挂靠的方法,是把它们加上static修饰符,定义为这个类或接口的静态成员。这方面的典型例子是java.lang.Math类——包含了大量的sin、cos这样的“函数”和PI、E这样的“常量”。

传统上,在访问这些挂靠了的函数、变量和常量的时候,需要在前面加上它们挂靠单位的名称。如果只是偶尔访问这些东西一下,这样的写法可以工作得很好;但是如果要频繁访问这些成员的话,这样的写法就显得比较罗嗦了。

 

J2SE 1.5里引入了“Static Import”机制,借助这一机制,可以用略掉所在的类或接口名的方式,来使用静态成员。

1.精确导入的方式

    精确的导入一个静态成员的方法,是在源文件的开头部分(任何类或接口的定义之前),加上类似这样的声明:
import static 包名.类或接口名.静态成员名;

    注意尽管这个机制的名目是叫做“Static Import”,但是在这里的次序却是正好相反的“import static”。一经导入之后,在整个源文件的范围内,就可以直接用这个成员的名字来访问它了。

 

清单1:用精确导入的方式,导入sin和PI

//精确的导入Math.sin和Math.PI
import static java.lang.Math.sin;
import static java.lang.Math.PI;

public class StaticImportSampleA {
    public static void main(String[] args) {
        System.out.println(sin(PI/2));//输出“1.0”
    }
}

2.按需导入的方式

Static Import机制也支持一种不必逐一指出静态成员名称的导入方式。这时,要采用这样的语法:

import static 包名.类或接口名.*;

注意这种方式只是指出遇到来历不明的成员时,可以到这个类或接口里来查找,并不是把这个类或接口里的所有静态成员全部导入。

清单2:用按需导入的方式,导入sin和PI

//声明遇到来历不明的成员时到java.lang.Math中去寻找
import static java.lang.Math.*;

public class StaticImportSampleB {
    public static void main(String[] args) {
        System.out.println(sin(PI/2));//输出“1.0”
    }
}

3.可以导入的种种东西

使用import static语句,可以导入一个类里的一切被static修饰的东西,包括变量、常量、方法和内类。

清单3:变量、常量、方法和内部类都可以导入

package com.example.p3;

public class StaticImportee {
    public static int one = 1;
    public static final int TWO = 2;
    public static int three() {
        return 3;
    }
    public static class Four {
        public int value() {
            return 4;
        }
    }
}
package com.example.p3;
import static com.example.p3.StaticImportee.*;

public class StaticImporter {
    public static void main(String[] args) {
        System.out.println(one);
        System.out.println(TWO);
        System.out.println(three());
        System.out.println(new Four());
    }
}

4.导入之间的冲突问题

不同的类(接口)可以包括名称相同的静态成员。因此,在进行Static Import的时候,可能会出现“两个语句导入同名的静态成员”的情况。

在这种时候,J2SE 1.5会这样来加以处理:

  • 如果两个语句都是精确导入的形式,或者都是按需导入的形式,那么会造成编译错误。
  • 如果一个语句采用精确导入的形式,一个采用按需导入的形式,那么采用精确导入的形式的一个有效。

注意,如果两个同名的静态成员一个是属性,而另一个是方法,那么因为使用时的写法有差异,不会造成任何的冲突。

清单4:采用精确导入的形式的一个有效

package com.example.p4;
import static com.example.p4.Importee1.name;
import static com.example.p4.Importee1.*;
import static com.example.p4.Importee2.*;
import static com.example.p4.Importee2.pass;

public class Importer {
    public static void main(String[] args) {
        System.out.println(name);//输出“Name1”
        System.out.println(pass);//输出“Pass2”
    }
}
package com.example.p4;

public class Importee1 {
    public static String name = "Name1";
    public static String pass = "Pass1";
}
package com.example.p4;

public class Importee2 {
    public static String name = "Name2";
    public static String pass = "Pass2";
}

5.本地和外来的竞争

有时候,导入的东西还可能和本地的东西相冲突,这种情况下的处理规则,是“本地优先”。

清单5:本地的优先于外来的

package com.example.p5;
import static com.example.Zero.*;

public class One {
    public static int flag = 1;
    public static void main(String[] args) {
        System.out.println(flag);//输出“1”
    }
}
package com.example.p5;

public class Zero {
    public static int flag = 0;
}

6.Static Import的负面影响

在编译期间,所有因Static Import的存在而简化了的名字,都会被编译器打回原型。因此在性能方面,Static Import没有任何影响。但是名字简化却可能造成一些维护方面的问题。

去掉静态成员前面的类型名,固然有助于在频繁调用时显得简洁,但是同时也失去了关于“这个东西在哪里定义”的提示信息,增加了阅读理解的麻烦。如果导入的来源很著名(比如java.lang.Math),或者来源的总数比较少,这个问题并不严重;但是在不属于这两种的情况下,这就不是基本可以忽略的问题了。

7.归纳总结

      借助J2SE 1.5里提供的Static Import机制,可以用一种更简单的方式,来访问类和接口的静态成员。不过,使用这一机制并不是没有代价的,在使用不当的时候可能给维护工作带来一定的困扰。因此,在具体使用之前,还要作一些两方面的权衡。

 
http://sunnylocus.iteye.com/blog/806185
 
相关文章
|
3月前
|
Java
java基础(4)public class 和class的区别及注意事项
本文讲解了Java中`public class`与`class`的区别和注意事项。一个Java源文件中只能有一个`public class`,并且`public class`的类名必须与文件名相同。此外,可以有多个非`public`类。每个类都可以包含一个`main`方法,作为程序的入口点。文章还强调了编译Java文件生成`.class`文件的过程,以及如何使用`java`命令运行编译后的类。
70 3
java基础(4)public class 和class的区别及注意事项
|
7月前
|
Java
java-基础-Interface、abstract类、Static class 、non static class的区别
【4月更文挑战第5天】Java中的接口、抽象类、静态类和非静态类各具特色:接口仅含抽象方法和常量,用于定义行为规范;抽象类可包含抽象和非抽象方法,提供部分实现,支持多继承;静态内部类不依赖外部类实例,可独立存在,访问外部类的静态成员;非静态内部类持有关联外部类引用,能访问其所有成员。这些机制根据设计需求和场景选择使用。
58 6
|
存储 Java 编译器
Java-关于main函数的修饰符(为什么要带public和static)
关于Java主函数的修饰符(为什么要带public和static) public修饰符 public是函数的权限,决定了该函数是否可以被外部的函数调用
JAVA中static、final、static final的区别
JAVA中static、final、static final的区别
125 0
|
设计模式 Java 编译器
java static作用
Java中的static关键字是一个非常重要的概念,它可以用来修饰类的成员变量、方法和代码块。在本文中,我们将深入探讨static的作用,以及如何正确地使用它。
|
Java 测试技术
为什么说java中能不用static就不用static?
为什么说java中能不用static就不用static?
Java基础关于含有和不含static的方法的访问
Java基础关于含有和不含static的方法的访问
|
设计模式 Java
java面对对象(下.1)(java关键字理解:static final main)
当我们编写一个类时,其实就是在描述其对象的属性和行为,而并没有产生实质上 的对象,只有通过 new 关键字才会产生出对象,这时系统才会分配内存空间给对象, 其方法才可以供外部调用。
java面对对象(下.1)(java关键字理解:static final main)
|
Java C++
Java | Static 与 final
Java | Static 与 final
java中是否可以重写一个private或者static方法?
java中是否可以重写一个private或者static方法?
1253 0