JavaSE——类和对象

目录

一、面向对象

1、定义

(1)、面向对象

(2)、面向过程

二、类的定义和使用

1、类的定义

三、类的实例化

1、定义

2、类和对象说明

四、this引用

1、定义

2、特性

五、对象的构造及初始化

1、初始化对象

(1)、定义

2、分类

(1)、默认初始化

(2)、就地初始化

3、构造方法

(1)、定义

(2)、特性

六、封装

1、封装的定义

2、访问限定符

3、封装扩展之包

(1)、定义

(2)、导入包中的类

(3)、自定义包

(4)、包的访问权限控制

(5)、常见的包

七、static成员

1、定义

2、static修饰成员变量

3、static修饰成员方法

4、static成员变量初始化

(1)、就地初始化

(2)、静态代码块初始化 

八、代码块

1、代码块定义

2、分类

(1)、普通代码块

(2)、构造代码块

(3)、静态代码块

(4)、同步代码块


一、面向对象

1、定义

(1)、面向对象

面向对象是一种解决问题的思想,当以面向对象方式来处理问题时,主要依靠对象之间的交互来解决问题,而不关注处理问题的具体步骤。在编程中运用面向对象的思想符合人们对事物的认知,对于大型程序的设计、扩展以及维护都非常友好。此外Java是一门纯面向对象的语言(Object Oriented Program,简称OOP)。

(2)、面向过程

注重于解决问题的具体步骤,按照该种方式来写代码,在扩展或者维护时较为麻烦。

注:面向过程和面相对象并不是哪一门语言,而是解决问题的方法,没有优劣之分,都有其专门的应用场景。

二、类的定义和使用

1、类的定义

类是用来描述一个对象的,主要描述该对象具有哪些属性、功能。

// 创建类
class ClassName{
    field;     // 属性(成员变量)
    method;     // 行为(成员方法)
}

//class为定义类的关键字,ClassName为类的名字,{}中为类的主体。
//属性主要是用来描述类,称之为类的成员属性或者类成员变量。
//方法主要说明类具有哪些功能,称为类的成员方法。

注:

  • 类名注意采用大驼峰定义
  • 成员前写法统一为public(后面会解释)
  • 此处写的方法不带 static 关键字(后面会解释)

例:

//定义一个狗类
class Dog {
    public String name;//名字
    public String color;//颜色
    // 狗的属性
    public void barks() {
        System.out.println(name + ": 旺旺旺~~~");
    } 
    // 狗的行为
    public void wag() {
        System.out.println(name + ": 摇尾巴~~~");
    }
}

注:

  1. 一般一个文件当中只定义一个类
  2. main方法所在的类一般要使用public修饰(注意:Eclipse默认会在public修饰的类中找main方法)
  3. public修饰的类必须要和文件名相同
  4. 不要轻易去修改public修饰的类的名称,如果要修改,通过开发工具修改

三、类的实例化

1、定义

定义一个类,就相当于在计算机中定义了一种新的类型,这与int,double类似,只不过int和double是java语言自带的内置类型,而类是用户自定义了一个新的类型。有了这些自定义的类型之后,就可以使用这些类来定义实例(或者称为对象)。用类类型创建对象的过程,称为类的实例化,在java中采用new关键字,配合类名来实例化对象。

public class Main{
    public static void main(String[] args) {
        PetDog dogh = new PetDog(); //通过new实例化对象
        dogh.name = "二哈";
        dogh.color = "黑色";
        dogh.barks();
        dogh.wag();
    }
} 
//输出结果:
//二哈: 旺旺旺~~~
//二哈: 摇尾巴~~~

注:

  • new 关键字用于创建一个对象的实例.
  • 使用 " . " 来访问对象中的属性和方法.
  • 同一个类可以创建对个实例

2、类和对象说明

类是一种自定义的类型,可以用来定义变量。从本质上来说类与模型相似,是用来对一个实体进行描述的,限定了类的成员,实际上不占用任何内存空间。此外一个类可以实例化出多个对象实例化出的对象占用内存空间,存储类成员变量。

四、this引用

1、定义

this引用指向当前对象(成员方法运行时调用该成员方法的对象),在成员方法中所有成员变量的操作,都是通过该引用去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。

public class Date {
    public int year;
    public int month;
    public int day;
    public void setDay(int year, int month, int day){
        this.year = year;
        this.month = month;
        this.day = day;
    }
    public void printDate(){
        System.out.println(this.year + "/" + this.month + "/" + this.day);
    }
}

 注:this引用的是调用成员方法的对象。

例:

