java中重载和多态的区别

简介: 本文详细解析了面向对象编程中多态与重载的概念及其关系。多态是OOP的核心,分为编译时多态(静态多态)和运行时多态(动态多态)。编译时多态主要通过方法重载和运算符重载实现,如Java中的同名方法因参数不同而区分;运行时多态则依赖继承和方法重写,通过父类引用调用子类方法实现。重载是多态的一种形式,专注于方法签名的多样性,提升代码可读性。两者结合增强了程序灵活性与扩展性,帮助开发者更好地实现代码复用。

最近发现自己对重载和多态竟然分不清楚,实在是觉得有点羞愧啊。特意写一篇记录一下,以后方便回顾。
关于详细解释多态(polymorphism)和重载(overloading)的关系我们需要深入理解面向对象编程(OOP)中他们各自的含义,应用的场景以及他们的具体实现方式。

多态

多态是OOP的核心概念之一,它允许对象以不同的方式响应相同的消息。多态提高了代码的可复用性和灵活性,使得程序更容易拓展和维护。在多态的概念下,同一个操作可以根据对象的类型产生不同的行为,通常可以分为以下两种形式:

编译时多态(Compile-time Polymorphism)

编译时多态又叫做静态多态,他的多态性在编译阶段就明确了。主要是通过方法重载运算符重载来实现它。

方法重载(Method Overloading):
指在同一个类中存在多种同名的方法,但是这些方法的参数各不相同(例如参数的数量,类型或者循序不同)。编译器在编译的过程中,根据具体调用的方法是提供的参数列表来判断选择对应的重载方法,这就是重载的多态性。
例如:

class Printer{
   
        void print(int value){
   
        System.out.println("good,good,good!!!" + value);
        }
         void print(String value){
   
        System.out.println("hi,hi,hi!!!" + value);
        }
        void print(int value,String suffix){
   
        System.out.println("nice,nice,nice!!!" + value + suffix);
        }
      }

在这个Printer类中,有三个重载的print方法。当我们调用print()时,编译器会根据传入参数的类型来选择适合的重载方法运行。

运算符重载 (Operator Overloading) :这种重载通常用于C++语言,在不同的类中对运算符(如+-等)重新定义其含义,使其能够处理类对象。例如,在C++中可以重载+运算符使其能够对两个自定义类进行操作。

运行时多态(Runtime Polymorphism)

运行时多态又叫动态多态,它是在运行的时候确定的,通常通过继承和方法重写来实现。在动态多态中,父类的引用指向子类的实例,这个引用可以调用子类中重写的方法。
方法重写:是指子类中定义的一个与父类中方法签名完全相同的方法,以便子类可以对这个方法进行不同的实现。当通过父类的引用来调用这个方法时,将根据引用所指向的实际对象类型来执行相应的方法版本。
例如:

class Animal {
    
    void sound() {
    
System.out.println("Some generic animal sound"); 
        } 
    } 
class Dog extends Animal {
   
    @Override 
    void sound() {
   
    System.out.println("Bark");
        } 
    } 
class Cat extends Animal {
    
    @Override
    void sound() {
    
        System.out.println("Meow");
        } 
    } 
public class TestPolymorphism {
    
    public static void main(String[] args) {
   
    Animal a; 
    a = new Dog(); 
    a.sound(); // 输出:Bark
    a = new Cat(); 
    a.sound(); // 输出:Meow 
        } 
    }

类图

classDiagram
    class Animal {
        +void sound()
    }

    class Dog {
        +void sound()
    }

    class Cat {
        +void sound()
    }

    Animal <|-- Dog
    Animal <|-- Cat

    class TestPolymorphism {
        +main(String[] args)
        - Animal a
    }

    TestPolymorphism ..> Animal : "a = new Dog() / a = new Cat()"

在上述例子中,通过父类Animal的引用调用sound()方法,但具体调用哪个实现是在运行时确定的,这就实现了运行时多态

重载

重载是方法的多样性体现,是一种编译时多态的实现方式。它允许在同一个类中定义多个同名方法,但这些方法的参数列表(类型、数量、顺序)必须不同。它的主要优点在于可以让方法名具有统一性,简化类的接口,使代码更加直观。

  • 参数类型不同:可以有不同的类型参数。

    void display(int a);
    void display(String b);
    
  • 参数个数不同:参数个数可以不同。

    void add(int a, int b);
    void add(int a, int b, int c);
    
  • 参数顺序不同:即使参数个数相同,但顺序不同,也可以构成重载。

    void process(int a, double b);
    void process(double b, int a);
    

多态与重载的关系

重载是实现多态的一种方式,属于编译时多态。多态是一个更广泛的概念,既可以通过重载实现(编译时多态),也可以通过方法重写实现(运行时多态)。两者的主要关系和区别在于:

  • 实现方式不同:

    • 重载是通过同名但参数不同的方法实现的,这些方法在同一个类中,属于编译时多态。
    • 运行时多态通过继承和方法重写来实现,通过父类的引用调用子类的重写方法,属于运行时多态。
  • 作用时期不同:

    • 重载在编译时确定,编译器会根据调用的参数选择合适的重载方法。
    • 运行时多态则是在运行时决定调用哪个版本的方法。
  • 代码扩展性:

    • 重载提高了代码的灵活性和可读性,减少了对方法命名的需求,使程序员可以通过统一的方法名称来表达不同的功能。
    • 运行时多态提供了更强大的扩展能力,使得程序可以根据具体的对象类型调用不同的实现,这对构建可扩展系统尤为重要。

