热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Java面向对象编程核心概念解析与应用

本文深入解析了Java面向对象编程的核心概念及其应用,重点探讨了面向对象的三大特性:封装、继承和多态。封装确保了数据的安全性和代码的可维护性;继承支持代码的重用和扩展;多态则增强了程序的灵活性和可扩展性。通过具体示例,文章详细阐述了这些特性在实际开发中的应用和优势。
面向对象的特征

三大特征

package com.cloud.day2;

/*

面向对象三大特征:

   1. 封装

   2. 继承

   3  多态。

需求:  使用java类描述百合网的会员。

问题:性别有问题??


根本原因:由于其他人可以直接操作sex属性。可以对sex属性进行了直接的赋值。


封装:

权限修饰符:权限修饰符就是控制变量可见范围的。


public :  公共的。 public修饰的成员变量或者方法任何人都可以直接访问。

private 私有的, private修饰的成员变量或者方法只能在本类中进行直接访问。

封装的步骤:

   1. 使用private修饰需要被封装的属性。

   2. 提供一个公共的方法设置或者获取该私有的成员属性。

         命名规范:

           set属性名();

           get属性名();

 

疑问:封装一定要提供get或者set方法吗?

      不一定,根据需求而定的。

 

规范在现实开发中一般实体类的所有成员属性(成员变量)都要封装起来。

 

实体类:实体类就是用于描述一类事物的就称作为实体类。

工具类(Arrays数组的工具类):

 

封装的好处:

   1. 提高数据的安全性。

   2. 操作简单。

   3. 隐藏了实现。

*/

class Member{

   public String name;

   private String sex;

   public int salary;

   public void setSex(String s){

      if(s.equals("")||s.equals("")){

        sex = s;

      }

      else{

        sex = "";

      }

   }

   public String getSex(){

      return sex;

   }

   public void talk(){

      System.out.println("comunicate very happy....");

   }

}

public class Demo5 {

   public static void main(String[] args) {

      Member m = new Member();

      m.name = "Spring";

      m.setSex("");

      m.salary = 8888;

      System.out.println(m.name+";"+m.getSex()+";"+m.salary+";");

   }

}

计算案例

package com.cloud.day2;

 

/*

需求:使用java类描述一个计算器类,计算器具备操作数1操作数2 、操作符三个公共的属性

,还具备计算的功能行为。

要求:不能直接对操作数1,操作数2,运算符这些属性进行直接的赋值,要封装起来。 (+ - * /  )

根据需求提供get或者set方法。

   需要提供set方法

*/

class Calculator{

   private int num1;

   private int num2;

   private char option;

   public void initCalculator(int n1 , int n2 , char o){

      num1 = n1 ;

      num2 = n2 ;

      if(o==‘+‘||o==‘-‘||o==‘*‘||o==‘/‘){

        option = o;

      }else{

        option = ‘+‘;

      }

   }

   public void calculator(){

      switch (option) {

      case ‘+‘:

        System.out.println(num1+num2);

        break;

      case ‘-‘:

        System.out.println(num1-num2);

        break;

      case ‘*‘:

        System.out.println(num1*num2);

        break;

      case ‘/‘:

        System.out.println(num1/num2);

        break;

      }

   }

}

public class Demo6 {

   public static void main(String[] args) {

      Calculator c = new Calculator();

      c.initCalculator(1, 2, ‘+‘);

      c.calculator();

   }

}

操作数组

package com.cloud.day2;

import java.util.Arrays;

/*

需求:目前存在数组:int[] arr = {0,0,12,1,0,4,6,0} ,编写一个函数

接收该数组,然后把该数组的0清空,然后返回一个不存在0元素的数组。

 

步骤:

   1. 计算机新数组的长度。  原来的数组长度-0的个数

*/

public class Demo7 {

   public static void main(String[] args) {

      int[] arr = {0,1,2,3,0,4};

      arr = cleanZera(arr);

      System.out.println(Arrays.toString(arr));

   }

   public static int[] cleanZera(int []arr){

      int count = 0;

      for(int i=0;ilength;i++){

        if(arr[i]==0){

           count++;

        }

      }

      int index = 0;

      int[] newArr = new int[arr.length-count];

      for(int i=0;ilength;i++){

        if(arr[i]!=0){

           newArr[index]=arr[i];

           index++;

        }

      }

      return newArr;

   }

}

构造函数

package com.cloud.day3;

