面向对象程序设计的一个一个重要特点是:封装性。
这里的封装性有两方面含义:一是将有关的数据和操作代码封装在一个对象中形成一个基本单位,各个对象之间相互独立互不干扰,二是将对象中某些部分对外隐蔽,即隐蔽其内部细节只留下少量接口以便与外界联系,接收外界的信息。这种隐蔽的做法称为信息隐蔽。
程序设计者的任务包括两个方面:一是设计所需要的各种类和 对象,即决定把哪些数据和操作封装在一起,二是考虑怎样向有关对象发送消息,已完成所需要的任务。
对象=算法+数据结构
程序=(对象+对象+对象+……)+消息,消息的作用就是对对象的控制。
类的定义形式为
[public][final][abstract]class<类名>[extends<父类名>][implements<接口名列表>]
{<成员名>}
[]中的内容都不是必需的,<>中的内容是必需的。
public
表示该类可以为所有类使用,还有一种是默认访问权限,只对同一个包中的类可见。(修饰外部类的就只有这两种访问权限)
final
表示该类为终结类,即不允许被其他任何类继承而派生出子类。即不能成为父类。
abstract
表示该类是一个抽象类。在抽象类中,允许所定义的方法只有声明没有定义,抽象类一定是作为一个父类使用,被其他相关的类继承,然后在子类中实现没有实现的方法。一般类中的方法必须是有定义的。
extends<父类名>
给出所定义的类需要继承的父类,父类名是一个已经存在的一个类的标识符。
implements<接口名列表>
给出所定义的类需要实现的接口列表,若接口列表中超过一个接口名,则前后接口名之间必须要用逗号隔开。
类体(花括号内的部分)包含成员和方法。成员用来反映类的属性,方法用来反映类的行为。
类中成员的定义格式:
[<访问权限>][static][final]<数据类型><变量名列表>;//同样<>内是必选项
数据成员有四种可选择的访问权限(访问属性),其中三种分别用关键字public,protected,private表示,剩余一种访问权限不需要关键字,为默认(缺省,隐含)的访问权限。不同的访问权限对应着不同的访问范围。
public
数据成员为共有访问权限,能被所有的类访问
protected
数据成员为保护访问权限,能够被同一包中的类和继承于本类的派生类(无论是否在同一个包中)所访问
privated
数据成员为私有访问权限,只能被本类成员所访问,不能被其他任何类中的成员所访问
缺省
数据成员能够被本类和同一包中的其他任何类中的成员所访问
包类似文件夹
总结:无论是何种访问属性都能被本类中的任何成员访问,默认权限还能够被同一包中的任何类所访问,保护访问权限时更进一步能够被不同包中的任何子类所访问,公有访问权限时可以被所有的类访问。
访问范围从小到大的顺序为:私有<默认<保护<共有
需要注意的是保护修饰的数据不仅不同包中的子类可以访问,同一个包中的所有类都是可以访问的。
如果想要了解更多可以移步去看大佬的博客,有更加详细的解释:传送门
如果被final
修饰,则必须对该成员进行初始化(包含在构造函数中赋值),并且之后不能再被修改(相当于C++中的const
)
若一个数据成员带有关键字static
修饰,则表示他是类的静态成员类中的静态成员对所有对象都是公共的,共享一个存储空间。而实例成员变量则是互相独立的。
类中函数成员的定义格式为
[<访问权限>][abstract][static][final]<返回类型><函数名>(<参数列表>){<函数体>}
函数成员的访问权限与数据成员相同:public
为公有访问权限,能够被所有的类访问。protected
为保护访问权限,能够被本包中的类和子类访问。private
为私有访问权限,只能够被本类访问。缺省:此函数成员能够被本包中的类访问。
abstract
表示为抽象成员函数,这样的函数只有函数头没有函数体,在函数头末尾使用分号表示结束定义。抽象成员函数只能出现在抽象类的定义中。
static
表示静态成员函数,可以直接通过类名进行访问而不用进行实例化。在静态成员函数中只能访问静态成员变量和调用静态成员函数,不允许访问实例成员变量和函数。
final
表示终结性成员函数,该函数不允许被继承类中的成员函数所覆盖,也就是说不能被继承类重新定义,不允许有多态性。
类中的构造函数是特殊的成员函数,作用是为创建的对象初始化,该函数名与类名相同,不带返回值类型,只允许使用访问权限的修饰符,不允许使用其他任何修饰符。
当类中没有定义任何构造函数时,系统将隐含定义个一个默认的构造函数,函数体将初始化成员变量为默认值,具体的讲,将数值和字符变量初始化为0,逻辑变量初始化为false,对类对象成员初始化为null。如果有自己定义的构造函数系统将不会再自动定义一个构造函数。
如果类的访问权限是public,则默认构造函数的访问权限是public,如果没有采用public修饰,则默认构造函数的访问权限是friendly。
可以有多个成员函数名字相同,但是参数列表必须不同,这种现象叫做函数的重载。
我们可以用类来定义和创建对象,具体格式为:
<类名标识符><对象标识符>[=<初值表达式>];//需要通过new操作符得到地址
new运算符为一元运算符,操作数是调用该类的构造函数,在内存中生成一个存储对象并进行初始化,以返回该存储对象的首地址作为运算结果。
没有被new的对象是没有意义的。相当于一个空指针。
创建对象以后就可以通过点运算符访问成员变量和成员函数。点运算符具有最高的优先级。
对于静态成员整个类对应同一个存储空间,所以允许通过类名来直接访问。(当然也可以通过对象访问),对于实例成员只能通过对象进行访问。
Java类中的程序是由类所构成的,每个类可以是一个独立的编译单元。也可以把及各类的定义保存在同一个程序文件中共同成为一个编译单元,编译后每个类都有对应各自的字节码文件,即拓展名为.class的文件。
当一个Java程序中包含多个类的定义时,只允许包含主方法main的类被指定为公有类,即在类定义的开始用public修饰,其他的类必须是缺省访问权限的类,也就是说一个Java程序文件中不管包含多少个类的定义,不能同时出现两个及以上的类为public类
对于成员变量其作用域为整个类,所以变量具有类作用域,成员函数中形参变量作用域为整个函数。对于成员函数中顶一个局部变量,其作用域为从定义点开始到所在的语句块结束。在同一个作用域中不能出现重名,作用域范围小的变量一般不能与包含他的作用域大的变量重名。(否则小范围的会覆盖大范围的)允许函数参数和函数体中的变量与成员变量重名,但是在访问成员变量时应该加上this.前缀表示是类的成员变量。
函数的形参和实参必须对应,或者实参的类型可以被系统自动转换成形参的类型。特殊的,对于类中的构造函数是通过new操作符进行调用而不是通过点运算符进行调用。当形参为数组类型或者类(对象)类型时对形参的改变也会改变实参(很显然,因为实际传递的是指针)。
封装可被理解为一种改用做保护的包装器,以防止代码和数据被包装器外部所定义的其他代码任意访问。Java使用访问权限控制来实现封装。
让可访问性尽可能的低是促成封装的原则之一,即尽可能采用可行状态下最高的访问权限(最小的访问范围)。
final在Java中并不常用,但是他给我们提供了定义常量的功能。而且final还可以控制成员、方法或者类不可以被覆写或者继承。
在定义变量时在前面加上final关键字,对于基本类型来讲就是其值不可改变,对于对象来讲就是引用不可改变,即一直指向同一个对象。只能在定义或者构造函数中进行赋值,而且只能任选其一。