public class Date {
    public int year;
    public int month;
    public int day;
    public void setDay(int y, int m, int d){
        year = y;
        month = m;
        day = d;
    }
    public void printDate(){
        System.out.println(year + "/" + month + "/" + day);
    }
    public static void main(String[] args) {
        // 构造三个日期类型的对象 d1 d2 d3
        Date d1 = new Date();
        Date d2 = new Date();
        Date d3 = new Date();
        // 对d1,d2,d3的日期设置
        d1.setDay(2022,9,15);
        d2.setDay(2022,9,16);
        d3.setDay(2022,9,17);
        // 打印日期中的内容
        d1.printDate();
        d2.printDate();
        d3.printDate();
    }
}
//以上代码定义了一个日期类,然后main方法中创建了三个对象,并通过Date类中的成员方法对对象进行设置和打印。

注:形参名不能与成员变量名相同:

public void setDay(int year, int month, int day){
    year = year;
    month = month;
    day = day;
}
//error!!!
//函数体无法分清赋值对象,此处需要使用 "this." !!!

2、特性

  • this的类型为对应类类型引用,即哪个对象调用就是哪个对象的引用类型
  • this只能在成员方法中使用,且在成员方法中this只能引用当前对象,不能再引用其他对象
  • this是“成员方法”第一个隐藏的参数,编译器会自动传递,在成员方法执行时,编译器会负责将调用成员方法对象的引用传递给该成员方法,this负责来接收

五、对象的构造及初始化

1、初始化对象

(1)、定义

在Java方法内部定义一个局部变量时,必须要初始化,否则会编译失败。要让上述代码通过编译只需在正式使用a之前,给a设置一个初始值即可。可如果是对象,就需要调用SetDate方法才可以将具体的日期设置到对象中。

public static void main(String[] args) {
    Date d = new Date();
    d.setDate(2021,6,9);
} 

2、分类

(1)、默认初始化

通过例子我们发现为局部变量在使用时必须要初始化,但是成员变量却可以不用初始化!

public class Date {
    public int year;
    public int month;
    public int day;
    public Date(int year, int month, int day) {
        // 成员变量在定义时,并没有给初始值, 却可以使用?
        System.out.println(this.year);
        System.out.println(this.month);
        System.out.println(this.day);
    }
    public static void main(String[] args) {
        // 此处a没有初始化,编译时报错:
        // Error:(24, 28) java: 可能尚未初始化变量a
        // int a;
        // System.out.println(a);
        Date d = new Date(2021,6,9);
    }
}

 要搞清楚这个过程,就需要知道 new 关键字背后所发生的一些事情

Date d = new Date(2021,6,9);

 在程序层面只是一条简单的语句,在JVM层面需要做好多事情:

  1. 检测对象对应的类是否加载,如果没有加载就加载
  2. 为对象分配内存空间
  3. 处理并发安全问题
  4. 初始化所分配的空间(例:空间被申请好之后,对象中包含的成员已经设置好了初始值)
  5. 设置对象头信息
  6. 调用构造方法,给对象中各个成员赋值

注:多个线程同时申请对象,JVM要保证给对象分配的空间不冲突

(2)、就地初始化

每次对象创建好后调用SetDate方法设置具体日期,比较麻烦,此时就可以在声明成员变量时,直接给出了初始值。

public class Date {
    public int year = 2022;
    public int month = 1;
    public int day = 1;
    public Date() { }
    public Date(int year, int month, int day) { }
    public static void main(String[] args) {
        Date d1 = new Date(2021,6,9);
        Date d2 = new Date();
    }
}

3、构造方法

(1)、定义

构造方法(也称为构造器)是一个特殊的成员方法,名字必须与类名相同,在创建对象时,由编译器自动调用,并且在整个对象的生命周期内只调用一次。

public class Date {
    public int year;
    public int month;
    public int day;
    // 构造方法:
    // 名字与类名相同,没有返回值类型,设置为void也不行
    // 一般情况下使用public修饰
    // 在创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次
    public Date(int year, int month, int day){
        this.year = year;
        this.month = month;
        this.day = day;
        System.out.println("Date(int,int,int)方法被调用了");
    }
    public void printDate(){
        System.out.println(year + "-" + month + "-" + day);
    }
    public static void main(String[] args) {
        // 此处创建了一个Date类型的对象,并没有显式调用构造方法
        Date d = new Date(2021,6,9); // 输出Date(int,int,int)方法被调用了
        d.printDate(); // 2021-6-9
    }
}

注:构造方法的作用就是对对象中的成员进行初始化,并不负责给对象开辟空间

(2)、特性