/*

java面向对象的语言: "万物皆对象": 任何事物都可以使用类进行描述。

需求:使用java类描述一个婴儿.

在现实生活中有两种婴儿,一种婴儿一出生就具备了名字(白户),还有一种婴儿就是出生之后才有名字(黑户)

 

构造函数:

构造函数的作用:给对应的对象进行初始化。

构造函数的定义的格式:

   修饰符  函数名(形式参数){

      函数体...

   }

构造函数要注意的细节:

   1. 构造函数是没有返回值类型的。

   2. 构造函数的函数名必须要与类名一致。

   3. 构造函数并不是由我们手动调用的,而是在创建对应的对象时,jvm就会主动调用到对应的构造函数。

   4. 如果一个类没有显式的写上一个构造方法时,那么java编译器会为该类添加一个无参的构造函数的。

   5. 如果一个类已经显式的写上一个构造方法时,那么java编译器则不会再为该类添加一个无参的构造方法。

   6. 构造函数是可以在一个类中以函数重载的形式存在多个的。 

疑问:创建对象时,jvm就会调用到对应的构造方法,那么我们以前没有学构造方法,那么

以前创建对象时,jvm是否也会调用构造方法呢?如果有?构造方法从何而来呢?


        会调用, java编译器在编译的时候给加上去的。


jdk提供了一个java开发工具(javap.exe)给我们进行反编译的。

javap反编译工具的使用格式:

      javap -c -l -private 类名

疑问: java编译器添加的无参构造方法的权限修饰符是什么?

      与类的权限修饰是一致的。

构造函数与普通函数的区别:

   1. 返回值类型的区别:

      1. 构造函数是没有返回值类型的,

      2. 普通函数是有返回值类型的,即使函数没有返回值,返回值类型也要写上void

   2. 函数名的区别:

      1. 构造函数的函数名必须要与类名一致,

      2. 普通函数的函数名只要符合标识符的命名规则即可。

   3. 调用方式的区别:

      1. 构造函数是在创建对象的时候由jvm调用的。

      2. 普通函数是由我们使用对象调用的,一个对象可以对象多次普通的函数,

   4. 作用上的区别:

      1. 构造函数的作用用于初始化一个对象。

      2. 普通函数是用于描述一类事物的公共行为的。

*/

class Baby{

   int id;

   String name;

   public Baby(int i , String n){

      id = i;

      name = n;

      System.out.println("属性初始化完毕...");

   }

   public Baby(){}

   public void cry(){

      System.out.println("Baby哭了...");

   }

}

public class Demo2 {

   public static void main(String[] args) {

      Baby b1 = new Baby(1,"summer");

      System.out.println(b1.id+";"+b1.name);

      b1.cry();

      b1.cry();

   }

}

构造函数案例

package com.cloud.day3;

/*

描述一个员工类,员工具备的属性:id\ name \ age

具备的公共行为:工作。

要求:一旦创建一个员工对象的时候,那么该员工对象就要对应的属性值。

*/

class Employee{

   int id;

   String name;

   int age;

   public Employee(int i,String n,int a){

      id = i;

      name = n;

      age = a;

   }

   public void work(){

      System.out.println(name+"好好工作");

   }

}

public class Demo3 {

   public static void main(String[] args) {

      Employee emp = new Employee(1, "ss", 12);

      System.out.println(emp.id+emp.name+emp.age);

   }

}

构造代码块

package com.cloud.day3;

/*

构造代码块:


构造代码块的作用:给对象进行统一的初始化。


构造函数的作用:给对应的对象进行初始化。

构造代码块的格式:

  

   {

      构造代码块

   }


注意:构造代码块的大括号必须位于成员位置上。

代码块的类别:

   1. 构造代码块。

   2. 局部代码块.   大括号位于方法之内。  作用:缩短局部变量的生命周期,节省一点点内存。

   3. 静态代码块  static

*/

class Body{

   int id;

   String name;

   {

      System.out.println("构造代码块....");

   }

   public Body(int i,String n){

      id = i;

      name = n;

   }

   public Body(){}

   public void talk(){

      System.out.println(name+":speak...");

   }

}

public class Demo4 {

   public static void main(String[] args) {

      Body b1 = new Body(1,"summer");

      b1.talk();

   }

}

 

注意事项

package com.cloud.day3;

/*

构造代码块要注意的事项:

   1. java编译器编译一个java源文件的时候,会把成员变量的声明语句提前至一个类的最前端。

   2. 成员变量的初始化工作其实都在在构造函数中执行的。

   3. 一旦经过java编译器编译后,那么构造代码块的代码块就会被移动构造函数中执行,是在构造函数之前执行的,构造函数的中代码是最后执行的。

   4. 成员变量的显示初始化与构造代码块的代码是按照当前代码的顺序执行的。

*/

