1.方法的签名和重载`
// >> TODO 重载的方法可以调用别的重载方法,当然也可以调用别的不重载的方法。
// >> TODO 实际上,像这种补充一些缺省的参数值,然后调用重载的方法,是重载的一个重要的使用场景。
// >> TODO 在这里我们举的例子就是这样的,但是不是语法要求一定要这样。重载的方法的方法体内代码可以随便写,
// TODO 可以不调用别的重载方法
public double buy() {
return buy(1);
}
public double buy(int count) { return buy(count, false);
}
// TODO 最后都补充好参数,调用参数最全的一个方法
public double buy(int count, boolean isVIP) {
if (this.count < count) {
return -1;
}
this.count -= count;
double totalCost = count * soldPrice;
if (isVIP) {
return totalCost * 0.95;
} else {
return totalCost;
}
}
在该类里面,有三个签名:buy(),buy(int count)和buy(int count, boolean isVIP)。这就叫方法重载。;
2.重载的参数匹配规则
public class MerchandiseV2AppMain {
public static void main(String[] args) {
MerchandiseV2 merchandise = new MerchandiseV2();
merchandise.init("书桌", "DESK9527", 40, 999.9, 500);
// >> TODO 使用int调用参数为double的方法
int count = 3;
// merchandise.buyDouble(count);
System.out.println("测试使用不完全匹配的参数调用重载方法");
// >> TODO 依次使用byte, short, int, long, float, double 类型的参数调用buy方法,哪个方法会被调用呢?
// >> TODO 无论是否重载参数类型可以不完全匹配的规则是"实参数可以自动类型转换成形参类型"
// >> TODO 重载的特殊之处是,参数满足自动自动类型转换的方法有好几个,重载的规则是选择最"近"的去调用
double countForOverride = 11;
merchandise.buy(countForOverride);
}
}
在调用有重载的类的方法的时候(其他类的方法也一样),并不是需要所有参数类型都匹配才会调用方法,在上面的代码块的TODO 中,说到,会按照byte,short ,long,float,double这样的顺序去调用,也就是说如果参数是byte类型,但是方法体有short和double类型时,会先自动调用short类型的方法体;
3.构造方法
public class MerchandiseV2WithConstructor {
public String name;
public String id;
public int count;
public double soldPrice;
public double purchasePrice;
// >> TODO 构造方法(constructor)的方法名必须与类名一样,而且构造方法没有返回值。这样的方法才是构造方法。
// >> TODO 构造方法可以有参数,规则和语法于普通方法一样。使用时,参数传递给 new 语句后类名的括号后面。
// >> TODO 如果没有显示的添加一个构造方法,Java会给每个类都会默认自带一个无参数的构造方法。
// TODO 如果我们自己添加类构造方法,Java就不会再添加无参数的构造方法。这时候,就不能直接 new 一个对象不传递参数了(看例子)
// >> TODO 所以我们一直都在使用构造方法,这也是为什么创建对象的时候类名后面要有一个括号的原因。
// >> TODO 构造方法无法被点操作符调用或者在普通方法里调用,只能通过 new 语句在创建对象的时候,间接调用。
// TODO 理解一下为什么构造方法不能有返回值,因为有返回值也没有意义,new 语句永远返回的是创建出来的对象的引用
public MerchandiseV2WithConstructor(String name, String id, int count,
double soldPrice, double purchasePrice) {
this.name = name;
this.id = id;
this.count = count;
this.soldPrice = soldPrice;
this.purchasePrice = purchasePrice;
}
在我们new一个对象的时候,会有一个括号:
MerchandiseV2WithConstructor merchandise = new MerchandiseV2WithConstructor();
其实这个括号就是用来给构造方法传参数的;
只是一般用java官方类的时候,都默认不用传参数,
但是我们自己定义构造方法的时候,java就不会再添加无参数的构造方法
这样我们在创建对象的时候,就要给构造方法传参数了;
public class RunLittleSupperMarketAppMain {
public static void main(String[] args) {
// 创建一个小超市类
LittleSuperMarket littleSuperMarket = new LittleSuperMarket(
"有家小超市", "浦东新区世纪大道666号",
100, 200, 200);
System.out.println("下面请利润最高的商品自我介绍:");
littleSuperMarket.getBiggestProfitMerchandise().describe();
}
}
构造方法也可以重载
public class MerchandiseV2 {
public String name;
public String id;
// >> TODO 构造方法执行前,会执行给局部变量赋初始值的操作
// >> TODO 我们说过,所有的代码都必须在方法里,那么这种给成员变赋初始值的代码在哪个方法里?怎么看不到呢?
// TODO 原来构造方法在内部变成了<init>方法。学习就是要脑洞大,敢想敢试,刨根问底。
public int count = 999;// 999/0;
public double soldPrice;
public double purchasePrice;
// >> TODO 构造方法(constructor)的重载和普通方法一样
public MerchandiseV2(String name, String id, int count, double soldPrice, double purchasePrice) {
this.name = name;
this.id = id;
this.count = count;
this.soldPrice = soldPrice;
this.purchasePrice = purchasePrice;
// soldPrice = 9/0;
}
// >> TODO 在构造方法里才能调用重载的构造方法。语法为this(实参列表)
// >> TODO 构造方法不能自己调用自己,这会是一个死循环
// >> TODO 在调用重载的构造方法时,不可以使用成员变量。因为用语意上讲,这个对象还没有被初始化完成,处于中间状态。
// >> TODO 在构造方法里才能调用重载的构造方法时,必须是方法的第一行。后面可以继续有代码
public MerchandiseV2(String name, String id, int count, double soldPrice) {
// double purPrice = soldPrice * 0.8;
// this(name, id, count, soldPrice, purchasePrice);
this(name, id, count, soldPrice, soldPrice * 0.8);
// double purPrice = soldPrice * 0.8;
}
// >> TODO 因为我们添加了构造方法之后,Java就不会再添加无参数的构造方法。如果需要的话,我们可以自己添加这样的构造方法
public MerchandiseV2() {
this("无名", "000", 0, 1, 1.1);
}
首先说一下为什么"子类的构造方法, 会强制在第一行调用父类的构造方法", 这个我在视频里提了一嘴, (人工划重点)这是因为子类可以访问父类的方法和属性. 那么必须保证子类的对象在做任何事情之前, 子类里那个隐含的父类对象, 必须初始化完成. 否则, 初始化都没完成, 调用方法, 返回属性, 都可能得到的是不对的状态/值(人工划重点结束).