1. 名字必须与类名相同

2. 并且没有返回值类型,设置为void也不行

3. 创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次

4. 构造方法可以重载(用户根据自己的需求提供不同参数的构造方法)

例:

public class Date {
    public int year;
    public int month;
    public int day;
    // 无参构造方法
    public Date(){
        this.year = 1900;
        this.month = 1;
        this.day = 1;
    }
    // 带有三个参数的构造方法
    public Date(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }
    public static void main(String[] args) {
        Date d = new Date();
    }
}

上述两个构造方法:名字相同,参数列表不同,因此构成了方法重载 

5. 如果用户没有显式定义,编译器会生成一份默认的构造方法,生成的默认构造方法一定是无参的

public class Date {
    public int year;
    public int month;
    public int day;
    public static void main(String[] args) {
        Date d = new Date();
    }
}

上述Date类中,没有定义任何构造方法,编译器会默认生成一个不带参数的构造方法。但是一旦用户自行定义构造函数,编译器则不再生成!!!

例:

public class Date {
    public int year;
    public int month;
    public int day;
    public Date(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }
    public static void main(String[] args) {
        // 如果编译器会生成,则生成的构造方法一定是无参的
        // 则此处创建对象是可以通过编译的
        // 但实际情况是:编译期报错
        Date d = new Date();
    }
} 
/*Error:(26, 18) java: 无法将类 extend01.Date中的构造器 Date应用到给定类型;
需要: int,int,int
找到: 没有参数
原因: 实际参数列表和形式参数列表长度不同
*/

6. 构造方法中,可以通过this调用其他构造方法来简化代码

public class Date {
    public int year;
    public int month;
    public int day;
    // 无参构造方法--内部给各个成员赋值初始值,该部分功能与三个参数的构造方法重复
    // 此处可以在无参构造方法中通过this调用带有三个参数的构造方法
    // 但是this(1900,1,1);必须是构造方法中第一条语句
    public Date(){
        //System.out.println(year); 注释取消掉,编译会失败
        this(1900, 1, 1);
    } 
    // 带有三个参数的构造方法
    public Date(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }
}

 注:

  • this( ) 必须是构造方法中第一条语句
  • 不能形成环
     
public Date(){
this(1900,1,1);
}
public Date(int year, int month, int day) {
this();
} /
* 无
参构造器调用三个参数的构造器,而三个参数构造器有调用无参的构造器,形成构造器的递归调用
编译报错:Error:(19, 12) java: 递归构造器调用
*/

7. 绝大多数情况下使用public来修饰,特殊场景下会被private修饰

六、封装

1、封装的定义

封装、继承、多态是面向对象的三大特性。而类和对象阶段,主要研究的就是封装特性。
封装就是将数据和操作数据的方法进行有机结合,隐藏对象的属性和实现细节,仅对外公开接口来和对象进行交互。

2、访问限定符

Java中主要通过类和访问权限来实现封装:类可以将数据以及封装数据的方法结合在一起,而访问权限用来控制方法或者字段能否直接在类外使用。

Java中提供了以下四种访问限定符:

注: 

  • protected主要是用在继承中,继承中会详细介绍
  • default权限指:什么都不写时的默认权限
  • 访问权限除了可以限定类中成员的可见性,也可以控制类的可见性

例:

public class Computer {
    private String cpu;
    private String memory; 
    public String screen; 
    String brand; // 品牌---->default属性
    public Computer(String brand, String cpu, String memory, String screen) {
        this.brand = brand;
        this.cpu = cpu;
        this.memory = memory;
        this.screen = screen;
    }
}
public class TestComputer {
    public static void main(String[] args) {
        Computer p = new Computer("HW", "i7", "8G", "13*14");
        System.out.println(p.brand); // default属性:只能被本包中类访问
        System.out.println(p.screen); // public属性: 可以任何其他类访问
        // System.out.println(p.cpu); // private属性:只能在Computer类中访问。
    }
}

注:一般情况下成员变量设置为private,成员方法设置为public。

3、封装扩展之包

(1)、定义

在面向对象体系中,提出了一个软件包的概念,即:为了更好的管理类,把多个类收集在一起成为一组,称为软件包。有点类似于目录。

比如:为了更好的管理电脑中的歌曲,一种好的方式就是将相同属性的歌曲放在相同文件下,也可以对某个文件夹下的音乐进行更详细的分类。

在Java中也引入了包,包是对类、接口等的封装机制的体现,是一种对类或者接口等的很好的组织方式。

比如:一个包中的类不想被其他包中的类使用。包还有一个重要的作用:在同一个工程中允许存在相同名称的类,只要处在不同的包中即可。

