Java面向对象-类的概念
一、类与对象
1. 创建类
class Phone{
String name;
int price;
public void call(){
System.out.println("手机用来打电话");
}
public void sendMessage(String message){
System.out.println("发送信息:" + message);
}
}
2. 创建类的第1个对象
public class PhoneTest {
public static void main(String[] args) {
//
// Scanner scan = new Scanner(System.in);
// int[] arr = new int[3];
Phone p1 = new Phone();//类的实例化、创建类的对象
System.out.println("name : " + p1.name + ",price : " + p1.price);
//"对象.属性" 或 "对象.方法"的方式完成功能的执行
p1.name = "HUAWEI mate30 pro";
p1.price = 5299;
System.out.println("name : " + p1.name + ",price : " + p1.price);
p1.call();
p1.sendMessage("有内鬼,停止交易");
//The method playGame() is undefined for the type Phone
//p1对象所在的类中没有声明过playGame()方法
// p1.playGame();
}
}
3. 创建第2个对象
Phone p2 = new Phone();
p2.name = "xiaomi 10";
p2.price = 3999;
System.out.println("name : " + p2.name + ",price : " + p2.price);
System.out.println("name : " + p1.name + ",price : " + p1.price);
创建好的对象只能调用其所在类中声明过的功能:属性、方法的调用。
如果创建一个类的多个对象,则每个对象都拥有一套类中的属性的“副本”。如果修改其中一个对象的属性值,不会影响到其他对象此属性的值。
4. 引用变量赋值
//p2和p3变量存储的值相同,此时的值即为堆空间中的对象的首地址。
//意味着p2和p3引用变量共同指向了堆空间中的同一个对象。
Phone p3 = p2;
System.out.println("name : " + p3.name + ",price : " + p3.price);
p3.price = 3499;
System.out.println("name : " + p2.name + ",price : " + p2.price);
如果编写“Phone p3 = p2”,此时并没有新创建一个对象p3。
p2和p3变量存储的值相同,此时的值即为堆空间中的对象的首地址。意味着p2和p3引用变量共同指向了堆空间中的同一个对象。
5. 对象的内存解析
6. 匿名对象
* 匿名对象:创建的对象是匿名的。
*
*
* 说明:
* 1. 只要在代码中看到了"new"关键字,那就表明要新创建一个对象、数组。
* 2. 匿名对象,因为没有变量名,所以只能调用一次。
public class PhoneTest1 {
public static void main(String[] args) {
new Phone().call();//创建了对象1
new Phone().sendMessage("好久不见,甚是想念");//创建了对象2
//还可以调用对象1吗?No
PhoneTest1 test = new PhoneTest1();
//Phone p1 = new Phone();
//test.show(p1);
//如果后续代码中不再使用p1,则可以简化如下:
//简化如下:开发中匿名对象的使用场景
test.show(new Phone());
}
public void show(Phone phone){
phone.call();
phone.sendMessage("你好!");
phone.name = "OPPO v1";
System.out.println(phone.name);
}
}
四、类的属性
1. 基本说明
* 1.回忆:变量的分类: 角度一:数据类型来分: 基本数据类型(8种) vs 引用数据类型(类、数组、接口)
* 角度二:在类中声明位置的不同来分: 成员变量 vs 局部变量
*
* 2. 成员变量 = 属性 = field = 字段、域
2. 对比成员变量 与 局部变量
3.1 相同点:
* ① 变量声明的格式相同: 数据类型 变量名 = 初始化值
* ② 变量必须先声明、后初始化、再使用。
* ③ 变量都有其对应的作用域。只在其作用域内是有效的
*
* 3.2不同点:
* ① 类中声明位置是不同的。
* >成员变量:直接声明在类中的变量
* >局部变量:声明在方法内部、方法的形参、构造器内部、代码块的内部
* ② 关于权限修饰符的使用
* >成员变量:在声明的类型前,可以使用权限修饰符。
* 权限修饰符有:public \ private \ protected \ 缺省
* 当前大家声明属性时,先使用缺省的情况即可。
* >局部变量:不可以在其类型前使用权限修饰符。
* ③ 关于初始化值的问题
* >成员变量:如果没有显式赋值,则成员变量根据其类型有默认初始化值。
* >整型的成员变量:byte\short\int\long 型数组的元素的默认值为: 0
* >浮点型的成员变量:float\double 型数组的元素的默认值为:0.0
* >布尔型的成员变量:boolean 型数组的元素的默认值为:false
* >字符型的成员变量:char 型数组的元素的默认值为:0或\u0000
* >引用数据类型的成员变量:数组的元素的默认值为:null
*
* >局部变量:在使用之前,一定要显式的初始化,没有默认初始化值。
* 注意:形参在声明时不赋值。在调用此方法时,显式的赋值。
* ④ 在内存中保存的位置也不同
* >成员变量:在堆空间中分配内存
*
* >局部变量:在栈空间中分配内存
成员变量
class Person{
//声明的变量,即为成员变量
public String name;//默认初始化
int age = 1;//显式初始化
boolean isMale;//默认初始化
}
局部变量
public void eat(String food){//形参:属于局部变量的一种
System.out.println("喜欢吃" + food);
}
public void sleep(){
int hour = 8;//声明了局部变量
System.out.println("保证睡眠" + hour +"个小时");
}
说明
public void info(){
age = 2;//在方法内可以使用成员变量
// hour = 10;//不能调用其他方法内部的局部变量
}
在方法内可以使用类的成员变量。
但是在方法内不能使用其它方法内定义的局部变量
五、类的方法
1. 基本说明
* 1. 使用举例:
* Math:random()、sqrt()
* Scanner:next()、nextXxx()
* Arrays:binarySearch()、equals()、sort()、toString()
*
* 2.声明举例
* public void sleep();
* public void eat(String food);
* public String getName();
* private String showNation(String nation);
2. 方法的声明格式
* 方法声明的格式:
* 权限修饰符 返回值类型 方法名(形参列表){
* //方法体
* }
*
* 说明:关于方法中声明static\final\abstract\native...等暂时先不考虑
3. 具体的内容
3. 具体的方法声明的细节
* 3.1 权限修饰符
* > 在每个方法声明的开始,可以使用权限修饰符进行修饰,表明此方法被调用时的范围的大小。
* > 权限修饰符有:public \ private \ protected \ 缺省
* > 暂时大家在讲封装性之前,声明方法的权限修饰符暂且都使用public
* 3.2 返回值类型 :方法在调用完以后,是否需要给方法的调用者返回相应的数据。
* > 分类: 没有返回值的类型 vs 有返回值的类型
* > 没有返回值的类型,使用void表示。 举例:Arrays类中的sort(int[] arr)
* > 有返回值的类型,需要指明返回值的类型。举例:Math:random()、sqrt(double i)
* > 对于有返回值类型的方法来说,在方法体内部一定要返回相应类型的数据。使用return来实现。
*
* > 问题:我们在声明方法时,如何判断是否需要有返回值类型?
* ① 看题目的要求
* ② 具体问题具体分析
* 3.3 方法名:是标识符。遵循标识符命名的规则和规范。见名知意。
*
* 3.4 形参列表:属于局部变量
* >分类:有形参列表 vs 没有形参列表的方法
*
* >问题:我们在声明方法时,是否需要声明形参列表?
* ① 看题目的要求
* ② 具体问题具体分析
* 3.5 方法体: 当调用一个方法时,其实就是执行其方法体。
public class UserTest {
public static void main(String[] args) {
UserTest test = new UserTest();
test.show();
User user = new User();
String s = user.showInfo();
System.out.println(s);
}
}
class User{
String name;
int age;
//This method must return a result of type boolean
public boolean isMale(){
// return 0;
return true;
}
public void eat(){
}
public String showInfo(){
if(age >= 18){
return "成年人";
}else{
return "未成年人";
}
//编译不通过
//System.out.println("hello");
}
public double sqrt(double d){
return Math.sqrt(d);
}
// public double sqrt(){
// double d = 123.4;
// return Math.sqrt(d);
// }
}
4. return关键字的使用
关键字:return
* 作用: ① 结束一个方法
* ② 结束一个方法的同时,可以返回数据给方法的调用者
*
* 说明:在return关键字的直接后面不能声明执行语句
public void show(){
String str = "hello";
if("hello".equals(str)){
System.out.println("hi~~~");
return;
}
System.out.println("你好");
}
public String showInfo(){
if(age >= 18){
return "成年人";
}else{
return "未成年人";
}
//编译不通过
//System.out.println("hello");
}
public double sqrt(double d){
return Math.sqrt(d);
}
// public double sqrt(){
// double d = 123.4;
// return Math.sqrt(d);
// }
5. 补充说明
1. 在方法内可以使用类的成员变量。但是在方法内不能使用其它方法内定义的局部变量
2. 在方法内,可以调用其他方法。但是在方法内,不允许定义方法。
public void eat(){
System.out.println("吃饭");
//在方法内可以调用其他方法
showInfo();
//方法内不可以定义方法
// public void info(){
//
// }
}
public String showInfo(){
}
六、相关练习
练习1:
创建一个Person类,其定义如下:
要求:(1)创建Person类的对象,设置该对象的name、age和sex属性,调用study方法,输出字符串“studying”,调用showAge()方法显示age值,调用addAge()方法给对象的age属性值增加2岁。
(2)创建第二个对象,执行上述操作,体会同一个类的不同对象之间的关系。
public class Person {
String name;
int age;
int sex;//性别 0:男性 1:女性
public void study(){
System.out.println("studying");
}
public void showAge(){
System.out.println("age : " + age);
}
public int addAge(int i){//i:给age属性增加i岁
age += i;
return age;
}
}
public class PersonTest {
public static void main(String[] args) {
Person p1 = new Person();
//对象调用属性
p1.name = "王浩波";
p1.age = 32;
p1.sex = 0;
//对象调用方法
p1.study();
p1.showAge();
p1.addAge(2);
System.out.println(p1.age);//34
//**********************************
//如果创建一个类的多个对象,则每个对象都拥有一套类中的属性的“副本”。
//如果修改其中一个对象的属性值,不会影响到其他对象此属性的值。
Person p2 = new Person();
p2.showAge();
p2.addAge(10);
p2.showAge();
}
}
练习2:
利用面向对象的编程方法,设计类Circle计算圆的面积。
public class Circle {
//属性:半径
double radius;
//计算圆的面积:pi * r * r
//正确的方式一
// public double findArea(){
//
// return 3.14 * radius * radius;
// }
//正确的方式二
public void findArea(){
double area = 3.14 * radius * radius;
System.out.println(area);
// return;
}
//错误的写法
// public double findArea(double r){
// return 3.14 * r * r;
// }
}
public class CircleTest {
public static void main(String[] args) {
Circle c1 = new Circle();
c1.radius = 2.3;
//对应正确的写法一:
// double area = c1.findArea();
// System.out.println("c1的面积为:" + area);
//对应正确的写法二:
c1.findArea();
//对应错误的写法
// double area = c1.findArea(2.3);
// System.out.println(area);
}
}
练习3:
3.1 编写程序,声明一个method方法,在方法中打印一个10*8 的*型矩形,在main方法中调用该方法。
3.2 修改上一个程序,在method方法中,除打印一个10*8的*型矩形外,再计算该矩形的面积,并将其作为方法返回值。在main方法中调用该方法,接收返回的面积值并打印。
3.3 修改上一个程序,在method方法提供m和n两个参数,方法中打印一个m*n的*型矩形,并计算该矩形的面积, 将其作为方法返回值。在main方法中调用该方法,接收返回的面积值并打印。
练习4:
对象数组题目:
定义类Student,包含三个属性:学号number(int),年级state(int),成绩score(int)。 创建20个学生对象,学号为1到20,年级和成绩都由随机数确定。
问题一:打印出3年级(state值为3)的学生信息。
问题二:使用冒泡排序按学生成绩排序,并遍历所有学生信息
提示:
1) 生成随机数:Math.random(),返回值类型double;
2) 四舍五入取整:Math.round(double d),返回值类型long。
/*
* 定义类Student,包含三个属性:学号number(int),年级state(int),成绩score(int)。
*/
public class Student {
int number;//学号
int state;//年级
int score;//成绩
public void info(){
System.out.println("number : " + number
+ ",state : " + state + ",score : " + score);
}
}
public class StudentTest {
public static void main(String[] args) {
// //....
// 对象数组
// String[] arr = new String[10];
// 数组的创建
Student[] students = new Student[20];
// 通过循环结构给数组的属性赋值
for (int i = 0; i < students.length; i++) {
// 数组元素的赋值
students[i] = new Student();
// 数组元素是一个对象,给对象的各个属性赋值
students[i].number = (i + 1);
students[i].state = (int) (Math.random() * 6 + 1);// [1,6]
students[i].score = (int) (Math.random() * 101);// [0,100]
}
// 问题一:打印出3年级(state值为3)的学生信息。
for (int i = 0; i < students.length; i++) {
if (students[i].state == 3) {
// System.out.println(
// "number:" + students[i].number + ",state:" + students[i].state + ",score:" + students[i].score);
students[i].info();
}
}
System.out.println("******************************");
// 问题二:使用冒泡排序按学生成绩排序,并遍历所有学生信息
// 排序前
for (int i = 0; i < students.length; i++) {
// System.out.println(
// "number:" + students[i].number + ",state:" +
// students[i].state + ",score:" + students[i].score);
students[i].info();
}
System.out.println();
// 排序:
for (int i = 0; i < students.length - 1; i++) {
for (int j = 0; j < students.length - 1 - i; j++) {
if (students[j].score > students[j + 1].score) {
Student temp = students[j];
students[j] = students[j + 1];
students[j + 1] = temp;
}
}
}
// 排序后:
for (int i = 0; i < students.length; i++) {
// System.out.println(
// "number:" + students[i].number + ",state:" +
// students[i].state + ",score:" + students[i].score);
students[i].info();
}
}
}
练习5:
声明一个日期类型MyDate:有属性:年year,月month,日day。创建2个日期对象,分别赋值为:你的出生日期,你对象的出生日期,并显示信息。
七、方法的重载(overload)
* 1. 定义:在同一个类中,允许存在一个以上的同名方法,只要它们的参数个数或者参数类型不同即可。
* 此时,多个同名的方法之间就称为方法的重载。
*
* 2. 总结为:两同一不同。(同一个类,相同的方法名,参数列表不同)
* > 参数列表不同,指参数个数或者参数类型不同
*
*
* 3. 如何判断两个方法是相同的呢?换个问法,如何编译器确定调用的是某个指定方法呢?
*
* ① 参照方法名
* ② 参数列表
*
* 4. 方法的重载与权限修饰符、返回值类型、形参的名字都没有关系!!
//如下的三个方法构成重载
public void getSum(int i,int j){
System.out.println(i + j);
}
public void getSum(int i,byte b){
System.out.println(i + b);
}
public void getSum(int i,int j ,int k){
}
//如下的两个方法不与上述三个方法构成重载
// public void getSum(int i,int k){
//
// }
// public int getSum(int i ,int k){
//
// }
面试题:
//面试题
public class InterviewTest {
public static void main(String[] args) {
int[] arr = new int[]{1,2,3};
System.out.println(arr);//地址值
char[] arr1 = new char[]{'a','b','c'};
System.out.println(arr1);//abc
boolean[] arr2 = new boolean[]{false,true,true};
System.out.println(arr2);//地址值
}
}
* 3. 如何判断两个方法是相同的呢?换个问法,如何编译器确定调用的是某个指定方法呢?
*
* ① 参照方法名
* ② 参数列表
*
* 4. 方法的重载与权限修饰符、返回值类型、形参的名字都没有关系!!
//如下的三个方法构成重载
public void getSum(int i,int j){
System.out.println(i + j);
}
public void getSum(int i,byte b){
System.out.println(i + b);
}
public void getSum(int i,int j ,int k){
}
//如下的两个方法不与上述三个方法构成重载
// public void getSum(int i,int k){
//
// }
// public int getSum(int i ,int k){
//
// }
面试题:
//面试题
public class InterviewTest {
public static void main(String[] args) {
int[] arr = new int[]{1,2,3};
System.out.println(arr);//地址值
char[] arr1 = new char[]{'a','b','c'};
System.out.println(arr1);//abc
boolean[] arr2 = new boolean[]{false,true,true};
System.out.println(arr2);//地址值
}
}
八、练习题
练习1:
/*编写程序,定义三个重载方法并调用。方法名为mOL。
三个方法分别接收一个int参数、两个int参数、一个字符串参数。
分别执行平方运算并输出结果,相乘并输出结果,输出字符串信息。
*/
public class OverloadTest1 {
public static void main(String[] args) {
OverloadTest1 o1 = new OverloadTest1();
System.out.println(o1.mOL(4));
System.out.println(o1.mOL(4, 7));
o1.mOL("hello");
}
public int mOL(int i) {
return i * i;
}
public int mOL(int i, int j) {
return i * j;
}
public void mOL(String i) {
System.out.println(i);
}
}
练习2:
/*
* 定义三个重载方法max(),
* 第一个方法求两个int值中的最大值,
* 第二个方法求两个double值中的最大值,
* 第三个方法求三个double值中的最大值,并分别调用三个方法。
*/
public class OverloadTest2 {
public static void main(String[] args) {
OverloadTest2 o1 = new OverloadTest2();
System.out.println(o1.max(3, 4));
System.out.println(o1.max(3.8, 4.6));
System.out.println(o1.max(8.9, -7.0, 15.5));
}
public int max(int i, int j) {
// if (i > j) {
// return i;
// } else {
// return j;
// }
return (i > j) ? i : j;
}
public double max(double i, double j) {
// if (i > j) {
// return i;
// } else {
// return j;
// }
return (i > j) ? i : j;
}
public double max(double i, double j, double k) {
double max1 = (i > j) ? i : j;
double max2 = (max1 > k) ? max1 : k;
return max2;
}
}
作业3:
画内存结构图,并指出如下语句的执行结果:
Student[] stus = new Student[5];
stus[0] = new Student();
sysout(stus[0].state);//0
sysout(stus[1]);//null
sysout(stus[1].number);//NullPointerException
stus[1] = new Student();
sysout(stus[1].number);//0
说明:通过"对象.属性"或"对象.方法"的方式,实现相应功能的调用。但是只要调用属性或方法的对象值为null,则一定报NullPointerExcepiton。
知识点:对于引用类型的变量来说,要么存储的是null值,要么存储的是其对象、数组实体在堆空间的地址值。
作业4:
声明一个日期类型MyDate:有属性:年year,月month,日day。创建2个日期对象,分别赋值为:你的出生日期,你对象的出生日期,并显示信息。
public class MyDateTest {
public static void main(String[] args) {
MyDate myDate = new MyDate();
myDate.year = 1992;
myDate.month = 5;
myDate.day = 9;
System.out.println("我的生日:" + myDate.toString());
MyDate girlFriendDate = new MyDate();
myDate.year = 1997;
myDate.month = 10;
myDate.day = 2;
System.out.println("女朋友的生日:" + myDate.toString());
}
}
class MyDate {
int year;
int month;
int day;
public String toString() {
return "year: " + year + " month: " + month + " day:" + day;
}
}
九、小结
-
类与对象的关系,以及如何实例化:类型 对象名 = new 类型();
- 通过"对象.属性"或"对象.方法"的方式,实现相应功能的调用
-
类中:属性的设计
- 变量分类:成员变量(或属性) vs 局部变量(形参、方法内定义的变量、构造器内定义的变量等)
- 成员变量(或属性) vs 局部变量的相同点和区别(声明的位置、权限修饰符、初始化值、内存的分配位置)
-
类中:方法的设计
- 一个类的功能是否强大,主要取决于方法的定义的多少!
- 掌握方法的声明的各个部分:权限修饰符、返回值类型、方法名、形参列表、方法体。
-
return 关键字
-
方法的重载 —>如何明确调用指定的方法:① 方法名 ② 方法的形参列表