public class Demo5 {

   public Demo5(){

      i=30000;

   }

   //代码块初始化

   {

      i=20000;

   }

   int i = 20000; //成员变量最先初始化

   public static void main(String[] args) {

      Demo5 d = new Demo5();

      System.out.println("i="+d.i);

   }

}

This关键字

package com.cloud.day3;

/*

需求:使用java类描述一个动物。

问题:存在同名的成员变量与局部变量时,在方法的内部访问的是局部变量(java 采取的是就近原则的机制访问的。)


this关键字:

this关键字代表了所属函数的调用者对象。

this关键字作用:

   1. 如果存在同名成员变量与局部变量时,在方法内部默认是访问局部变量的数据,可以通过this关键字指定访问成员变量的数据。

   2. 在一个构造函数中可以调用另外一个构造函数初始化对象。

this关键字调用其他的构造函数要注意的事项:

   1. this关键字调用其他的构造函数时,this关键字必须要位于构造函数中的第一个语句。

   2. this关键字在构造函数中不能出现相互调用的情况,因为是一个死循环。

this关键字要注意事项:

   1. 存在同名的成员变量与局部变量时,在方法的内部访问的是局部变量(java 采取的是就近原则的机制访问的。)

   2. 如果在一个方法中访问了一个变量,该变量只存在成员变量的情况下,那么java编译器会在该变量的前面添加this关键字。

*/

class Animal{

   String name;

   String color;

   public Animal(String n,String c){

      this.name = n;

      this.color = c;

   }

   public void eat(){

      System.out.println("this:"+this);

      String name = "sum";

      System.out.println(name+"...");

   }

}

public class Demo6 {

   public static void main(String[] args) {

      Animal dog = new Animal("dog","bai");

      Animal cat = new Animal("cat","black");

      cat.eat();

   }

}

注意事项

package com.cloud.day3;

/*

this关键字调用其他的构造函数要注意的事项:

   1. this关键字调用其他的构造函数时,this关键字必须要位于构造函数中的第一个语句。

   2. this关键字在构造函数中不能出现相互调用的情况,因为是一个死循环。

*/

class Student{

   int id;

   String name;

   public Student(int id,String name){

      this(name);

      this.id = id;

      System.out.println("两个参数的构造方法被调用了");

   }

   public Student(){

      System.out.println("调用了无参的构造方法...");

   }

   public Student(String name){

      this.name = name;

      System.out.println("一个参数的构造方法被调用了...");

   }

}

public class Demo7 {

   public static void main(String[] args) {

      Student s1 = new Student(12,"Sum");

      System.out.println(s1.id+";"+s1.name);

      Student s2 = new Student("Spring");

      System.out.println(s2.name);

   }

}

构造方法中的this

package com.cloud.day3;

/*

需求:使用java定义一个人类,人具备 idname age三个属性,还具备一个比较年龄的方法。

要求:必须写上构造函数,构造函数也必须要使用上this关键字。

*/

class Person{

   int id;

   String name;

   int age;

   public Person(int id,String name,int age){

      this.id = id;

      this.name = name;

      this.age = age;

   }

   public void compareAge(Person p2){

      if(this.age>p2.age){

        System.out.println(this.name+"年龄大");

      }else if(this.ageage){

        System.out.println(p2.name+"年龄大");

      }else{

        System.out.println("一样大");

      }

   }

}

public class Demo8 {

   public static void main(String[] args) {

      Person p1 = new Person(110,"nihao",17);

      Person p2 = new Person(111,"haha",12);

      p1.compareAge(p2);

   }

}

Static关键字

package com.cloud.day3;

/*

static(静态)

需求:描述一下学生类。  都是中国人....

目前存在的问题:所有的学生都是中国的,有n个学生就会有n份中国的数据存内存中,这样子

会浪费内存。

目前方案:中国这个数据移动到数据共享区中,共享这个数据给所有的Student对象使用即可。

 

问题2如何才能把这个数据移动到数据共享区中共享呢?

解决方案:只需要使用static修饰该数据即可。

静态的成员变量只会在数据共享区中维护一份,而非静态成员变量的数据会在每个对象中都维护一份的。。

*/

class Student1{

   String name;

   //使用了static修饰country,那么这时候country就是一个共享的数据。

   static String county = "China";