(2)、导入包中的类

Java 中提供了很多现成的类供我们使用。

例如Date类:可以使用 java.util.Date 导入 java.util 这个包中的 Date类.

public class Test {
    public static void main(String[] args) {
        java.util.Date date = new java.util.Date();
        // 得到一个毫秒级别的时间戳
        System.out.println(date.getTime());
    }
}

简化写法,可以使用 import语句导入包

import java.util.Date;
public class Test {
    public static void main(String[] args) {
        Date date = new Date();
        // 得到一个毫秒级别的时间戳
        System.out.println(date.getTime());
    }
}

 如果还需要使用 java.util 中的其他类, 可以使用 import java.util.*

import java.util.*;
public class Test {
    public static void main(String[] args) {
        Date date = new Date();
        // 得到一个毫秒级别的时间戳
        System.out.println(date.getTime());
    }
}

建议使用完整类名的显式导入,否则可能会出现冲突的情况。

例:

import java.util.*;
import java.sql.*;
public class Test {
    public static void main(String[] args) {
        // util 和 sql 中都存在一个 Date 这样的类, 此时就会出现歧义, 编译出错
        Date date = new Date();
        System.out.println(date.getTime());
    }
} 
// 编译出错
Error:(5, 9) java: 对Date的引用不明确
java.sql 中的类 java.sql.Date 和 java.util 中的类 java.util.Date 都匹配

可以使用import static导入包中静态的方法和字段

import static java.lang.Math.*;
public class Test {
public static void main(String[] args) {
double x = 30;
double y = 40;
// 静态导入的方式写起来更方便一些.
// double result = Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2));
double result = sqrt(pow(x, 2) + pow(y, 2));
System.out.println(result);
}
}

注:import 和 C++ 的 #include 差别很大。C++ 必须 #include 来引入其他文件内容,但是 Java 不需要。import 只是为了写代码的时候更方便,import 更类似于 C++ 的 namespace 和 using

(3)、自定义包

定义:

  • 在文件的最上方加上一个 package 语句指定该代码在哪个包中。
  • 包名需要尽量指定成唯一的名字, 通常会用公司的域名的颠倒形式(例如 com.bit.demo1 )。
  • 包名要和代码路径相匹配。例如创建 com.bit.demo1 的包,那么会存在一个对应的路径 com/bit/demo1 来存储代码。
  • 如果一个类没有 package 语句,则该类被放到一个默认包中。

(4)、包的访问权限控制

Computer类位于com.bit.demo1包中,TestComputer位置com.bit.demo2包中

package com.bit.demo1;
    public class Computer {
    private String cpu; 
    public String screen; 
    String brand; 
    public Computer(String brand, String cpu, String screen) {
        this.brand = brand;
        this.cpu = cpu;
        this.screen = screen;
    }
}     
package com.bite.demo2;
import com.bite.demo1.Computer;
public class TestComputer {
    public static void main(String[] args) {
        Computer p = new Computer("HW", "i7", "13*14");
        System.out.println(p.screen);
        // System.out.println(p.cpu); // 报错:cup是私有的,不允许被其他类访问
        // System.out.println(p.brand); // 报错:brand是default,不允许被其他包中的类访问
    }
}
// 注意:如果去掉Computer类之前的public修饰符,代码也会编译失败

(5)、常见的包

  • java.lang:系统常用基础类(String、Object),此包从JDK1.1后自动导入
  • java.lang.reflect:java 反射编程包
  • java.net:进行网络编程开发包
  • java.sql:进行数据库开发的支持包
  • java.util:是java提供的工具程序包
  • java.io:I/O编程开发包

七、static成员

1、定义

被static修饰的成员,称之为静态成员,也可以称为类成员,其不属于某个具体的对象,是被所有对象所共享的。

2、static修饰成员变量

static修饰的成员变量,称为静态成员变量.

静态成员变量最大的特性:不属于某个具体的对象,是所有对象所共享的。

静态成员特性:

  • 不属于某个具体的对象,是类的属性,所有对象共享的,不存储在某个对象的空间中
  • 既可以通过对象访问,也可以通过类名访问,但一般更推荐使用类名访问
  • 类变量存储在方法区当中
  • 生命周期伴随类的一生(即:随类的加载而创建,随类的卸载而销毁)

例:

public class Student{
    public String name;
    public String gender;
    public int age;
    public double score;
    public static String classRoom = "Bit306";