实例对比

举个例子来说明重载和运行时多态的不同之处:

  • 重载例子:

    class MathOperation {
         
        // 重载方法:同名,但参数不同
        int multiply(int a, int b) {
         
            return a * b;
        }
    
        double multiply(double a, double b) {
         
            return a * b;
        }
    }
    
    public class TestOverloading {
         
        public static void main(String[] args) {
         
            MathOperation operation = new MathOperation();
            System.out.println(operation.multiply(4, 5));    // 输出:20
            System.out.println(operation.multiply(3.5, 2.5)); // 输出:8.75
        }
    }
    
  • 运行时多态例子:

    class Vehicle {
         
        void start() {
         
            System.out.println("Vehicle starting");
        }
    }
    
    class Car extends Vehicle {
         
        @Override
        void start() {
         
            System.out.println("Car starting");
        }
    }
    
    public class TestRuntimePolymorphism {
         
        public static void main(String[] args) {
         
            Vehicle v = new Car(); // 父类引用指向子类对象
            v.start();             // 输出:Car starting
        }
    }
    

在运行时多态的例子中,虽然变量v的类型是Vehicle,但实际指向的是Car的实例,因此在运行时会调用Car类中的start方法。

总结

graph LR
    OOP[面向对象编程] --> P[多态 Polymorphism]

    P --> CP[编译时多态]
    P --> RP[运行时多态]

    CP --> MO[方法重载]
    CP --> OO[运算符重载]

    MO --> M1[参数类型不同]
    MO --> M2[参数个数不同]
    MO --> M3[参数顺序不同]

    OO --> O1[C++特有]
    OO --> O2[重定义运算符]

    RP --> R1[通过继承实现]
    RP --> R2[方法重写]

    R2 --> RW1[子类重写父类方法]
    R2 --> RW2[方法签名相同]
    R2 --> RW3[运行时确定调用]

    style OOP fill:#f9f,stroke:#333,stroke-width:2px
    style P fill:#bbf,stroke:#333,stroke-width:2px
    style CP fill:#ddf,stroke:#333,stroke-width:2px
    style RP fill:#ddf,stroke:#333,stroke-width:2px
  • 多态 (Polymorphism) 是面向对象编程的重要特性之一,体现了对象可以以不同的形式表现。它通过方法重载和重写实现,包括编译时多态和运行时多态。
  • 重载 (Overloading) 是实现编译时多态的一种方式,通过在同一类中定义参数不同的同名方法来实现。
  • 运行时多态通过继承和方法重写实现,使得程序能够根据实际的对象类型在运行时决定调用哪个方法版本。

重载和多态的结合可以使得面向对象程序设计既灵活又强大,而且更好地实现代码的复用和扩展。希望这回不能忘了!!!

073E332B.gif

目录
相关文章
|
1月前
|
Java
Java 中 Exception 和 Error 的区别
在 Java 中,`Exception` 和 `Error` 都是 `Throwable` 的子类,用于表示程序运行时的异常情况。`Exception` 表示可被捕获和处理的异常,分为受检异常(Checked)和非受检异常(Unchecked),通常用于程序级别的错误处理。而 `Error` 表示严重的系统级问题,如内存不足或 JVM 错误,一般不建议捕获和处理。编写程序时应重点关注 `Exception` 的处理,确保程序稳定性。
|
3月前
|
开发框架 算法 Java
Java方法重写(Override)与重载(Overload)的详细对比
在Java编程中,多态性是面向对象编程的核心概念之一。Java通过两种重要的机制来实现多态:方法重写(Override)和方法重载(Overload)。虽然这两个概念都与方法的不同实现有关,但它们在本质上有很大的区别。本文将详细对比这两种机制,帮助开发者更好地理解和应用它们。
325 5
|
5月前
|
Java 程序员 调度
Java 高级面试技巧:yield() 与 sleep() 方法的使用场景和区别
本文详细解析了 Java 中 `Thread` 类的 `yield()` 和 `sleep()` 方法,解释了它们的作用、区别及为什么是静态方法。`yield()` 让当前线程释放 CPU 时间片,给其他同等优先级线程运行机会,但不保证暂停;`sleep()` 则让线程进入休眠状态,指定时间后继续执行。两者都是静态方法,因为它们影响线程调度机制而非单一线程行为。这些知识点在面试中常被提及,掌握它们有助于更好地应对多线程编程问题。
215 9
|
5月前
|
Java
Java 面向对象编程的三大法宝:封装、继承与多态
本文介绍了Java面向对象编程中的三大核心概念:封装、继承和多态。
373 15
|
5月前
|
安全 Java 程序员
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
184 12
|
21天前
|
算法 Java 调度
Java多线程基础
本文主要讲解多线程相关知识,分为两部分。第一部分涵盖多线程概念(并发与并行、进程与线程)、Java程序运行原理(JVM启动多线程特性)、实现多线程的两种方式(继承Thread类与实现Runnable接口)及其区别。第二部分涉及线程同步(同步锁的应用场景与代码示例)及线程间通信(wait()与notify()方法的使用)。通过多个Demo代码实例,深入浅出地解析多线程的核心知识点,帮助读者掌握其实现与应用技巧。
|
4月前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
272 60
【Java并发】【线程池】带你从0-1入门线程池
|
2月前
|
Java 中间件 调度
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
本文涉及InheritableThreadLocal和TTL,从源码的角度,分别分析它们是怎么实现父子线程传递的。建议先了解ThreadLocal。
109 4
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
|
1月前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
|
30天前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
73 0