大数据之路week01--day02我实在被继承super这些东西搞的头疼,今天来好好整理以下。
这一周的第一天的内容是面向对象的封装,以及对方法的调用。实在时没法单独拿出来单说,就结合这一节一起说了。
我实在是被继承中的super用法给弄的有点晕,程序总是不能按照我想的那样,不是说结果,而是实现过程,很明显复杂,后来进行推敲后,原来是我的理解出了差错。
先把前对象的的内容补补
在开始面向对象之前,我们先来了解一下,Java的内存分配:(在之前我也说过,只是简单的提到)
Java把内存划分为4个部分 1. 代码区 1、栈区 3、堆区 4、静态区域
1、栈区(stacksegment)—由编译器自动分配释放,存放函数的参数值,局部变量的值等,具体方法执行结束之后,系统自动释放JVM内存资源
2、堆区(heapsegment)—一般由程序员分配释放,存放由new创建的对象和数组,jvm不定时查看这个对象,如果没有引用指向这个对象就回收
3、静态区(datasegment)—存放全局变量,静态变量和字符串常量,不释放
4、代码区(codesegment)—存放程序中方法的二进制代码,而且是多个对象共享一个代码空间区域
一、面向对象的三大特性:
封装、继承、多态。
1、封装
在面向对象程式设计方法中,封装(英语:Encapsulation)是指一种将抽象性函式接口的实现细节部份包装、隐藏起来的方法。
封装可以被认为是一个保护屏障,防止该类的代码和数据被外部类定义的代码随机访问。
要访问该类的代码和数据,必须通过严格的接口控制。
封装最主要的功能在于我们能修改自己的实现代码,而不用修改那些调用我们代码的程序片段。
适当的封装可以让程式码更容易理解与维护,也加强了程式码的安全性。
1.1 封装的优点
-
1. 良好的封装能够减少耦合。
-
2. 类内部的结构可以自由修改。
-
3. 可以对成员变量进行更精确的控制。
-
4. 隐藏信息,实现细节。
1.2 实现Java封装的步骤
1. 修改属性的可见性来限制对属性的访问(一般限制为private),例如:
这段代码中,将 name 和 age 属性设置为私有的,只能本类才能访问,其他类都访问不了,如此就对信息进行了隐藏。
2. 对每个值属性提供对外的公共方法访问,也就是创建一对赋取值方法,用于对私有属性的访问,例如:
1 public class Person{ 2 private String name; 3 private int age; 4 public int getAge(){ 5 return age; 6 } 7 public String getName(){ 8 return name; 9 } 10 public void setAge(int age){ 11 this.age = age; 12 } 13 public void setName(String name){ 14 this.name = name; 15 } 16 }
采用 this 关键字是为了解决实例变量(private String name)和局部变量(setName(String name)中的name变量)之间发生的同名的冲突。
1.3 this
this指针:(实际上,每个非static方法中都隐含这一个this指针指向当前正在调用该方法的对象。)
1.4 static
static(静态的):
1、如果在一个属性的前面加上static ,那么这个属性就不是某一个对象的了,而是N个对象共用的了。在静态区存放,不释放。
static属性变量是属于类本身的,没有对象我们仍然可以通过类名的方式进行访问该类内部的static属性。(static方法也可以)
注意:类可以直接调用,实例化出来的对象当然也可以进行调用。
理解:模型都可以,模型造的对象当然可以。
2、加上static的确是属于类本身的,但是要去访问它必须控制符是非私有的,如果控制符是私有的,则不能直接通过类名进行访问!!
理解:private static void a(){} private static int i;都是不可以直接通过类名进行访问,因为前面的控制符是private
3、非静态的方法可以访问静态成员,反之不可以。非私有的静态成员才能通过类名直接调用。
理解:静态的方法和静态的变量属于类本身,不是静态的方法必须通过对象进行调用,在实例化对象的同时也包含了静态的方法,当然可以。反之不可以。
下面,我们用实例来理解static的作用:
实例1:实现一个类只允许有一个对象
1 package com.wyh.Iterator; 2 3 /** 4 * @author WYH 5 * @version 2019年11月10日 上午10:18:07 6 * 该程序证明: 7 * 实现一个类只允许实例化一个对象 8 */ 9 10 class A2{ 11 public int i = 2; 12 13 private static A2 aa = new A2(); 14 15 private A2() { 16 17 } 18 19 public static A2 getA2() { 20 return aa; 21 } 22 } 23 24 public class static03 { 25 public static void main(String[] args) { 26 A2 aa1 = A2.getA2(); 27 A2 aa2 = A2.getA2(); 28 aa1.i = 90; 29 System.out.println("aa2:"+aa2.i);//说明aa1和aa2是同一个对象 30 31 if(aa1 == aa2) { 32 System.out.println("说明aa1和aa2是同一个对象"); 33 }else { 34 System.out.println("说明aa1和aa2不是同一个对象"); 35 } 36 } 37 38 }
实例2:一个类无论实例化多少对象,其中一个变量只有一个。
1 package com.wyh.Iterator; 2 3 /** 4 * @author WYH 5 * @version 2019年11月10日 上午9:32:57 6 */ 7 public class static01 { 8 private static int i; 9 10 public static01() { 11 i = i+1; 12 } 13 14 // static void add() { 15 // i = i+1; 16 // } 17 // 18 19 public static void main(String[] args) { 20 static01 s1 = new static01(); 21 // s1.add(); 22 static01 s2 = new static01(); 23 // s2.add(); 24 static01 s3 = new static01(); 25 // s3.add(); 26 static01 s4 = new static01(); 27 // s4.add(); 28 static01 s5 = new static01(); 29 // s5.add(); 30 System.out.println("总共实例化了:"+i+"个对象"); 31 } 32 33 }
2、继承
什么是继承?在我们的后面实际操作过程中,很多时候,许多的对象拥有一些重复的操作,而且,这些对象都是同一种现实属性的派生。例如,我们的男人和女人都是人类,那么人类就是男人,女人的父类,男人和女人继承了人类这个父类,称之为子类。
1、继承,我们用extends来表示:比如刚刚说的男人继承了人类这个类 public class man extends human{}
2、提到了继承父类,那继承过来的成员变量和一些方法,我们怎么去使用呢?
2.1 super
终于讲到令人头疼的super了。
重写:当子类需要父类的功能,而子类有新的内容,可以重写父类中的方法。在实际开发过程中,随着代码量的逐渐增加,维护成了一个很大的问题,如果需要对某个方法进行修改,其本身代码以及其子类代码都会受到影响,而重写则很好的解决了这个问题。方法重写又称为方法覆盖、方法复写。
方法重写特点
在子类和父类中,出现了方法声明相同的情况
子类的方法声明要和父类相同
子类要重写的方法,方法的权限修饰符不能比父类更低(public 、protected 、default 、private 权限依次增加)
父类私有的方法,子类不能进行方法重写
方法重写和方法重载的区别:
方法重写:子类和父类中方法相同,两个类之间的关系,函数的返回值类型、函数名、参数列表都一样
方法重载:指在同一个类中,多个方法名相同,他们的参数列表不同(个数不同,数据类型不同)
重写方法时要注意的地方:
1.方法重写时, 方法名与形参列表必须一致。
2.方法重写时,子类的权限修饰符必须要大于或者等于父类的权限修饰符。
3.方法重写时,子类的返回值类型必须要小于或者 等于父类的返回值类型。
4.方法重写时, 子类抛出的异常类型要小于或者等于父类抛出的异常类型。 Exception(最坏) RuntimeException(小坏)
先来看一个例子:
A类:(父类)
1 package day02; 2 3 /** 4 * @author WYH 5 * @version 2019年11月12日 下午2:52:37 6 */ 7 public class A { 8 9 protected String name; 10 protected String health; 11 protected String love; 12 13 14 15 public String getName() { 16 return name; 17 } 18 public String setName(String name) { 19 return this.name = name; 20 } 21 public String getHealth() { 22 return health; 23 } 24 public void setHealth(String health) { 25 this.health = health; 26 } 27 public String getLove() { 28 return love; 29 } 30 public void setLove(String love) { 31 this.love = love; 32 } 33 public A() { 34 super(); 35 // TODO Auto-generated constructor stub 36 } 37 public void eat() 38 { 39 System.out.println("吃饭!"); 40 } 41 public void work() 42 { 43 System.out.println("工作!"); 44 } 45 public void sleep() 46 { 47 System.out.println("睡觉!"); 48 } 49 50 }
B类:(子类,继承了A父类)
1 package day02; 2 3 /** 4 * @author WYH 5 * @version 2019年11月12日 下午2:53:02 6 */ 7 public class B extends A{ 8 private int i; 9 10 11 12 public int getI() { 13 return i; 14 } 15 16 public void setI(int i) { 17 this.i = i; 18 } 19 20 public B(){ 21 super(); 22 } 23 24 public void work() 25 { 26 super.work(); //使用super关键字调用父类方法 27 System.out.println("学习!"); 28 System.out.println("娱乐!"); 29 30 } 31 }
D类:(子类,继承了A父类)
1 package day02; 2 3 /** 4 * @author WYH 5 * @version 2019年11月12日 下午2:53:02 6 */ 7 public class D extends A{ 8 private int i; 9 10 11 12 public int getI() { 13 return i; 14 } 15 16 public void setI(int i) { 17 this.i = i; 18 } 19 20 public D(){ 21 super(); 22 } 23 24 public void work() 25 { 26 super.work(); //使用super关键字调用父类方法 27 System.out.println("学习!"); 28 System.out.println("娱乐!"); 29 30 } 31 }
C类:(测试类,主类)
1 package day02; 2 3 /** 4 * @author WYH 5 * @version 2019年11月12日 下午2:53:34 6 */ 7 public class C { 8 public static void main(String[] args) { 9 /*A aa = new A(); 10 aa.work();*/ 11 12 A bb = new B(); 13 A dd = new D(); 14 bb.name = bb.setName("456"); 15 dd.name = dd.setName("123"); 16 System.out.println(bb.name); 17 System.out.println(dd.name); 18 System.out.println(bb.getName()); 19 bb.work(); 20 } 21 22 }
解说:
我们可以看到,我现在父类,A类种定义了name;health;love;三个成员变量,为了可以实现继承,我们不将它定义成私有变量,同时父类里面拥有三个方法,吃饭,睡觉,工作。
B类和D类继承了A类,同时我们将work方法进行了重写,添加了自己特有的方法。
然后,我们来看看结果:
后来,我们发现,通过super,可以调用父类的方法,以及可以通过super调用父类的成员变量。这个例子很好的概括,this指针,super的用法,重写的用法。
多态我们后续进行补充。。。