   public Student1(String name){

      this.name = name;

   }

}

public class Demo9 {

   public static void main(String[] args) {

      Student1 s1 = new Student1("Sum");

      Student1 s2 = new Student1("Aut");

      System.out.println(s1.name+";"+s1.county);

      System.out.println(s2.name+";"+s2.county);

      //因为county是共享的数据,这里修改了

      s2.county = "Anhui";

      System.out.println(s1.name+";"+s1.county);

      System.out.println(s2.name+";"+s2.county);

   }

}

Static修饰成员变量

package com.cloud.day3;

/*

static(静态\修饰符)

   1. static修饰成员变量:如果有数据需要被共享给所有对象使用时,那么就可以使用static修饰。

      静态成员变量的访问方式:

           方式1可以使用对象进行访问。

              格式:对象.变量名。


           方式二:可以使用类名进行访问。

              格式:类名.变量名;

        注意:

           1. 非静态的成员变量只能使用对象进行访问,不能使用类名进行访问。

           2. 千万不要为了方便访问数据而使用static修饰成员变量,只有成员变量的数据是真正需要被共享的时候

           才使用static修饰。


      static修饰成员变量的应用场景:如果一个数据需要被所有对象共享使用的时候,这时候即可好实用static修饰。

   2. static修饰成员函数:

*/

public class Demo10 {

   static String name;//非静态成员变量

   static String color = "white";//静态成员变量

   public static void main(String[] args) {

     

   }

}

Static统计案例

package com.cloud.day3;

/*

需求:统计一个类被使用了多少次创建对象,该类对外显示被创建的次数。

*/

class CountUse{

   static int count = 0;

   String name;

   {

      count++;

   }

   public CountUse(String name){

      this.name = name;

   }

   public CountUse(){}

   public int showCount(){

      return count;

   }

}

public class Demo11 {

   public static void main(String[] args) {

      CountUse cu1 = new CountUse();

      CountUse cu2 = new CountUse();

      CountUse cu3 = new CountUse();

      System.out.println(cu3.showCount());

   }

}

静态函数

package com.cloud.day4;

/*

静态函数:

static(静态、修饰符)

   static修饰成员变量时:static修饰成员变量时,那么该成员变量的数据就是一个共享的数据.

      静态成员变量的访问方式:

           方式一:使用对象进行访问。

                 对象.属性名

           方式二:可以使用类名进行访问。

                 类名.属性名

      注意:

        1. 非静态成员变量不能类名直接访问,只能使用对象进行访问。

        2. 千万不要为了方便访问成员变量而使用static修饰,一定要是该数据是共享数据时才使用static修饰。

 

   static修饰方法(静态的成员方法):

      访问方式:

        方式一:可以使用对象进行访问。

              对象.静态的函数名();

        方式二:可以使用类名进行访问。

              类名.静态函数名字。

      推荐使用是类名直接访问静态的成员。

 

静态的成员变量与非静态的成员变量的区别:

   1. 作用上的区别:

      1. 静态的成员变量的作用共享一个数据给所有的对象使用。

      2. 非静态的成员变量的作用是描述一类事物的公共属性。

   2. 数量与存储位置上的区别:

      1. 静态成员变量是存储方法区内存中,而且只会存在一份数据。

      2. 非静态的成员变量是存储在堆内存中,有n个对象就有n份数据。

   3. 生命周期的区别:

      1. 静态的成员变量数据是随着类的加载而存在,随着类文件的消失而消失。

      2.非静态的成员数据是随着对象的创建而存在,随着对象被垃圾回收器回收而消失。

 

静态函数要注意的事项:

   1. 静态函数是可以调用类名或者对象进行调用的,而非静态函数只能使用对象进行调用。

   2. 静态的函数可以直接访问静态的成员,但是不能直接访问非静态的成员。  

      原因:静态函数是可以使用类名直接调用的,这时候可能还没有存在对象,

      而非静态的成员数据是随着对象的存在而存在的。

 

   3. 非静态的函数是可以直接访问静态与非静态的成员。

      原因:非静态函数只能由对象调用,当对象存在的时候,静态数据老早就已经存在了,而非静态

      数据也随着对象的创建而存在了。

 

   4. 静态函数不能出现this或者super关键字。

      原因:因为静态的函数是可以使用类名调用的,一旦使用类名调用这时候不存在对象,而this

      关键字是代表了一个函数的调用者对象,这时候产生了冲突。

 

静态的数据的生命周期:静态的成员变量数据是优先于对象存在的。

static什么时候修饰一个函数?

   如果一个函数没有直接访问到非静态的成员时,那么就可以使用static修饰了。一般用于工具类型的方法

  

静态函数不能访问非静态的成员?

    静态函数只要存在有对象,那么也可以访问非静态的数据。只是不能直接访问而已。

*/

