java 静态方法_80后程序员,教你学Java核心技术:用户自定义类+静态域静态方法

用户自定义类

在第3章中,已经开始编写了一些简单的类。但是,那些类都只有一个简单的main方法。现在让我们开始学习如何设计复杂应用程序所需要的各种“主力类”(workhorse class)。通常,这些类没有main方法,而有自定义的实例域和实例方法。要想创建一个完整的程序,应该将若干类组合在一起,其中一个类有main方法。

一个Employee类

在Java中,最简单的类定义形式为:

d2765f0bf17e4893b03f9f7278c835f7.png

下面看一个非常简单的Employee类。在编写薪金管理系统时可能会用到。

5edd7cae8b02b5f58081269882961d01.png
6a4dbca174594ba78ad0978d9854acbf.png
cb59844e46c11a5fa506335a4add8ba2.png

这里将这个类的实现细节分成以下几个部分,并分别在稍后的几节中给予介绍。下面先看看例4-2,它展示了一个使用Employee类的程序代码。

在这个程序中,构造了一个Employee数组,并填入了三个雇员对象:

7a42afd6f63514e0688b701ef5c3e2a0.png

接下来,使用雇员类的raiseSalary方法将每个雇员的薪水提高5%:

fee9f73863fb03dce91553faa0f0e21f.png

最后,调用getName方法、getSalary方法和getHireDay方法打印每个雇员的信息:

dd7b6a677729d5d1b93fdce13fbf341d.png

注意,在这个例子程序中包含两个类:一个Employee类;一个带有public访问修饰符的EmployeeTest类。EmployeeTest类包含了main方法,其中使用了前面介绍的指令。

源文件名是EmployeeTest.java,这是因为文件名必须与public类的名字相匹配。在一个源文件中,只能有一个公有类,但可以有任意数目的非公有类。

接下来,当编译这段源代码的时候,编译器将在目录下创建两个类文件:EmployeeTest.class和Employee.class。

将程序中包含main方法的类名字提供给字节码解释器,以便启动这个程序:

java EmployeeTest

字节码解释器开始运行EmployeeTest类的main方法中的代码。在这段代码中,先后构造了三个新Employee对象,并显示它们的状态。

例4-2 EmployeeTest.java

ae4ec625d81a383e4bc024f4e291c1a2.png
acea7cb893e1e15129c5b34ef567bfbf.png
01ceef55c6e143605e328b8266c7233a.png

多个源文件的使用

在例4-2中,一个源文件包含了两个类。许多程序员习惯于将每一个类存入一个单独的源文件中。例如,将Employee类存放在文件Employee.java中,将EmployeeTest类存放在文件EmployeeTest.java中。

如果喜欢这样组织文件,将有两种编译源程序的方法。一种是使用通配符调用Java编译器:

javac Employee*.java

于是,所有与通配符匹配的源文件都将被编译成类文件。或者,仅键入下列命令:

javac EmployeeTest.java

可能会感到惊讶,使用第二种方式,并没有显式地编译Employee.java。然而,当Java编译器发现看到EmployeeTest.java中使用了Employee类时,就会查找名为Employee.class的文件。如果没有找到这个文件,就会自动地搜索Employee.java,然后,对它进行编译。更重要的是:如果Employee.java版本较已有的Employee.class文件版本新,Java编译器就会自动地重新编译这个文件。

解析Employee类

下面对Employee类进行一下剖析。首先从这个类的方法开始。通过查看源代码会发现,这个类有一个构造器和4个方法:

public Employee(String n, double s, int year, int month, int day)

public String getName( )

public double getSalary( )

public Date getHireDay( )

public void raiseSalary(double byPercent)

这个类的所有方法都被标记为public。关键字public意味着任何类的任何方法都可以调用这个方法。(共有4种访问级别,我们将在本章和下一章中加以介绍。)

接下来,需要注意在Employee类的实例中有三个实例域用来存放将要操作的数据。

private String name;

private double salary;

private Date hireDay;

关键字private确保只有Employee类自身的方法能够访问这些实例域,而其他类的方法不能够读写这些域。

最后,请注意,有两个实例域本身就是对象:name域是String类对象,hireDay域是Date类对象。这种情形十分常见:类通常包括类类型的实例域。

从构造器开始

下面先看看Employee类的构造器。

7be647d99db49229b2181e3132c54552.png

