零基础学Java(12)静态字段与静态方法
静态字段与静态方法
之前我们都定义的main
方法都被标记了static
修饰符,那到底是什么意思?下面我们来看看
静态字段
如果将一个字段定义为static
,每个类只有一个这样的字段。而对于非静态的实例字段,每个对象都有自己的一个副本。例如,假设需要给每一个员工赋予唯一的标识码。这里给Employee
类添加一个实例字段id和一个静态字段nextId
:
class Employee {
// 定义静态字段nextId
private static int nextId = 1;
private int id;
}
现在,每一个Employee
对象都有一个自己的id字段,但这个类的所有实例将共享一个nextId
字段。换句话说,如果有1000个Employee
类对象,则有1000个实例字段id,分别对应每一个对象。但是,只有一个静态字段nextId
。即使没有Employee
对象,静态字段nextId
也存在。它属与类,而不属于任何单个的对象。
下面实现一个简单的方法:
public void setId() {
id = nextId;
nextId++;
}
假定为harry设置员工标识码:
harry.setId();
harry的id字段被设置为静态字段nextId
当前的值,并且静态字段nextId
的值加1:
harry.id = Employee.nextId;
Employee.nextId++
静态常量
静态变量使用的比较少,但静态常量却很常用。例如,在Math类中定义一个静态常量:
public class Math {
...
public static final double PI = 3.14159265358979323846;
...
}
在程序中,可以用Math.PI
来访问这个常量。
如果省略关键字static
,PI
就变成了Math
类的一个实例字段。也就是说,需要通过Math
类的一个对象来访问PI
,并且每一个Math
对象都有它自己的一个PI
副本。
你已经多次使用的另一个静态常量是System.out
。它在System
类中声明如下:
public class System {
...
public static final PrintStream out = ...;
...
}
前面曾经多次提到过,由于每个类对象都可以修改公共字段,所以,最好不要有公共字段。然而,公共常量(即final字段)却没问题。因为out
被声明为final
,所以,不允许再将它重新赋值为另一个打印流:
System.out = new PrintStream(...); // ERROR -- out is final
静态方法
静态方法是不在对象上执行的方法。例如,Math
类的pow
方法就是一个静态方法。表达式Math.pow(x, a)
会计算幂x的a次方。在完成运算时,它并不使用任何Math
对象。换句话说,它没有隐式参数。
可以认为静态方法是没有this
参数的方法(在一个非静态的方法中,this
参数指示这个方法的隐式参数)
Employee
类的静态方法不能访问id实例字段,因为它不能在对象上执行操作。但是,静态方法可以访问静态字段。下面是这样一个静态方法的示例:
public static int getNextId() {
return nextId; // returns static field
}
可以提供类名来调用这个方法:
int n = Employee.getNextId();
这个方法可以省略关键字static
吗?答案是肯定的。但是,这样一来,你就需要通过Employee
类对象的引用来调用这个方法。
注意:可以使用对象调用静态方法,这是合法的。例如,如果harry是一个Employee
对象,可以用harry.getNextId()
代替Employee.getNextId()
。不过,这种写法很容易造成混淆,其原因是getNextId
方法计算的结果与harry
毫无关系。我们建议使用类名而不是对象来调用静态方法。
在下面两种情况下可以使用静态方法:
- 方法不需要访问对象状态,因为它需要的所有参数都通过显式参数提供(例如:Math.pow)
- 方法只需要访问类的静态字段(例如:Employee.getNextId)。
工厂方法
静态方法还有另外一种常见的用途。类似LocalDate
和NumberFormat
的类使用静态工厂方法来构造对象。我们之前使用过工厂方法LocalDate.now
和LocalDate.of
。NumberFormat
类如下生成不同风格的格式化对象:
NumberFormat currencyFormatter = NumberFormat.getCurrencyInstance();
NumberFormat percentFormatter = NumberFormat.getPercentInstance();
double x = 0.1;
System.out.println(currencyFormatter.format(x)); // prints $0.10
System.out.printIn(percentFormatter.format(x)); // prints 10%
为什么NumberFormat
类不利用构造器完成这些操作呢?这主要有两个原因:
- 无法命名构造器。构造器的名字必须与类名相同。但是,这里希望有两个不同的名字,分别得到货币实例和百分比实例。
- 使用构造器时,无法改变所构造对象的类型。而工厂方法实际上将返回
DecimalFormat
类的对象,这是NumberFormat
的一个子类。
main方法
需要注意,可以调用静态方法而不需要任何对象。例如,不需要构造Math
类的任何对象就可以调用Math.pow
同理,main
方法也是一个静态方法。
public class Application {
public static void main(String[] args) {
// construct objects here
...
}
}
main
方法不对任何对象进行操作。事实上,在启动程序时还没有任何对象。静态的main
方法将执行并构造程序所需要的对象。
提示:每一个类可以由一个main
方法。这是常用于对类进行单元测试的一个技巧
例子
接下来我们创建Employee
类,其中有一个静态字段nextId
和一个静态方法getNextId
。这里将三个Employee
对象填入一个数组,然后打印员工信息。最后,打印出下一个可用的员工标识码来展示静态方法。
// 文件StaticTest.java
public class StaticTest {
public static void main(String[] args) {
System.out.println("1111");
Employee[] staff = new Employee[3];
staff[0] = new Employee("Tom", 40000);
staff[1] = new Employee("Dick", 60000);
staff[2] = new Employee("Harry", 65000);
for (Employee e: staff) {
e.setId();
System.out.println("name=" + e.getName() + ", id=" + e.getId() + ", salary=" + e.getSalary());
}
int n = Employee.getNextId();
System.out.println("Next available id=" + n);
}
}
class Employee {
// 静态字段nextId
private static int nextId = 1;
private String name;
private double salary;
private int id;
// 构造器
public Employee(String n, double s) {
name = n;
salary = s;
id = 0;
}
public String getName() {
return name;
}
public double getSalary() {
return salary;
}
public int getId() {
return id;
}
public void setId() {
id = nextId;
nextId++;
}
// 设置静态方法,静态方法中能调用静态字段
public static int getNextId() {
return nextId;
}
public static void main(String[] args) {
Employee e = new Employee("Harry", 5000);
System.out.println(e.getName() + " " + e.getSalary());
}
}
这里我们定义了2个类StaticTest
和Employee
,这两个类分别有一个main
函数
执行命令以下命令
java Employee
结果如下:
Harry 5000.0
当我们执行
java StaticTest
结果如下:
name=Tom, id=1, salary=40000.0
name=Dick, id=2, salary=60000.0
name=Harry, id=3, salary=65000.0
Next available id=4
两者会分别执行各自的main
方法
零基础学Java(12)静态字段与静态方法的更多相关文章
- 零基础学Java第五节(面向对象一)
本篇文章是<零基础学Java>专栏的第五篇文章,文章采用通俗易懂的文字.图示及代码实战,从零基础开始带大家走上高薪之路! 本文章首发于公众号[编程攻略] 类与对象 在哲学体系中,可以分为主 ...
- 零基础学Java第一节(语法格式、数据类型)
本篇文章是<零基础学Java>专栏的第一篇文章,从本篇文章开始,将会连更本专栏,带领大家将Java基础知识彻底学懂,文章采用通俗易懂的文字.图示及代码实战,从零基础开始带大家走上高薪之路! ...
- 零基础学Java,PayPal技术专家手把手带你入门
在最权威的 TIOBE 编程语言排名榜单上,Java 常年稳居第一,可以说是世界上应用最为广泛的一门语言. 同时,在微服务.云计算.大数据.Android App 开发等领域,Java 也是当之无愧的 ...
- 零基础学Java第四节(字符串相关类)
本篇文章是<零基础学Java>专栏的第四篇文章,文章采用通俗易懂的文字.图示及代码实战,从零基础开始带大家走上高薪之路! String 本文章首发于公众号[编程攻略] 在Java中,我们经 ...
- 零基础学Java第三节(基本输入输出)
本篇文章是<零基础学Java>专栏的第三篇文章,文章采用通俗易懂的文字.图示及代码实战,从零基础开始带大家走上高薪之路! 本文章首发于公众号[编程攻略] Java程序的命令行参数 我们可以 ...
- 零基础学Java第二节(运算符、输入、选择流程控制)
本篇文章是<零基础学Java>专栏的第二篇文章,文章采用通俗易懂的文字.图示及代码实战,从零基础开始带大家走上高薪之路! 第一章 运算符 1.1 算术运算符的概述和用法 运算符 对常量和变 ...
- 零基础学Java第六节(面向对象二)
本篇文章是<零基础学Java>专栏的第六篇文章,文章采用通俗易懂的文字.图示及代码实战,从零基础开始带大家走上高薪之路! 本文章首发于公众号[编程攻略] 继承 创建一个Person类 我们 ...
- 零基础学Java(1)初识Java程序
前言 就国内来说,Java毫无疑问是后端语言中的No.1没有之一,所以今天我们也来0基础学习Java!!! Java的好处(针对测试工程师) 面试加分->涨薪 大多数公司服务端用的都是Java, ...
- 零基础学Java(14)对象构造
对象构造 之前学习了编写简单的构造器,可以定义对象的初始状态.但是,由于对象构造非常重要,所以Java提供了多种编写构造器的机制. 重载 有些类有多个构造器.例如,可以如下构造一个空的StringBu ...
随机推荐
- KLOOK客路旅行基于Apache Hudi的数据湖实践
1. 业务背景介绍 客路旅行(KLOOK)是一家专注于境外目的地旅游资源整合的在线旅行平台,提供景点门票.一日游.特色体验.当地交通与美食预订服务.覆盖全球100个国家及地区,支持12种语言和41种货 ...
- 697. Degree of an Array - LeetCode
697. Degree of an Array - LeetCode Question 697. Degree of an Array - LeetCode Solution 理解两个概念: 数组的度 ...
- 11┃音视频直播系统之 WebRTC 进行文本聊天并实时传输文件
一.RTCDataChannel WebRTC 不但可以让你进行音视频通话,而且还可以用它传输普通的二进制数据,比如说可以利用它实现文本聊天.文件的传输等 WebRTC 的数据通道(RTCDataCh ...
- 文件操作(Java)
学习内容:文件操作 1.输入流:InputStream类是字节输入流的抽象类,常用的一些方法有: raed()方法:从输入流中读取数据的下一个字节 reset()方法:将输入指针返回到当 ...
- 上半年最中意的 GitHub 更新「GitHub 热点速览 v.22.21」
2022 年的热点速览加入 GitHub 产品动态之后,这次的图片显示是最得我意的,因为可以提升图片查看人的看片体验.而 GitHub 社区方面,GitHub 官方发布了部分开源项目的新版本.与此同时 ...
- Python数据分析--Numpy常用函数介绍(4)--Numpy中的线性关系和数据修剪压缩
摘要:总结股票均线计算原理--线性关系,也是以后大数据处理的基础之一,NumPy的 linalg 包是专门用于线性代数计算的.作一个假设,就是一个价格可以根据N个之前的价格利用线性模型计算得出. 前一 ...
- PostMan 快快走开, ApiFox 来了, ApiFox 强大的Api调用工具
简介 为什么要用ApiFox呢, 一般现在程序员开发测试, 一般都是PostMan, PostWoman等Api调用工具, 我之前也是一直在用, 但是今天我发现了一款相比于Postman更加好用的工具 ...
- java基础内存分配介绍
java内存分配介绍 栈 堆 方法区 本地方法栈 寄存器 方法区:字节码文件加载时进入的内存. 栈内存:方法运行时所进入的内存,变量也在这里. 堆内存:new出来的东西在这块内存中开辟空间并且 ...
- hibernate-validator的基本使用
validator是用来校验参数使用! 一般来说校验参数的工作可以放在前端去执行,但是假如有人不经过前端直接调用后端的接口呢?很可能就出现非法数据而导致一些问题,所有服务端也要做数据的校验. 前端校验 ...
- Redis配置登录密码
更新记录 2022年6月14日 发布. 打开配置文件 vi /etc/redis/redis.conf 搜索来找到下面这行注释 #requirepass foobared 取消注释,把 foobare ...