    public static void main(String[] args) {
        // 静态成员变量可以直接通过类名访问
        System.out.println(Student.classRoom);
        Student s1 = new Student("Li leilei", "男", 18, 3.8);
        Student s2 = new Student("Han MeiMei", "女", 19, 4.0);
        Student s3 = new Student("Jim", "男", 18, 2.6);
        // 也可以通过对象访问:但是classRoom是三个对象共享的
        System.out.println(s1.classRoom);
        System.out.println(s2.classRoom);
        System.out.println(s3.classRoom);
    }
}

3、static修饰成员方法

(1)、访问private修饰的static属性。

Java中,被static修饰的成员方法称为静态成员方法,是类的方法,不是某个对象所特有的。静态成员一般是通过静态方法来访问的。

public class Student{
    private static String classRoom = "Bit306";
    public static String getClassRoom(){
        return classRoom;
    }
}
public class TestStudent {
    public static void main(String[] args) {
        System.out.println(Student.getClassRoom());
    }
}
//输出:Bit306

静态方法特性:

  • 不属于某个具体的对象,是类方法
  • 可以通过对象调用,也可以通过类名.静态方法名( ) 方式调用,推荐使用后者。
  • 不能在静态方法中访问任何非静态成员变量

例:

public static String getClassRoom(){
    System.out.println(this);
    return classRoom;
} 
// 编译失败:Error:(35, 28) java: 无法从静态上下文中引用非静态 变量 this
public static String getClassRoom(){
    age += 1;
    return classRoom;
} 
// 编译失败:Error:(35, 9) java: 无法从静态上下文中引用非静态 变量 age

(2)、静态方法中不能调用任何非静态方法

因为非静态方法有this参数,在静态方法中调用时候无法传递this引用

public static String getClassRoom(){
    doClass();
    return classRoom;
} 
// 编译报错:Error:(35, 9) java: 无法从静态上下文中引用非静态 方法 doClass()

(3)、静态方法无法重写,不能用来实现多态

4、static成员变量初始化

静态成员变量一般不会放在构造方法中来初始化,构造方法中初始化的是与对象相关的实例属性

静态成员变量的初始化分为两种:就地初始化和静态代码块初始化。

(1)、就地初始化

在定义时直接给出初始值

public class Student{
    private String name;
    private double score;
    private static String classRoom = "Bit306";
}

(2)、静态代码块初始化 

(后文代码块处讲解)

八、代码块

1、代码块概念以及分类

使用 " { } " 定义的一段代码称为代码块。根据代码块定义的位置以及关键字,可分为以下四种:
普通代码块、构造块、静态块、同步代码块

2、分类

(1)、普通代码块

定义在方法中的代码块

public class Main{
public static void main(String[] args) {
{ //直接使用{}定义,普通方法块
int x = 10 ;
System.out.println("x1 = " +x);
} in
t x = 100 ;
System.out.println("x2 = " +x);
}
} /
/ 执行结果
x1 = 10
x2 = 100

(2)、构造代码块

构造块:定义在类中的代码块(不加修饰符),也叫实例代码块。                                                    构造代码块一般用于初始化实例成员变量

例:

public class Student{
    //实例成员变量
    private String name;
    private int age;
    private double score;
    public Student() {
        System.out.println("I am Student init()!");
    } 
    //实例代码块
    {
        this.name = "bit";
        this.sex = "man";
        System.out.println("I am instance init()!");
    }
    public void show(){
        System.out.println("name: "+name+" age: "+age);
    }
}

public class Main {
    public static void main(String[] args) {
        Student stu = new Student();
        stu.show();
    }
} 
// 运行结果
//I am instance init()!
//I am Student init()!
//name: bit age: 12 

(3)、静态代码块

使用static定义的代码块称为静态代码块。一般用于初始化静态成员变量

public class Student{
    private String name;
    private double score;
    private static String classRoom;
    // 实例代码块
    {
        this.name = "bit";
        System.out.println("I am instance init()!");
    } 
    // 静态代码块
    static {
        classRoom = "bit306";
        System.out.println("I am static init()!");
    }
    public Student(){
        System.out.println("I am Student init()!");
    }
    public static void main(String[] args) {
        Student s1 = new Student();
        Student s2 = new Student();
    }
}

注:

  • 静态代码块不管生成多少个对象,其只会执行一次
  • 静态成员变量是类的属性,因此是在JVM加载类时开辟空间并初始化的
  • 如果一个类中包含多个静态代码块,在编译代码时,编译器会按照定义的先后次序依次执行(合并)
  • 实例代码块只有在创建对象时才会执行

(4)、同步代码块

下一篇内部类会讲解

  • 28
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 27
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 27
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值