已经看到,构造器与类同名。在构造Employee类对象时,构造器被运行,并用于将实例域初

始化为所希望的状态。

例如,当使用下面这条代码创建Employee类实例时:

new Employee("James Bond", 100000, 1950, 1, 1);

将会把实例域设置为:

name = "James Bond";

salary = 100000;

hireDay = January 1, 1950;

构造器与其他的方法有一个重要的不同。构造器总是伴随着new操作符的执行被调用,而不

能对一个已经存在的对象调用构造器来重新设置实例域。例如,

james.Employee("James Bond", 250000, 1950, 1, 1); // ERROR

将产生编译错误。

本章稍后,还会更加详细地介绍有关构造器的内容。现在只需要记住:

• 构造器与类同名。

• 每个类可以有一个以上的构造器。

• 构造器可以有0个、1个或1个以上的参数。

• 构造器没有返回值。

• 构造器总是伴随着new操作符一同使用。

隐式参数与显式参数

方法用于操作对象以及存取它们的实例域。例如,方法

3823b43d9a644e5ac1a8bc92cfc69739.png

将调用这个方法的对象的salary实例域设置为新值。看看下面这个调用

b8fd25c839b1ad84b1ff58deba22bbe8.png

它的结果将number007.salary域的值增加了5%。具体地说,这个调用执行了下述指令:

9de5ade3ae8fa13b9f21aba72307827b.png

raiseSalary方法有两个参数。第一个参数被称为隐式参数,是出现在方法名前的Employee类对象。第二个参数位于方法名后面括号中的数值,这是一个显式参数。

已经看到,显式参数是明显地列在方法声明中的显示参数,例如double byPercent。隐式参数没有出现在方法声明中的参数。

在每一个方法中,关键字this表示隐式参数。如果需要的话,可以编写raiseSalary方法如下:

9038958a18082b0758a4feb22ee6b2f7.png

有些程序员更偏爱这样的风格,因为它将实例域与局部变量明显地区分开来。

封装的优点

最后,再仔细地看一下非常简单的getName方法、getSalary方法和getHireDay方法。

7433df0d6b51177678ae2917d6fbc216.png

这些都是典型的访问器方法。由于它们只返回实例域值,因此又被称为域访问器。

将name、salary和hireDay域标记为public,以此来取代独立的访问器方法会不会更容易些呢?

关键在于name是一个只读域。一旦在构造器中设置完毕,就没有任何一个办法可以对它进行修改,这样来确保name域不会受到外界的干扰。

虽然salary不是只读域,但是它只能用raiseSalary方法修改。特别是一旦这个域值出现了错误,只要调试这个方法就可以了。如果salary域是public的,那么破坏这个域值的捣乱者可能会出没在任何地方。

在有些时候,需要获取或设置实例域的值。因此,应该提供下面三项内容:

• 一个私有的数据域。

• 一个公有的域访问器方法。

• 一个公有的域更改器方法。

这样做要比提供一个简单的公有数据域复杂些,但是却有着下列明显的好处:

1)可以改变内部实现,除了该类的方法之外,不会影响其他代码。

例如,如果将存储名字的域改为:

String firstName;

String lastName;

那么getName方法可以改为返回

firstName + " " + lastName

对于这点改变,程序的其他部分完全不可见。

当然,为了进行新旧数据表示之间的转换,访问器方法和更改器方法有可能需要做许多工

作。但是,这将为我们带来了第二点好处。

2)更改器方法可以执行错误检查,然而直接对域进行赋值将不会做这些处理。

例如,setSalary方法可能会检查薪金是否小于0。

基于类的访问权限

从前面已经知道,方法可以访问所调用对象的私有数据。一个方法可以访问所属类的所有对象的私有数据,这令很多人感到奇怪!例如,让我们考察一下用来比较两个雇员的equals方法。

da0035239db4f233b1edf104c18047ba.png

一个典型的调用是

if (harry.equals(boss)) . . .

这个方法访问harry的私有域,这没什么奇怪的。然而,它还访问boss的私有域。这是合法的,因为boss是Employee类对象,并且Employee类的方法可以访问Employee类的任何一个对象的私有域。

私有方法

当实现一个类的时候,由于公有数据非常危险,所以我们将所有的数据域都设置为私有的。

