类型转换问题
显示转换:
double b = 1.77;
int a = (int)b;//a = 1,只会向下取整,不会四舍五入。
隐式转换:
int a = 1;
double b = a;
break 与 continue
1、必须存在循环中,如果上层没有循环会报错:break cannot be used outside of a loop or a switch。2、当存在嵌套,break中断的是上层循环。
break:中断当前循环。
for(int i=0;i<100;i++)
{
if(i >= 10)
break;
System.out.println(i);
}//输出0~9
continue:跳过本轮循环。
for(int i=0;i<20;i++)
{
if(i % 4 == 0)
continue;
System.out.println(i);
}
数组
int[] arr = new int[30];//数组创建,default都为0.
int[] arr = {98,89,78,87};//每个值之间用','隔开,用"{}"括起。
对象与引用
先看一个例子,
int[] a = { 11, 11, 11, 11 };
int[] b = a;
b[3] = 45;
for(int i=0; i<a.length; i++)
{
System.out.print( a[i] + " ");
}
代码输出 11,11,11,45.
因为b和a都是该对象的引用,a和b指向相同的对象。
在Java语言里,对象Object是一个基础概念。
int[] a = new int[4];
等号右侧:创建了一个数组对象
等号左侧: 变量 a 称为该对象的引用( Reference)
称作: 变量a 指向了一个对象
有时也简称为:a是一个对象, 或a是对象的名字。
失去引用的对象-体现回收机制
int[] a = { 8, 8, 8 };
a = new int[4];
a[0] = a[1] = a[2] = a[3] = 17;
① 创建一个数组对象obj1,内容为8,8,8
② 创建另一个数组对象obj2, 内容0,0,0,0
obj2的内容变为 17,17,17,17
此时,对象obj1无人引用。对象obj2被a引用。
失去引用的对象,会被Java运行环境自动删除
( 垃圾自动回收 GC )
new: 创建一个对象
何时删除:当一个对象不再被使用时(失去引用),会被Java系统自动删除。
Getter and Setter
一键添加
Getter and Setter: 右键 -> Sourse -> Getter and Setter
重写
调用父类方法:
super.function();
一键添加重写:右键 -> Sourse -> Override
构造方法的继承
创建子类对象,先调用父类构造方法,再调用子类构造方法。
父类引用指向子类对象(转载)
方法的重写、重载与动态连接构成多态性。Java之所以引入多态的概念,原因之一是它在类的继承问题上和C++不同,后者允许多继承,这确实给其带来的非常强大的功能,但是复杂的继承关系也给C++开发者带来了更大的麻烦,为了规避风险,Java只允许单继承,派生类与基类间有IS-A的关系(即“猫”is a “动物”)。这样做虽然保证了继承关系的简单明了,但是势必在功能上有很大的限制,所以,Java引入了多态性的概念以弥补这点的不足,此外,抽象类和接口也是解决单继承规定限制的重要手段。同时,多态也是面向对象编程的精髓所在。
理解多态,首先要知道“向上转型”。
我定义了一个子类Cat,它继承了Animal类,那么后者就是前者是父类。我可以通过
Cat c = new Cat();
实例化一个Cat的对象,这个不难理解。但当我这样定义时:
Animal a = new Cat();
这代表什么意思呢?
很简单,它表示我定义了一个Animal类型的引用,指向新建的Cat类型的对象。由于Cat是继承自它的父类Animal,所以Animal类型的引用是可以指向Cat类型的对象的。这就是“向上转型”。
那么这样做有什么意义呢?因为子类是对父类的一个改进和扩充,所以一般子类在功能上较父类更强大,属性较父类更独特, 定义一个父类类型的引用指向一个子类的对象既可以使用子类强大的功能,又可以抽取父类的共性。 所以,父类类型的引用可以调用父类中定义的所有属性和方法,而对于子类中定义而父类中没有的方法,父类引用是无法调用的;
那什么是动态链接呢?当父类中的一个方法只有在父类中定义而在子类中没有重写的情况下,才可以被父类类型的引用调用; 对于父类中定义的方法,如果子类中重写了该方法,那么父类类型的引用将会调用子类中的这个方法,这就是动态连接。
下面是一个典型例子:
class Father{
public void func1(){
func2();
}
//这是父类中的func2()方法,因为下面的子类中重写了该方法
//所以在父类类型的引用中调用时,这个方法将不再有效
//取而代之的是将调用子类中重写的func2()方法
public void func2(){
System.out.println("AAA");
}
}
class Child extends Father{
//func1(int i)是对func1()方法的一个重载,主要不是重写!
//由于在父类中没有定义这个方法,所以它不能被父类类型的引用调用
//所以在下面的main方法中child.func1(68)是不对的
public void func1(int i){
System.out.println("BBB");
}
//func2()重写了父类Father中的func2()方法
//如果父类类型的引用中调用了func2()方法,那么必然是子类中重写的这个方法
public void func2(){
System.out.println("CCC");
}
}
public class PolymorphismTest {
public static void main(String[] args) {
Father child = new Child();
child.func1();//打印结果将会是什么?
child.func1(68);
}
}
上面的程序是个很典型的多态的例子。子类Child继承了父类Father,并重载了父类的func1()方法,重写了父类的func2()方法。重载后的func1(int i)和func1()不再是同一个方法,由于父类中没有func1(int i),那么,父类类型的引用child就不能调用func1(int i)方法。而子类重写了func2()方法,那么父类类型的引用child在调用该方法时将会调用子类中重写的func2()。
那么该程序将会打印出什么样的结果呢?
很显然,应该是“CCC”。
对于多态,可以总结以下几点:
一、使用父类类型的引用指向子类的对象;
二、该引用只能调用父类中定义的方法和变量;
三、如果子类中重写了父类中的一个方法,那么在调用这个方法的时候,将会调用子类中的这个方法;(动态连接、动态调用)
四、变量不能被重写(覆盖),”重写“的概念只针对方法,如果在子类中”重写“了父类中的变量,那么在编译时会报错。