8.3 static关键字使用的注意细节

最后更新于:2022-04-01 14:11:30

这一节我们看静态在使用过程中应注意的几个细节。 上一节我们在学习成员变量与静态变量的区别时,对于两种变量的调用方式不同时出现了“无法从静态上下文中访问非静态变量”的错误,这个问题我们在以后会经常遇到。那么这是什么问题呢?我们从例子说起,我们把上一节中的Person类中方法printInfo进行静态修饰: ~~~ class Person { String name;//姓名,这是一个成员变量 static String country = "中国";//国籍,这是一个静态变量 public static void printInfo() { System.out.println(name+":"+country); } } class StaticTest { public static void main(String[] args) { Person p = new Person(); p.name = "小强"; p.printInfo(); } } ~~~ 结果: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-05-18_573c41ed38250.jpg) 我们看到了一个和上一节完全相同的错误,而当我们把方法进行修改: ~~~ public static void printInfo() { System.out.println(":"+country); } ~~~ 结果: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-05-18_573c41ed4d5c7.jpg) 我们看到当把printInfo方法中的非静态变量name去掉之后,程序正常运行,因此我们可以总结出static关键字使用中的一个常见问题,即: **第一个细节:静态方法中不能调用非静态成员,只能调用静态成员.**这是由于我们知道静态成员是随着类的加载而存在,是先于对象存在的,而非静态的成员变量是随着对象的创建而存在,所以当我们在静态方法中调用非静态成员时,还没有创建对象,当然就不存在非静态变量.比如我们上面的例子,静态方法printInfo和静态变量country都会随着Person类的加载而加载,而成员变量name只有随着Person对象的创建才会存在,当我们在静态方法中直接调用name时,此时根本就不存在,当然会出错. 当然对于这个细节,也可以说是静态方法的一个局限性.非静态方法既可以访问静态成员,也可以访问非静态成员. 我们再看下一个比较常见的问题. **第二个细节:静态方法中不可以使用this或者super关键字.**原因当然可以用静态的特点解释,super我们还没有学习,这里暂不说明,就拿this关键字来说,我们说this代表对象,那么当我们在静态方法中使用this时,this代表的就是当前类的实例,而静态方法和类是一起存在的,而该类的对象并不存在,所以会出错.我们看一个例子: ~~~ class Person { String name; static String country = "中国"; public static void printInfo() { System.out.println(this.name+":"+country); } } ~~~ 结果: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-05-18_573c41ed5e64e.jpg) 我们很清楚的就看到这是不合法的. **第三个细节:主函数是静态的.**这个我们大家肯定都不陌生,我们从开始学习就在搞main函数,它的正确格式是: ~~~ public static void main(String[] args) ~~~ 那么我们再来看一个例子: ~~~ class StaticTest { int num = 4; public static void main(String[] args) { printNum(); } public void printNum() { System.out.println(num); } } ~~~ 结果: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-05-18_573c41ed731ce.jpg) 我们会发现我们没有用好第一个细节,所以我们进一步修改: ~~~ class StaticTest { int num = 4; public static void main(String[] args) { printNum(); } public static void printNum() { System.out.println(num); } } ~~~ 结果: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-05-18_573c41ed85af2.jpg) 哦,还有一个非静态成员变量,再一次修改: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-05-18_573c41ed9a5e2.jpg) 这个过程真麻烦哦,那么我们是不是要在主函数中调用的成员都是静态的呢?如果真是这样,那么要对象干什么呢?我们知道主函数是程序的入口,对于面向对象语言,那么主函数也就对象访问的入口,是指挥对象工作的,如果没有对象不知道有多么可怕啊. 我们之前学习了两种调用方式,一种是通过类名调用,而另一种方式就是通过对象调用,那么我们为了规避静态的局限性,我们把上面的方法进行非静态化,并且通过对象去调用: ~~~ class StaticTest { int num = 4; public static void main(String[] args) { new StaticTest().printNum(); } public void printNum() { System.out.println(num); } } ~~~ 并且程序正常运行了. **第四个细节:在本类中,非静态成员前面省略的是this,而在静态成员前面省略的却是类名.**比如上面的代码,实际上在Person类中,成员变量name前面省略的this,即this.name,而在静态变量country前面省略的类名Person,即Person.country,这一点我们在以后的使用多注意就可以了.
';