然而,方法又应该如何设计呢?尽管绝大多数方法都设计为公有的,但在特殊情况下,也可能会设计为私有的。在有些时候,可能希望将一个计算代码划分成若干个独立辅助的方法。通常,这些辅助方法不应该成为公有接口的一部分,这是由于它们往往与当前的实现机制非常紧密,或者需要一个特别的协议以及一个特别的调用次序。这样的方法最好被设计为private的。

在Java中,为了实现一个私有的方法,只需要将关键字public改为private即可。

对于私有方法,如果改用其他方法实现相应的操作,则不必保留原有的方法。如果数据的表达方式发生了变化,那么该方法可能会因此变得难以实现,或者不再需要。然而,只要方法是私有的,类的设计者就可以确信:它不会被外部的其他类操作调用,可以将其删去。如果方法是公有的,就不能将其删去,因为其他的代码很可能调用它。

final实例域

可以将实例域定义为final。构建对象时必须初始化这样的域。也就是说,必须确保在每一个构造器执行之后,这个域的值被设置,并且在后面的操作中,不能够再对它进行修改。例如,可以将Employee类中的name域声明为final,因为在对象构建之后不会再被修改,即没有setName方法。

27bc9a5af6e15015d7b86d1df5eb9c16.png

final修饰符大都应用于基本数据(primitive)类型域,或不可变(immutable)类的域。(如果类中的每个方法都不会改变其对象,这种类就是不可变的类。例如,String类就是一个不可变的类。)对于可变的类,使用final修饰符可能会对读者造成混乱。例如,

2e3f607365135c5b0c6d343e540527a8.png

仅仅意味着存储在hiredate变量中的对象引用在对象构造之后不能被改变。而并不意味着hiredate对象是一个常量。任何方法都可以对hiredate引用的对象调用setTime更改器。

静态域与静态方法

在前面给出的例子程序中,main方法都被标记上static修饰符。现在,我们来讨论一下这个修饰符的含义。

静态域

如果将域定义为static,那么每个类中只有一个这样的域。而每一个对象对于所有的实例域却都有自己的一份拷贝。例如,假定需要给每一个雇员赋予唯一的标识码。这里给Employee类添加一个实例域id和一个静态域nextId:

432e8b9e2894936188849688cd7d38f7.png

现在,每一个雇员对象都有一个自己的id域,但这个类的所有实例将共享一个nextId域。换句话说,如果有1000个Employee类的对象,则有1000个实例域id。但是,只有一个静态域nextId。即使没有一个雇员对象,静态域nextId也存在。它属于类,而不属于任何独立的对象。

下面来实现一个简单的方法:

1689c2ee288ba8181e18a1f6826564ee.png

假定为harry设定雇员标识码:

3892fd640b830f2cdea306e996a91da3.png

那么harry的id域被设置,并且静态域nextId的值加1:

4d231b564f677dc718e90921acd123da.png

常量

静态变量使用得比较少,但静态常量却使用得比较多。例如,在Math类中定义了一个静态常量:

0d9b50c9dbb032b22e3b48c234903d20.png

在程序中,可以通过Math.PI来访问这个常量。

如果关键字static被省略,PI就成了Math类的一个实例域。即需要用Math类的对象来访问PI,并且每一个Math对象都有它自己的一份PI拷贝。

已经使用多次的另一个静态常量是System.out。它在System类中声明:

c102e22175e17c47b5b48d434b565c73.png

前面曾经提到过,由于每个类对象都可以对公有域进行修改,所以将域设计为public并不是一种好的想法。然而,公有常量(即final域)却没问题。因为out被声明为final,所以,不允许再将其他打印流赋给它:

System.out = new PrintStream(. . .); // ERROR--out is final

静态方法

静态方法是不能向对象实施操作的方法。例如,Math类的pow方法就是一个静态方法。表达式Math.pow(x, a)

计算幂xa 。它在运算的时候,不使用任何Math对象。换句话说,没有隐式的参数。

可以认为静态方法是没有this参数的方法。(在一个非静态的方法中,this参数表示该方法的隐式参数。)

因为静态方法不能操作对象,所以不能在静态方法中访问实例域。但是,静态方法可以访问自身类中的静态域。下面是这种静态方法的一个例子:

public static int getNextId( ){return nextId; // returns static field}

可以通过类名调用这个方法:

int n = Employee.getNextId( );

这个方法可以省略关键字static吗?答案是肯定的。但是,需要通过Employee类对象的引用调用这个方法。

Factory方法

静态方法还有一种常见的用途。NumberFormat类使用Factory方法产生不同风格的格式对象。

NumberFormat currencyFormatter = NumberFormat.getCurrencyInstance( );NumberFormat percentFormatter = NumberFormat.getPercentInstance( );double x = 0.1;System.out.println(currencyFormatter.format(x)); // prints $0.10System.out.println(percentFormatter.format(x)); // prints 10%

为什么NumberFormat类不利用构造器来完成这些操作呢?这主要有两个原因:

• 无法命名构造器。构造器的名字必须与类名相同。但是,这里希望将得到的货币实例和百分比实例采用不用的名字。

• 当使用构造器时,无法改变所构造的对象类型。而Factory方法将返回一个DecimalFormat类对象,这是NumberFormat的子类。(有关继承的详细内容请参阅第5章。)

main方法

需要注意,不必使用对象调用静态方法。例如,不需要构造Math类对象就可以调用 Math.pow。

同理,main方法也是一个静态方法。

cff0002be22d432f8f3e96929844678a.png

main方法不对任何对象进行操作。事实上,在启动程序的时候还没有任何一个对象。静态的main方法将执行并创建程序所需要的对象。

例4-3中的程序包含了Employee类的一个简单版本,其中包含一个静态域nextId和一个静态方法getNextId。这里用三个Employee对象填充数组,然后打印雇员信息。最后,打印出下一个可用的员工标识码来作为对静态方法使用的演示。

需要注意,Employee类也有一个静态的main方法用于单元测试。试试运行java Employee和java StaticTest来执行两个main方法。

例4-3 StaticTest.java

ebf4a7c2cc868e2a2f36e942306e97c2.png
18f94a5d951e85f0bdd13f8a7a4d88ce.png
ca9987aab7fa00df4539d5dc9b0a93e2.png

觉得文章不错的话,可以转发此文关注小编,之后持续更新干货文章~~~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/521072.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

容器服务kubernetes弹性伸缩高级用法

前言 近期,阿里云容器服务kubernetes发布了cluster-autoscaler的支持,开发者可以通过页面简单快捷的配置节点的弹性伸缩,支持普通实例、GPU实例以及竞价实例帮助开发者实现架构弹性和运营成本之间的博弈。阿里云容器服务kubernetes的cluster…

主进程中发生javascript错误_你知道 JavaScript 中的错误对象有哪些类型吗?

每当 JavaScript 中发生任何运行时错误时,都会引发Error对象。在许多情况下,我们还可以扩展这些标准Error对象,以创建我们自己的自定义Error对象。属性Error 对象具有2个属性name ——设置或返回错误名称。具体来说,它返回错误所属…

CSDN 1024,“猿”来在等你!

戳蓝字“CSDN云计算”关注我们哦!技术深不可测、薪资难以想象、着装招人吐槽、发量让人惊叹、笑点着实密集、情商令人堪忧......在这个你我他她它通过网络紧密互联、消息实时互通的 21 世纪,人们对身处技术至高点的程序员们仍然有着以上不接地气、呆板保…

BigData:值得了解的十大数据发展趋势

当今,世界无时无刻不在发生着变化。对于技术领域而言,普遍存在的一个巨大变化就是为大数据(Big data)打开了大门,并应用大数据技相关技术来改善各行业的业务并促进经济的发展。目前,大数据的作用已经上升到…

基于FPGA的图像Robert变换实现,包括tb测试文件和MATLAB辅助验证

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 fpga的结果导入到matlab显示: 2.算法运行软件版本 vivado2019.2 matlab2022a 3.部分核心程序 ..................................…

RabbitMQ 镜像集群配置_05

接上一篇:(企业级) RabbitMQ 普通集群配置_04 文章目录一、RabbitMQ 策略实战1. 创建RabbitMQ 策略2. ly-01节点查看创建RabbitMQ 策略3. 登录ly-02管控台查看创建RabbitMQ 策略4. 登录ly-03管控台查看创建RabbitMQ 策略二、添加队列测试2.1. 添加队列2.2. Queues菜…

2018年9月杭州云栖大会Workshop - 基于日志的安全分析实战

基于日志的安全分析实战 背景 越来越多的企业开始重视构建基于日志的安全分析与防护系统。我们会讲述如何使用日志服务从0到1收集海量日志,并从中实时筛选、甄别出可疑操作并快速分析,进一步构建安全大盘与可视化。并通过实战方式,演练覆盖…

不服OceanBase跑分?今天起可到阿里云上一战

蚂蚁金服自研数据库OceanBase登顶TPC-C榜单的消息振奋人心,同时引起国内技术圈的广泛讨论,第一个云上跑出来的数据库分数含金量如何?其他数据库有没有可能更强? 针对这些疑惑,10月24日阿里云以一种最为直接的方式作出…

张勇:新技术是阿里“五新战略”的引擎

9月19日,云栖大会再次在杭州开幕。上千位顶级学者、行业专家,来自64个国家的CEO和CTO齐聚云栖小镇。这已经是这个盛大的年度技术大会的第十年。 阿里巴巴集团CEO张勇在主论坛致辞中表示,“阿里巴巴永远是一家技术驱动,使商业有所…

钉钉视频会议

基于 DingTalk_v5.0.0.74版本制作

图书馆管理系统怎么做_亚马逊erp管理系统有免费的吗?亚马逊erp管理系统怎么免费做...

我做跨境电商也有六年的时间了,在电商这个行业也有自己的一些经验。经验也许没有其他大卖家丰富,但会将我知道的都进行分享。如果有不懂得亚马逊问题可以我(V:772024802)。我这里给大家安排一堂直播课,可以系统的帮你解决做亚马逊…

程序员去交友网站找女友,没想到找到了这个...

1024程序员节,CSDN旗下的码书商店为程序员放个“价”(10月25日截止),全场所有书籍8折,电子产品可以拥有大额优惠券,购买前可加文末客服微信领取优惠券哦卫衣原价249元,1024活动价159元&#xff…

云栖大会 | 马云提出“新制造”战略将影响全球

9月19日,马云在“2018杭州•云栖大会”全面阐释对于新制造的思考。他表示,新制造很快会对全中国乃至全世界的制造业带来席卷性的威胁和席卷性的机会,所有的制造行业所面临的痛苦将远远超出想象,新制造为企业带来新机遇。 马云还特…

流言终结者- Flutter和RN谁才是更好的跨端开发方案?

背景 论坛上很多小伙伴关心为什么闲鱼选择了Flutter而不选择其他跨端方案?站在质量的角度,高性能是一个很重的因素,我们使用Flutter重写了宝贝详情页之后,对比了Flutter和Native详情页的性能表现,结论是中高端机型上F…

【独家揭秘】阿里怎么做双11全链路压测?| CSDN 博文精选

戳蓝字“CSDN云计算”关注我们哦!作者 | 牛兔转自 | CSDN企业博客责编 | 阿秃阿里妹导读:全链路压测是阿里的首创,我们将从工作内容、操作过程、运行总结等多个方向来介绍下阿里内部典型电商活动(如双11准备&#xff…

Centos7 使用Docker 安装Oracle 截图+关键步骤说明

yum install dockerdocker -v systemctl start docker systemctl status dockerdocker拉取镜像 docker pull registry.cn-hangzhou.aliyuncs.com/helowin/oracle_11g#查看拉去的oracle镜像 docker images创建Oracle容器 docker run -d -p 1521:1521 --name oracle_11g registr…

阿里程序员深夜智救31楼跳楼邻居

“我妈跳楼了,快救救她!”。 8月20日凌晨的四点半左右,我被一阵急促的锤门声音吵醒。 听到这句话,没来得及思考,我就冲出了门。 我们住在31楼,出事的地点在邻居家的主卧,当时女主人整个人都悬…

数学建模亚太赛优秀论文_2019亚太地区大学生数学建模竞赛志愿者等级评定结果公布!...

2019亚太地区大学生数学建模竞赛志愿者通过赛氪官网招募,本届竞赛共招到来自193所高校的394名志愿者,根据志愿者邀请队伍数和志愿者任务的完成情况,共评选出344名志愿者,6个优秀组织社团。一路相伴,感谢有你~下面是志愿…

看完秒懂的排序算法

戳蓝字“CSDN云计算”关注我们哦!作者 | 奎哥责编 | 阿秃之前的文章咱们已经聊过了下图是常用排序算法的时间空间复杂度:排序算法这么多,这里先将排序算法做个简单分类:一、可以根据待排序的数据量规模分类:内部排序&…