侧边栏壁纸
博主头像
极客日记 博主等级

行动起来,活在当下

  • 累计撰写 93 篇文章
  • 累计创建 17 个标签
  • 累计收到 1 条评论

目 录CONTENT

文章目录

Java 基础 - 作用域

Jack.Jia
2022-09-13 / 0 评论 / 0 点赞 / 5 阅读 / 0 字

作用域

在 Java 中,我们经常看到 publicprotectedprivate 这些修饰符。在 Java 中,这些修饰符可以用来限定访问作用域。

public

定义为 publicclassinterface 可以被其他任何类访问:

package abc;

public class Hello {
    public void hi() {
    }
}

上面的 Hellopublic,因此,可以被其他包的类访问:

package xyz;

class Main {
    void foo() {
        // Main可以访问Hello
        Hello h = new Hello();
    }
}

定义为 publicfieldmethod 可以被其他类访问,前提是首先有访问 class 的权限:

package abc;

public class Hello {
    public void hi() {
    }
}

上面的 hi() 方法是 public,可以被其他类调用,前提是首先要能访问 Hello 类:

package xyz;

class Main {
    void foo() {
        Hello h = new Hello();
        h.hi();
    }
}

private

定义为 privatefieldmethod 无法被其他类访问:

package abc;

public class Hello {
    // 不能被其他类调用:
    private void hi() {
    }

    public void hello() {
        this.hi();
    }
}

实际上,确切地说,private 访问权限被限定在 class 的内部,而且与方法声明顺序无关。推荐把 private 方法放到后面,因为 public 方法定义了类对外提供的功能,阅读代码的时候,应该先关注 public 方法:

package abc;

public class Hello {
    public void hello() {
        this.hi();
    }

    private void hi() {
    }
}

由于 Java 支持嵌套类,如果一个类内部还定义了嵌套类,那么,嵌套类拥有访问 private 的权限:

// private
public class Main {
    public static void main(String[] args) {
        Inner i = new Inner();
        i.hi();
    }

    // private方法:
    private static void hello() {
        System.out.println("private hello!");
    }

    // 静态内部类:
    static class Inner {
        public void hi() {
            Main.hello();
        }
    }
}

定义在一个 class 内部的 class 称为嵌套类(nested class),Java 支持好几种嵌套类。

protected

protected 作用于继承关系。定义为 protected 的字段和方法可以被子类访问,以及子类的子类:

package abc;

public class Hello {
    // protected方法:
    protected void hi() {
    }
}

上面的 protected 方法可以被继承的类访问:

package xyz;

class Main extends Hello {
    void foo() {
        // 可以访问protected方法:
        hi();
    }
}

package

最后,包作用域是指一个类允许访问同一个 package 的没有 publicprivate 修饰的 class,以及没有 publicprotectedprivate 修饰的字段和方法。

package abc;
// package权限的类:
class Hello {
    // package权限的方法:
    void hi() {
    }
}

只要在同一个包,就可以访问 package 权限的 classfieldmethod

package abc;

class Main {
    void foo() {
        // 可以访问package权限的类:
        Hello h = new Hello();
        // 可以调用package权限的方法:
        h.hi();
    }
}

注意,包名必须完全一致,包没有父子关系,com.apachecom.apache.abc 是不同的包。

局部变量

在方法内部定义的变量称为局部变量,局部变量作用域从变量声明处开始到对应的块结束。方法参数也是局部变量。

package abc;

public class Hello {
    void hi(String name) { // ①
        String s = name.toLowerCase(); // ②
        int len = s.length(); // ③
        if (len < 10) { // ④
            int p = 10 - len; // ⑤
            for (int i=0; i<10; i++) { // ⑥
                System.out.println(); // ⑦
            } // ⑧
        } // ⑨
    } // ⑩
}

我们观察上面的 hi() 方法代码:

  1. 方法参数 name 是局部变量,它的作用域是整个方法,即①~⑩;
  2. 变量 s 的作用域是定义处到方法结束,即②~⑩;
  3. 变量 len 的作用域是定义处到方法结束,即③~⑩;
  4. 变量 p 的作用域是定义处到 if 块结束,即⑤~⑨;
  5. 变量 i 的作用域是 for 循环,即⑥~⑧。

使用局部变量时,应该尽可能把局部变量的作用域缩小,尽可能延后声明局部变量。

final

Java 还提供了一个 final 修饰符。final 与访问权限不冲突,它有很多作用。

final 修饰 class 可以阻止被继承:

package abc;

// 无法被继承:
public final class Hello {
    private int n = 0;
    protected void hi(int t) {
        long i = t;
    }
}

final 修饰 method 可以阻止被子类覆写:

package abc;

public class Hello {
    // 无法被覆写:
    protected final void hi() {
    }
}

final 修饰 field 可以阻止被重新赋值:

package abc;

public class Hello {
    private final int n = 0;
    protected void hi() {
        this.n = 1; // error!
    }
}

final 修饰局部变量可以阻止被重新赋值:

package abc;

public class Hello {
    protected void hi(final int t) {
        t = 1; // error!
    }
}

最佳实践

  1. 如果不确定是否需要 public,就不声明为 public,即尽可能少地暴露对外的字段和方法。
  2. 把方法定义为 package 权限有助于测试,因为测试类和被测试类只要位于同一个 package,测试代码就可以访问被测试类的 package 权限方法。
  3. 一个.java 文件只能包含一个 public 类,但可以包含多个非 public 类。如果有 public 类,文件名必须和 public 类的名字相同。

小结

  1. Java 内建的访问权限包括 publicprotectedprivatepackage 权限;
  2. Java 在方法内部定义的变量是局部变量,局部变量的作用域从变量声明开始,到一个块结束;
  3. final 修饰符不是访问权限,它可以修饰 classfieldmethod
  4. 一个.java 文件只能包含一个 public 类,但可以包含多个非 public 类。
0

评论区