什么叫面向对象(java面向对象概述和基础语法入门)
面向对象思想
概述
面向对象是软件开发方法。面向对象的概念和应用已超越了程序设计和软件开发,是一种对现实世界理解和抽象的方法,是计算机编程技术发展到一定阶段后的产物。
面向对象是相当于面向过程来讲的,指的是把相关的数据和方法组织为一个整体来看待,从更高的层次来进行系统建模,更贴近事物的自然运行模式。
面向过程到面向对象思想层面的转变:
面向过程关注的是执行的过程,面向对象关注的是具备功能的对象。
面向过程到面向对象,是程序员思想上 从执行者到指挥者的转变。
三大思想
面向对象思想从概念上讲分为以下三种:OOA、OOD、OOP
OOA:面向对象分析(Object Oriented Analysis)
OOD:面向对象设计(Object Oriented Design)
OOP:面向对象程序(Object Oriented Programming
三大特征
封装性 :所有的内容对外部不可见
概述:封装的意义在于保护或者防止代码(数据)被我们无意中破坏
保护成员属性,不让类意外的程序直接访问和修改
封装原则:
隐藏对象的属性和实现细节,仅对外公开访问方法,并且控制访问级别
继承性 :将其他的功能继承下来继续发展
继承是Java面向对象编程技术的一块基石,因为它允许创建分等级层次的类
继承就是子类继承父类的特征和行为,使子类对象(实例)具有父类的实例和方法,或子类从父类继承方法,使得子类具有父类相同的行为。
继承原则:
java中只有单继承,多重继承。没有多继承
多态性 :
方法的重载本身就是一个多态性的体现
类与对象
两者关系
类表示一个共性的产物,是一个综合的特征,而对象,是一个个性的产物,是一个个体的特征。 (类似生活中的图纸与实物的概念。)
类必须通过对象才可以使用,对象的所有操作都在类中定义。
类由属性和方法组成:
· 属性:就相当于人的一个个的特征
· 方法:就相当于人的一个个的行为,例如:说话、吃饭、唱歌、睡觉
类时创建对象的模板,对象时类的实例化。
类的定义格式
class 类名称{
成员属性
成员方法
}
属性与方法
属性定义格式:
数据类型 属性名 ;
属性定义并赋值的格式:
数据类型 属性名 = 初始化值 ;
方法定义格式:
权限修饰符 返回值类型 方法名(形式参数列表){
//方法体
return 返回值;
}
对象的创建与使用
一个类要想真正的进行操作,则必须依靠对象,对象的定义格式如下:
类名称 对象名称 = new 类名称() ;
如果要想访问类中的属性或方法(方法的定义),则可以依靠以下的语法形式:
访问类中的属性: 对象.属性 ;
调用类中的方法: 对象.方法(实际参数列表) ;
创建对象内存分析
栈
Java栈的区域很小 , 大概2m左右 , 特点是存取的速度特别快
栈存储的特点是, 先进后出
存储速度快的原因:
指针向下移动,会创建新的内存,向上移动,会释放这些内存
这种方式速度特别块,仅次于PC寄存器
但是这种移动的方式,必须要明确移动的大小与范围
明确大小与范围是为了方便指针的移动,这是一个对于数据存储的限制,存储的数据大小是固定的,影响了程序的灵活性
所以我们把更大部分的数据 存储到堆内存中
存储的是:
基本数据类型的数据 以及 引用数据类型的引用
例如:
int a = 10;
Person p = new Person();
10存储在栈内存中,第二句代码创建的对象的引用(p)存在栈内存中
堆
存放的是类的对象 .
Java是一个纯面向对象语言, 限制了对象的创建方式:
所有类的对象都是通过new关键字创建
new关键字, 是指告诉JVM , 需要明确的去创建一个新的对象 , 去开辟一块新的堆内存空间:
堆内存与栈内存不同, 优点在于我们创建对象时 , 不必关注堆内存中需要开辟多少存储空间 , 也不需要关注内存占用
时长 !
堆内存中内存的释放是由GC(垃圾回收器)完成的
垃圾回收器 回收堆内存的规则:
当栈内存中不存在此对象的引用时,则视其为垃圾 , 等待垃圾回收器回收 !
例如:
Person p0 = new Person();
Person p1 = p0;
Person p2 = new Person()
方法区
存放的是
类信息
静态的变量
常量
成员方法
方法区中包含了一个特殊的区域(常量池)(存储的是使用static修饰的成员)
PC寄存器
PC寄存器保存的是 当前正在执行的JVM指令的地址
在java程序中,每个线程启动时,都会创建一个PC寄存器
本地方法栈
保存本地(native)方法的地址
构造方法(构造器)
回顾对象创建
Person p = new Person();
在右侧Person后面出现的小括号, 其实就是在调用构造方法 !
概述
作用:
用于对象初始化。
执行时机:
在创建对象时,自动调用
特点:
所有的Java类中都会至少存在一个构造方法
如果一个类中没有明确的编写构造方法, 则编译器会自动生成一个无参的构造方法, 构造方法中没有任何的代码!
如果自行编写了任意一个构造器, 则编译器不会再自动生成无参的构造方法
方法名和类名相同,也没有
定义格式
定义的格式:
与普通方法基本相同, 区别在于: 方法名称必须与类名相同, 没有返回值类型的声明 !
案例:
public class Demo3{
public static void main(String[] args){
Person p = new Person();
p = new Person();
p = new Person();
p = new Person();
}
}
class Person{
public Person(){
System.out.println("对象创建时,此方法调用");
}
}
###
构造方法设计
建议自定义无参构造方法,不要对编译器形成依赖,避免错误发生。
当类中有非常量成员变量时,建议提供两个版本的构造方法,一个是无参构造方法,一个是全属性做参数的构造方法。
当类中所有成员变量都是常量或者没有成员变量时,建议不提供任何版本的构造。
方法的重载
方法名称相同, 参数类型或参数长度不同, 可以完成方法的重载 ! 方法的重载与返回值无关!
方法的重载 ,可以让我们在不同的需求下, 通过传递不同的参数调用方法来完成具体的能。
构造方法的重载
一个类, 可以存在多个构造方法 :
参数列表的长度或类型不同即可完成构造方法的重载 ~
构造方法的重载 ,可以让我们在不同的创建对象的需求下, 调用不同的方法来完成对象的初化
匿名对象
没有对象名称的对象 就是匿名对象。
匿名对象只能使用一次,因为没有任何的对象引用,所以将称为垃圾,等待被G·C回收。
只使用一次的对象可以通过匿名对象的方式完成,这一点在以后的开发中将经常使用到。
面向对象进阶
this
在java基础中,this关键字是一个最重要的概念,使用this关键字可以完成一下的操作
调用类中的属性
调用类中的方法或构造方法
在一个构造方法中,调用另一个构造方法时,调用的代码必须编写在构造方法的第一行
表示当前对象
this():调用当前类中的构造方法
this(参数):有几个参数就调用当前类中有几个参数的构造方法
this():必须放在构造的第一行的位置上
package day06; /**
* Description:
*
* @author domoreda
* @date 2020/8/12 0012 11:13
*/
public class Test1 {
public static void main(String[] args) {
Person1 p1 = new Person1("张三",18);
Person1 p2 = new Person1("李四",20);
p1.say();
p2.say(); Person1 p3 = new Person1();
p1.say();
}
}
class Person1{
private String name;
private int age;
Person1(){
this("默认姓名",1); //在一个构造方法中,调用另一个构造方法时,调用的代码必须编写在构造方法的第一行
}
Person1(String name,int age){
this.name = name;
this.age = age;
}
void say(){
System.out.println("姓名" + this.name + ",年龄:" +this.age);
}
}
static
概述
static表示“静态”的意思,可以用来修饰变量和成员方法(后续还会学习 静态代码块和静态内部类)
static的主要作用在于创建独立于具体对象的域变量或者方法
简单理解:
被static关键字修饰的方法或者变量不需要依赖于对象来进行访问,只要类被加载了,就可以通过类名去进行访问。
并且不会因为对象的多次创建 ,而在内存中建立多份数据
重点:
1、静态成员 在类加载时加载并初始化
2、无论一个类存在多少个对象,静态的属性,永远在内存中只有一份(可以理解为所有对象公用)
3、在访问时:静态不能访问非静态,(因为对象还可能未创建)。非静态可以访问静态。
静态方法:类名.方法名()
实例方法:对象名.属性名 每一个对象单独使用的
权限修饰符
1)使用public修饰的变量和方法,在任何一个类中创建对象后都可以访问到,包括当前类、同包的类、子孙类以及外包的类。
2)使用protected修饰符的变量和方法,不可以访问外包类创建的对象。
3)不写修饰符(friendly/default)的变量和方法,为默认类,只可以访问同一个包中类创建的对象。不可一访问子孙类
4)使用private修饰符的变量和方法,只能在当前类中使用
注意:protected虽然不可一在不同包下访问,但是父类和子类在不同包下时,子类可以访问父类中protected修饰的字段和方法。
代码块
普通代码块
在执行的流程中 出现的代码块,我们称其为普通代码块(可以进行初始化工作)
构造代码块
在类中的成员代码块,我们称其为构造代码块,在每次对象创建时执行,执行在构造方法之前。
静态代码块
在类中使用static修饰的成员代码块,我们称其为静态代码块,在类加载时执行,每次程序启动到关闭,只会执行一次的代码块
同步代码块
解决程序中不符合现实的规律(线程的时候,例如,买火车票 位置:12A 张三买好, 李四不能买了)
面试题:
构造方法与构造代码块以及静态代码块的执行顺序
静态代码块—构造代码块----构造方法
package:
package 包名;
多个单词组成,单词的字母小写,单词与单词之间用.隔开
super:
通过super,可以访问父类构造方法
调用super构造方法的代码,必须写在子类构造方法的第一行
通过super,可以访问父类的属性
通过super,可以访问父类的方法
子类当中如果没有写构造方法,java会提供一个默认的无参数的构造方法。
重写规则:
1、参数列表必须完全与被重写方法相同
2、返回类型必须完全与重写方法的返回类型相同
3、访问权限不能比父类中被重写的方法的权限更低。例如,如果父类的一个方法被声明为public,那么在子类中重写方法就不能声明为protected。
4、父类的成员方法只能被它的子类重写
5、声明为static和private的方法不能被重写,但是能够被再次声明
面试题:
重写(Override)和重载(Overload)的区别
1、发生的位置
重载:一个类中
重写:子父类中
2、参数列表限制
重载:必须不同的
重写:必须相同
3、返回值类型
重载:与返回值类型无关
重写:返回值类型必须一直
4、访问权限
重载:与访问权限无关
重写:子的方法权限必须不能小于父的方法权限
5、异常处理
重载:与异常无关
重写:异常范围可以更小,但不能抛出新的异常
final
final用于修饰属性、变量
变量成为了常量,无法对其再次进行赋值
final修饰的局部变量,只能复制一次(可以先声明后赋值)
final修饰的是成员属性,必须在声明时赋值
全局常量(public static final)
常量的命名规范
由一个或多个单词组成,单词与单词之间必须使用下划线隔开,单词中所有字母大写
例如:SQL-INSERT
final用于修饰类 :
final修饰的类,不可以被继承
final用于修饰方法:
final修饰的方法,不能被子类重写。
抽象类
概念
抽象类必须使用abstract class说明
一个抽象类中可以没有抽象方法。抽象方法必须写在抽象类或者接口中
格式:
abstract class 类名{ //抽象类
}
抽象方法
只声明而未实现的方法称为抽象方法(未实现指的是:没有“{}”方法体),抽象方法必须使用abstract关键字声明
格式:
abstract class 类名{ //抽象类
public abstract void 方法名();//抽象方法,只声明而未实现。没有方法体
}
不能被实例化
在抽象的使用中有几个原则:
抽象类本身是不能直接进行实例化操作的,即:不能直接使用关键字new完成
一个抽象类必须被子类所继承,被继承的子类(如果不是抽象类)则必须覆写(重写)抽象类中的全部抽象方法
常见问题
1、抽象能否使用final声明?
不能,因为final属修饰的类是不能有子类的,而抽象类必须有子类才有意义,所以不能。
2、抽象类能否有构造方法?
能有构造方法,而且子类对象实例化的时候的流程和普通类的继承是一样的,都是要先调用父类中的构造方法(默认是无参的),之后再调用子类自己的构造方法。
抽象类和普通类的区别
1、抽象类必须用public或protected修饰(如果为private修饰,那么子类则无法继承,也就无法实现其抽象方法)。默认为public
2、抽象类不可以使用new关键字创建对象,但是在子类创建对象时,抽象父类也会被JVM实例化
3、如果一个子类继承抽象类,那么必须实现其所有的抽象方法。如果有为实现的抽象方法,那么子类也必须定义为abstract类
接口
概念
如果一个类中的全部方法都是抽象方法,全部属性都是全局变量,那么此时就可以将这个类定义成一个接口。定义格式:
interface 接口名称{
全局变量;
抽象方法;
}
面向接口编程思想
这种思想是接口是定义(规范,约束)与实现(名实分离的原则)的分离
优点:
1、降低程序耦合性
2、易于程序的扩展
3、有利于程序的维护
全局变量和抽象方法的简写
因为接口本身都是由全局常量和抽象方法组成 , 所以接口中的成员定义可以简写:
1、全局常量编写时, 可以省略public static final 关键字,例如:
public static final String INFO = “内容” ;
简写后:
String INFO = “内容” ;
2、抽象方法编写时, 可以省略 public abstract 关键字, 例如:
public abstract void print() ;
简写后:
void print() ;
接口的实现 implements
接口可以多实现:
格式:
class 子类 implements 父接口1,父接口2...{
}
//以上的代码称为接口的实现。那么如果一个类即要实现接口,又要继承抽象类的话,则按照以下的格式编写即可:
class 子类 extends 父类 implements 父接口1,父接口2...{
}
接口的继承
接口因为都是抽象部分, 不存在具体的实现, 所以允许多继承,例如:
interface C extends A,B{
}
注意
如果一个接口要想使用,必须依靠子类,子类(如果不是抽象类的话)要实现接口中的所有抽象方法
接口和抽象类的区别
1、抽象类要被子类继承,接口要被类继承
2、接口只能声明抽象方法,抽象类中可以声明抽象方法,也可以写非抽象方法
3、接口里定义的变量只能是公共的静态的变量,抽象类中的变量是普通变量
4、抽象类使用继承来使用,无法多继承。接口使用实现来使用,可以多实现
5、抽象类中可以包含sattic方法,但是接口中不允许(静态方法不能被子类重写,因此接口中不能声明静态方法)
6、接口不能有构造方法,但是抽象类中可以有
多态
概念
多态:就是对象的多种表现形式(多种体现形态)
多态的体现
对象的多态性,从概念上非常好理解,在类中有子类和父类之分,子类就是父类的一种形态,对象多态性就从此而来。
实现的前天条件:
继承
重写
向上转换
ps:方法的重载 和重写也是多态的一种,不过是方法的多态(相同方法名的多种形态)
重载:一个类中方法的多态性体现
重写:子父类中方法的多态性体现
多态的使用:对象的类型转换
类似于基本数据类型的转换:
· 向上转型:将子类实例变为父类实例
|- 格式:父类 父类对象 = 子类实例 ;
· 向下转型:将父类实例变为子类实例
|- 格式:子类 子类对象 = (子类)父类实例 ;
instanceof
作用:
判断某个对象是否是指定类的实例,则可以使用instanced关键字
格式:
实例化对象 instanceof 类 //此操作返回boolean类型的数据
Object类
概念
Object类是所有类的父类(基类),如果一个类没有明确的继承某一个具体的类,则将默认继承Object
例如我们定义一个类:
public class Person{
}
//其实它被使用时 是这样的:
public class Person extends Object{
}
Object的多态
使用Object可以接收任意的引用数据类型
toString
建议重写Object中的toString方法。 此方法的作用:返回对象的字符串表示形式。
Object的toString方法, 返回对象的内存地址
equals :比较两个对象是否相等
hasnhode:返回对象的哈希码值
toString():返回对象的字符串
内部类
概念
在java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类
广泛意义上内部类一般来说包括这四种:
1、成员内部类
2、局部内部类
3、匿名内部类
4、静态内部类
成员内部类
成员内部类是最普通的内部类,它的定义为位于另一个类的内部,形如下面的形式:
class Outer {
private double x = 0;
public Outer(double x) {
this.x = x;
}
class Inner { //内部类
public void say() {
System.out.println("x="+x);
}
}
}
特点: 成员内部类可以无条件访问外部类的所有成员属性和成员方法(包括private成员和静态成员)。
不过要注意的是,当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问
的是成员内部类的成员。如果要访问外部类的同名成员,需要以下面的形式进行访问:
外部类 . this . 成员变量
外部类 . this . 成员方法
外部使用成员内部类
Outter outter = new Outter ();
Outter . Inner inner = outter . new Inner ();
局部内部类
局部内部类是定义在一个方法或者一个作用域里面的类,它和成员内部类的区别在于局部内部类的访问仅限于方法内或 者该作用域内。
例如:
class Person{
public Person() {
}
}
class Man{
public Man(){
}
public People getPerson(){
class Student extends People{ //局部内部类
int age =0;
}
return new Student();
}
}
注意:局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。
匿名内部类
匿名内部类由于没有名字,所以它的创建方式有点儿奇怪。创建格式如下:
new 父类构造器(参数列表)|实现接口()
{
//匿名内部类的类体部分
}
在这里我们看到使用匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一
个接口。同时它也是没有class关键字,这是因为匿名内部类是直接使用new来生成一个对象的引用。当然这个引用是隐
式的。
注意
在使用匿名内部类的过程中,我们需要注意如下几点
1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或
者实现一个接口。
2、匿名内部类中是不能定义构造函数的。
3、匿名内部类中不能存在任何的静态成员变量和静态方法。
4、匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。
5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。
6、只能访问final型的局部变量
面试:匿名内部类和局部内部类为社么要用final修饰局部变量?
因为内部类会被单独的编译成一个字节码文件,为了保证单独的文件中用到底变量(备份的变量)与外面的值绝对一致,所以使用final修饰限定了值的更改
静态内部类
静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。
静态内部类是不需要依赖于外部类对象的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员 变量或者方法
格式:
public class Test {
public static void main(String[] args) {
Outter.Inner inner = new Outter.Inner();
}
}
class Outter {
public Outter() {
}
static class Inner {
public Inner() {
}
}
}
包装类
概述
在Java中有一个设计的原则“一切皆对象”,那么这样一来Java中的一些基本的数据类型,就完全不符合于这种设计思 想,因为Java中的八种基本数据类型并不是引用数据类型,所以Java中为了解决这样的问题,引入了八种基本数据类型 的包装类。
以上的八种包装类,可以将基本数据类型按照类的形式进行操作。
但是,以上的八种包装类也是分为两种大的类型的:
· Number:Integer、Short、Long、Double、Float、Byte都是Number的子类表示是一个数字。
· Object:Character、Boolean都是Object的直接子类。
装箱和拆箱操作
以下以Integer和Float为例进行操作
将一个基本数据类型变为包装类,那么这样的操作称为装箱操作。
将一个包装类变为一个基本数据类型,这样的操作称为拆箱操作,
因为所有的数值型的包装类都是Number的子类,Number的类中定义了如下的操作方法,以下的全部方法都是进行拆箱的操作。
可变参数
一个方法中定义完了参数,则在调用的时候必须传入与其一一对应的参数,但是在JDK 1.5之后提供了新的功能,可以根 据需要自动传入任意个数的参数。
语法:
返回值类型 方法名称(数据类型…参数名称){
//参数在方法内部 , 以数组的形式来接收
}
注意:
可变参数只能出现在参数列表的最后。
递归
递归,在数学与计算机科学中,是指在方法的定义中使用方法自身。也就是说,递归算法是一种直接或间接调用自身方法的算法
本文地址:https://blog.csdn.net/weixin_44741610/article/details/108032202
上一篇: 小林求职记(六)踩过Dubbo坑,回答印象深,干货整理
下一篇: ES6——对象