class Student{

   String name;

   static String county = "中国";

   //静态代码块:静态代码块是在Student.class文件加载到内存的时候就马上执行的。

   static {

      System.out.println("静态代码块执行了...");

   }

   public Student(String name){

      this.name = name;

   }

   public void study(){

      System.out.println("好好学习..."+this);

   }

   //静态方法与非静态方法的字节码文件是同时存在内存中的。只是静态的成员变量数据是优先于对象存在而已。

   public static void sleep(){

      Student s1 = new Student("sum");

      System.out.println(s1.name+"睡觉了...");

   }

}

public class Demo1 {

   public static void main(String[] args) {

      Student.sleep();

   }

}

Java面向对象常见概念


推荐阅读
  • 如何在PHP中安装Xdebug扩展
    本文介绍了如何从PECL下载并编译安装Xdebug扩展,以及如何配置PHP和PHPStorm以启用调试功能。 ... [详细]
  • 嵌套列表的扁平化处理
    本文介绍了一种方法,用于遍历嵌套列表中的每个元素。如果元素是整数,则将其添加到结果数组中;如果元素是一个列表,则递归地遍历这个列表。此方法特别适用于处理复杂数据结构中的嵌套列表。 ... [详细]
  • 本文详细探讨了BCTF竞赛中窃密木马题目的解题策略,重点分析了该题目在漏洞挖掘与利用方面的技巧。 ... [详细]
  • 二维码的实现与应用
    本文介绍了二维码的基本概念、分类及其优缺点,并详细描述了如何使用Java编程语言结合第三方库(如ZXing和qrcode.jar)来实现二维码的生成与解析。 ... [详细]
  • JUnit下的测试和suite
    nsitionalENhttp:www.w3.orgTRxhtml1DTDxhtml1-transitional.dtd ... [详细]
  • 本文介绍了如何通过C#语言调用动态链接库(DLL)中的函数来实现IC卡的基本操作,包括初始化设备、设置密码模式、获取设备状态等,并详细展示了将TextBox中的数据写入IC卡的具体实现方法。 ... [详细]
  • 本文将从基础概念入手,详细探讨SpringMVC框架中DispatcherServlet如何通过HandlerMapping进行请求分发,以及其背后的源码实现细节。 ... [详细]
  • 本文介绍了SIP(Session Initiation Protocol,会话发起协议)的基本概念、功能、消息格式及其实现机制。SIP是一种在IP网络上用于建立、管理和终止多媒体通信会话的应用层协议。 ... [详细]
  • 本文详细介绍了C++中的构造函数,包括其定义、特点以及如何通过构造函数进行对象的初始化。此外,还探讨了转换构造函数的概念及其在不同情境下的应用,以及如何避免不必要的隐式类型转换。 ... [详细]
  • 本文详细介绍了iOS应用的生命周期,包括各个状态及其转换过程中的关键方法调用。 ... [详细]
  • 本文探讨了如何通过Service Locator模式来简化和优化在B/S架构中的服务命名访问,特别是对于需要频繁访问的服务,如JNDI和XMLNS。该模式通过缓存机制减少了重复查找的成本,并提供了对多种服务的统一访问接口。 ... [详细]
  • Windows操作系统提供了Encrypting File System (EFS)作为内置的数据加密工具,特别适用于对NTFS分区上的文件和文件夹进行加密处理。本文将详细介绍如何使用EFS加密文件夹,以及加密过程中的注意事项。 ... [详细]
  • Jupyter Notebook多语言环境搭建指南
    本文详细介绍了如何在Linux环境下为Jupyter Notebook配置Python、Python3、R及Go四种编程语言的环境,包括必要的软件安装和配置步骤。 ... [详细]
  • Android与JUnit集成测试实践
    本文探讨了如何在Android项目中集成JUnit进行单元测试,并详细介绍了修改AndroidManifest.xml文件以支持测试的方法。 ... [详细]
  • protobuf 使用心得:解析与编码陷阱
    本文记录了一次在广告系统中使用protobuf进行数据交换时遇到的问题及其解决过程。通过这次经历,我们将探讨protobuf的特性和编码机制,帮助开发者避免类似的陷阱。 ... [详细]
author-avatar
众大文化_724
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有