当前位置: 首页 > news >正文

类和对象(4)——多态:方法重写与动态绑定、向上转型和向下转型、多态的实现条件

目录

1. 向上转型和向下转型

1.1 向上转型

1.2 向下转型

1.3 instanceof关键字

2. 重写(overidde)

2.1 方法重写的规则

2.1.1 基础规则

2.1.2 深层规则

2.2 三种不能重写的方法

final修饰

private修饰 

static修饰

3. 动态绑定

3.1 动态绑定的概念

3.2 动态绑定与静态绑定

4. 多态

4.1 多态的实现场景

1. 基类形参方法 

2. 基数组

4.2 多态缺陷

1. 属性(字段)没有多态性 

2. 向上转型不能使用子类特有的方法

3. 构造方法没有多态性


上一篇文章中,我们深度学习了继承的概念与实现。在继承篇中,我们最重要的就是“弄清楚通过子类实例变量来访问与父类相同的成员会怎么样”;而在多态篇中,最核心的内容就是“弄清楚通过父类实例变量来访问与子类相同的方法会怎么样”。

1. 向上转型和向下转型

在了解多态之前,我们还要补充几个知识点,这首先就是向上转型和向下转型。

1.1 向上转型

向上转型是指将一个子类对象的引用赋值给一个父类类型的实例变量

语法格式:

父类类型 对象名 = new 子类类型();

//例如 Animal animal = new Cat("小咪", 2);

animal是父类类型的实例变量,但引用的是一个子类Cat对象,因为这是从小范围向大范围的转换。类似基础数据类型中的隐式类型转换(例如长整形long接收整形int的数据)


向上转型的3种使用场景:

  1. 直接赋值:子类对象的引用直接赋值给父类类型的实例变量
  2. 方法传参:子类对象的引用作为参数,传递给方法中的父类类型的形参
  3. 方法返回:方法的返回类型是父类类型返回的值是子类类型

例如:

public class TestAnimal {// 2. 方法传参:形参为父类型引用,可以接收任意子类的对象public static void eatFood(Animal a){a.eat();}// 3. 作返回值:返回任意子类对象public static Animal buyAnimal(String var){if("狗".equals(var) ){return new Dog("狗狗",1);}else if("猫" .equals(var)){return new Cat("猫猫", 1);}else{return null;}}public static void main(String[] args) {Animal cat = new Cat("元宝",2);   // 1. 直接赋值:子类对象赋值给父类对象Dog dog = new Dog("小七", 1);eatFood(cat);eatFood(dog);Animal animal = buyAnimal("狗");animal.eat();animal = buyAnimal("猫");animal.eat();}}

1.2 向下转型

向下转型是将父类对象强制转换为子类对象的过程,需要用到类型转换运算符( ) 。

【注意】

  • 只能对已向上转型的对象进行向下转型:不能直接将一个父类对象强制转换为子类对象,除非这个父类对象实际上是子类对象的向上转型。也就是说,必须先创建一个子类对象,然后将其向上转型为父类对象,最后再进行向下转换。
  • 向上转型的子类类型 与 向下接收的子类类型必须一致

例如:

先看父类和子类的具体代码:

public class Animal {public String name;public Animal(String name){this.name = name;}public void eat(){System.out.println(name+"在吃东西");}
}public class Dog extends Animal{public Dog(String name){super(name);}public void eat(){System.out.println(name+"在吃狗粮");}//Dog类的专属方法public void bark(){System.out.println(name+"在汪汪叫");}
}public class Cat extends Animal{public Cat(String name){super(name);}public void eat(){System.out.println(name+"在吃猫粮");}//Cat类的专属方法public void mew(){System.out.println(name+"在喵喵叫");}
}

测试1:父类实例animal是Dag类的向上转型,再让animal向下转型传给子类实例dog。

public class Test {public static void main(String[] args) {Animal animal = new Dog("旺财");Dog dog;dog = (Dog) animal;dog.bark();}
}

运行成功


测试2:父类实例animal是Dag类的向上转型,再让animal向下转型传给子类实例cat。(向上转型的子类与向下接收的子类不一致

public class Test {public static void main(String[] args) {Animal animal = new Dog("旺财");Cat cat;cat = (Cat) animal;//抛出异常}
}

抛出异常


1.3 instanceof关键字

向下转型用的比较少,而且不安全,万一转换失败,运行时就会抛异常。Java中为了提高向下转型的安全性,引入了instanceof关键字。

语法:

        Object  instanceof  ClassName

其中,object 是要测试的对象或实例变量,ClassName 是要测试的类名。

作用和返回值:

如果 object 是 ClassName 的实例其子类的实例,则表达式返回 true;否则返回 false

有了instanceof,我们向下转型就可以更安全了:

public class TestAnimal {public static void main(String[] args) {Cat cat = new Cat("元宝",2);Dog dog = new Dog("小七", 1);// 向上转型Animal animal = cat;animal.eat();animal = dog;animal.eat();if(animal instanceof Cat){    //检查类型cat = (Cat)animal;cat.mew();}if(animal instanceof Dog){    //检查类型dog = (Dog)animal;dog.bark();}}}

animal最后是Dog类的引用,所以通过了第2个检查,由dog接收animal的向下转型。

2. 重写(overidde)

2.1 方法重写的规则

2.1.1 基础规则

方法重写:也称为方法覆盖,即外壳不变,核心重写。

  1. 子类在重写父类的方法时,一般必须与父类的方法原型一致【返回值类型、方法名、参数列表完全一致】。 
  2. 重写的方法,可以在子类方法头的上一行使用“ @Override ”注解来显式指定。有了这个注解能帮我们进行一些合法性校验(例如不小心将方法名字拼写错了,比如eat写成 aet,那么此时编译器就会发现父类中没有 aet 方法, 就会编译报错, 提示无法构成重写)

例如:

如果显示指定@overidde,但方法与父类原型不一致的话,系统会报错:

2.1.2 深层规则

  1. 返回值类型:其实子类重写的方法返回类型也可以与父类不一样,但是必须是具有父子关系的。       
    1.  该要求其实隐藏了一个情况,那就是此时的方法返回值类型是类类型的。
    2.  对于这种情况的方法重写,父类方法的返回值类型 必须是 子类方法的返回值类型基类
  2. 访问限定符:访问权限不能比父类中被重写的方法的访问权限更低。(即子类重写的方法访问权限可以更宽松,不能更严格)

对于“返回值类型”要求的举例:

​class Parent {public Number display() {return 42; // 返回一个Integer类型的值}
}class Child extends Parent {// 重写父类的display方法,并改变返回类型为Double,这是允许的,因为Double是Number的子类型@Overridepublic Double display() {return 42.0;}
}public class Test {public static void main(String[] args) {Parent parent = new Parent();System.out.println(parent.display()); // 输出: 42Child child = new Child();System.out.println(child.display()); // 输出: 42.0}
}

Parent类的display方法的返回类型是Number类,Child类的display方法的返回类型是Double类。其中Parent类是Child类的父类,Number类又是Double类和Interger类的父类,这符合方法重写的深层规则。

如果该例子中的 方法返回值类型的父子关系反过来 会报错:


对于“访问限定符”要求的举例:


关于方法重写还有更深层更严格的规定,这些规定与异常、线程等有关。本章重点是继承与多态,所以不再具体展开。

2.2 三种不能重写的方法

如果父类方法被final、private或static修饰,则子类不能重写该方法。

final修饰

final:

final修饰成员方法时,就是用来防止该方法被子类重写 或者 不想让该方法被重写。

例如:


private修饰 

private:

父类的方法被private修饰时,说明这个方法是父类私有的,子类也没有办法去访问该方法。

  • 如果private修饰了父类的方法,子类又写了一个与父类方法原型一样的方法,系统并不会报错
    (因为系统检查方法重写时,会自动把父类的私有方法忽略掉)
  • 如果private修饰了子类的方法,父类又有一个与子类方法原型一样的非private方法,那么系统会报错
    (此时系统会认为你想要让子类重写父类方法,又因为重写后的方法是私有的而父类的方法非私有,所以会提醒你“分配了更低的访问权限”并报错)

例1:

父类方法是私有的,build没有问题

例2:

子类方法是私有的,父类方法原型与子类一致。系统认为你要重写,但子类的方法权限更低,所以报错:


static修饰

static:

静态方法是在类加载时就绑定到类本身,而不是在运行时绑定到具体的对象实例,所以static修饰的方法不能被重写。即静态方法不能实现动态绑定,也就不能被覆盖(重写)。

例如:

虽然静态方法是可以被继承的,但如果子类定义了一个与父类相同签名的静态方法这只是对父类静态方法的一种隐藏,而非真正意义上的重写。

  • 子类对象向上转型后,当通过父类实例变量引用调用该方法时,仍然会执行父类的静态方法,而不是子类的静态方法。

例如:

class Parent {static void display() {System.out.println("Parent display method");}
}class Child extends Parent {static void display() {System.out.println("Child display method");}
}public class Test {public static void main(String[] args) {Parent parent1 = new Parent();parent1.display();  //调用父类静态方法Child child = new Child();child.display();   //调用子类静态方法Parent parent2 = new Child();   //向上转型parent2.display();  //调用父类静态方法}
}

3. 动态绑定

3.1 动态绑定的概念

刚刚在解释static修饰方法时,我们提到了一个词叫动态绑定。下面让我们看看什么是动态绑定。

概念:

动态绑定也叫后期绑定,是指在运行时根据对象的实际类型来确定调用哪个方法,而不是在编译时就决定。当一个父类引用指向其子类的对象,并且通过该引用调用一个被重写的方法时,会在运行时根据对象的实际类型来调用相应的方法实现,这就是重写方法的动态绑定。

动态绑定重写方法的实现条件:

  1. 存在继承关系必须有一个基类(父类)和至少一个派生类(子类),子类继承自父类。
  2. 方法重写子类要实现父类中至少一个方法的重写。
  3. 向上转型在程序中存在向上转型的情况,即把子类对象的引用赋值给父类的实例变量

例如:

class Animal {public String name;public Animal(String name){this.name = name;}public void eat(){System.out.println(name+"在吃东西");}
}public class Dog extends Animal{public Dog(String name){super(name);}public void eat(){System.out.println(name+"在吃狗粮");}
}public class Cat extends Animal{public Cat(String name){super(name);}public void eat(){System.out.println(name+"在吃猫粮");}
}public class Test {public static void main(String[] args) {Animal animal = new Animal("动物");//animal动态绑定到Animal类animal.eat();animal = new Dog("小狗");//animal动态绑定到Dog类animal.eat();animal = new Cat("小猫");//animal动态绑定到Cat类animal.eat();}
}

3.2 动态绑定与静态绑定

静态绑定也称为早期绑定:是指在程序编译时就已经确定了方法调用的具体对象和方法实现。与动态绑定相对应,静态绑定不需要运行时进行额外的判断和查找来确定调用哪个方法。

静态绑定的适用情况

  • 基本数据类型的方法调用(可重载的方法):对于基本数据类型的操作方法,如数学运算等,通常是静态绑定。例如,int a = 5; int b = 10; int c = a + b; 中 + 运算符对应的加法方法是在编译时就确定的。
  • 私有方法、静态方法和 final 方法:这些方法不能被重写或具有特殊的性质,所以它们的调用可以在编译时确定。例如,class Example { private void privateMethod() {...} static void staticMethod() {...} final void finalMethod() {...} } 中的私有方法、静态方法和 final 方法都是静态绑定的。
  • 构造方法:构造方法在创建对象时被调用,每个类都有特定的构造方法,且在编译时就可以确定是哪个类的构造方法会被调用。例如,new Example() 会调用 Example 类的构造方法,这是在编译时就已经决定的。

方法重载(静态绑定)是一个类的多态性表现【例如工具类Arrays】,而方法重写(动态绑定)是子类与父类间的多态性的表现。

4. 多态

多态的概念:

去完成某个行为时,当不同的对象去完成时会产生出不同的状态。又或者同一件事情,发生在不同对象身上,就会产生不同的结果

打个比方,语文老师要求同学们背一首诗,同学A背了一首李白的诗、同学B背了一首杜甫的诗、同学C背了一首李清照的诗……每个同学背的诗都不同,但不管怎么说他们都完成了“背一首诗”的任务,这就是多态。

4.1 多态的实现场景

在java中要实现多态,必须要满足如下几个条件,缺一不可:

  1. 必须在继承体系下 
  2. 子类必须要对父类中方法进行重写 
  3. 通过父类的引用调用

下面我来介绍两种常见的多态实现。

1. 基类形参方法 

基类形参方法:指的是形参数据类型为基类类型的方法。

该方法的形参的类型是父类类型,我们一般在该方法中使用被重写的方法。不同的子类实例变量传参进去并发生向上转型,该基类形参方法就能够通过动态绑定来调用不同的重写方法,从而实现多态。

例如:

//有继承关系的类
public class Animal {public String name;public Animal(String name){this.name = name;}public void eat(){System.out.println(name+"在吃东西");}
}public class Dog extends Animal{public Dog(String name){super(name);}public void eat(){System.out.println(name+"在吃狗粮");}
}public class Cat extends Animal{public Cat(String name){super(name);}public void eat(){System.out.println(name+"在吃猫粮");}
}
————————————————————————————————————————————————————————
————————————————————————————————————————————————————————
//含基类形参方法的类
public class Test {public void eat(Animal animal){ //基类形参方法animal.eat();}public static void main(String[] args) {Test test = new Test();    //如果Test的eat方法是静态方法,那么可以不用new一个Test对象test.eat(new Animal("小动物"));test.eat(new Dog("小狗"));test.eat(new Cat("小猫"));}
}


这种方法有点类似C语言中的函数指针和回调函数的用法。详细请看《指针之旅(4)—— 指针与函数:函数指针、转移表、回调函数》

2. 基数组

基数组:指的是数组元素的类型都是基类类型。

由于可以向上转型,在基数组中可以存放子类对象,从而实现多态。

(有点类似C语言中的函数指针数组)

例如:

​//Animal类、Dog类和Cat类的内容如上面的一致
public class Test {public static void main(String[] args) {//基数组animalsAnimal[] animals = {new Animal("小动物"), new Dog("小狗"), new Cat("小猫")};for(Animal x: animals){x.eat();    //临时变量x通过动态绑定实现多态}}
}

如果有新的动物增加,我们可以在基数组animals中添加,这就是多态的好处,十分便捷。

如果不基于多态来实现刚刚的代码内容,我们需要多个if-else语句,如下:

在这种情况下,如果要增加一个动物,不仅字符串数组animals要变,而且在for-each循环中还要加多一条else-if语句,十分不便。

4.2 多态缺陷

1. 属性(字段)没有多态性 

当父类和子类都有同名属性的时候,通过父类实例变量引用只能引用父类自己的成员属性

例如:

public class Parent {public String str = "parent";
}public class Child extends Parent{public String str = "child";
}public class Test {public static void main(String[] args) {Parent parent = new Parent();System.out.println(parent.str);//打印parentChild child = new Child();System.out.println(child.str);//打印childparent = child;     //向上转型System.out.println(parent.str);//属性没有多态性,打印的还是父类的str}
}

2. 向上转型不能使用子类特有的方法

方法调用在编译时进行类型检查,编译器只检查引用变量类型中定义的方法,而不考虑实际对象的类型

例如,这里子类Dog比父类Animal类多了一个特殊方法bark()。如果用Animal类型的实例变量来接收Dog类的对象,我们会发现无法通过该实例变量调用bark方法:

3. 构造方法没有多态性

父类的构造方法中调用一个被重写的方法时,实际执行的是子类中的实现。然而,此时子类可能还未完成初始化,其成员变量尚未赋值或处于默认状态,这就可能导致程序行为不确定,甚至引发错误。

父类构造方法中如果调用了被重写的方法,那么该重写的方法使用的是子类的方法

我们创建两个类, B 是父类, D 是子类. D 中重写 func 方法. 并且在 B 的构造方法中调用 func:

​
class B {public B() {// do nothingfunc();}public void func() {System.out.println("B.func()");}
}class D extends B {private int num = 10;@Overridepublic void func() {System.out.println("D.func() " + num);}
}public class Test {public static void main(String[] args) {D d = new D();}
}​

  • 构造 D 对象的同时,会调用 B 的构造方法.
  • B 的构造方法中调用了 func 方法, 此时会触发动态绑定,会调用到 D 中的 func 。此时 D 对象自身还没有构造,此时 num 处在未初始化的状态,值为 0.
  • 如果具备多态性,num的值应该是10.

结论: "用尽量简单的方式使对象进入可工作状态", 尽量不要在构造器中调用方法(如果这个方法被子类重写, 就会触 发动态绑定, 但是此时子类对象还没构造完成), 可能会出现一些隐藏的但是又极难发现的问题。


本期分享完毕,感谢大家的支持Thanks♪(・ω・)ノ

相关文章:

类和对象(4)——多态:方法重写与动态绑定、向上转型和向下转型、多态的实现条件

目录 1. 向上转型和向下转型 1.1 向上转型 1.2 向下转型 1.3 instanceof关键字 2. 重写(overidde) 2.1 方法重写的规则 2.1.1 基础规则 2.1.2 深层规则 2.2 三种不能重写的方法 final修饰 private修饰 static修饰 3. 动态绑定 3.1 动态绑…...

ui-automator定位官网文档下载及使用

一、ui-automator定位官网文档简介及下载 AndroidUiAutomator:移动端特有的定位方式,uiautomator是java实现的,定位类型必须写成java类型 官方地址:https://developer.android.com/training/testing/ui-automator.html#ui-autom…...

董事会办公管理系统的需求设计和实现

该作者的原创文章目录: 生产制造执行MES系统的需求设计和实现 企业后勤管理系统的需求设计和实现 行政办公管理系统的需求设计和实现 人力资源管理HR系统的需求设计和实现 企业财务管理系统的需求设计和实现 董事会办公管理系统的需求设计和实现 公司组织架构…...

ESP32和STM32在处理中断方面的区别

为了通俗地讲解ESP32和STM32在处理中断方面的区别,我们可以把它们想象成两个不同的“智能管家”系统,各自负责管理一个家庭(即嵌入式项目)的各种任务。我们将重点放在如何处理突发事件(即中断)上。 ESP32 …...

零售业革命:改变行业的顶级物联网用例

mpro5 产品负责人Ruby Whipp表示,技术进步持续重塑零售业,其中物联网(IoT)正引领这一变革潮流。 研究表明,零售商们正在采用物联网解决方案,以提升运营效率并改善顾客体验。这些技术能够监控运营的各个方面…...

字符串算法笔记

字符串笔记 说到字符串,首先我们要注意的就是字符串的输入以及输出,因为字符串的输入格式以及要求也分为很多种,我们就来说几个比较常见的格式 g e t s gets gets 我们先来说这个函数的含义...

在Ubuntu上用Llama Factory命令行微调Qwen2.5的简单过程

半年多之前写过一个教程:在Windows上用Llama Factory微调Llama 3的基本操作_llama-factory windows-CSDN博客 如果用命令行做的话,前面的步骤可以参考上面这个博客。安装好环境后, 用自我认知数据集微调Lora模块:data/identity.j…...

ThinkPhp伪静态设置后,访问静态资源也提示找不到Controller

ThinkPhp没有配置伪静态时,除了默认的IndexController能访问,其他路由Controller都访问不到,提示404错误。配置了伪静态后就解决了这个问题。 但是当我的ThinkPhp后台项目中有静态资源放在public目录(或子目录)中需要…...

JavaScript赋能智能网页设计

构建AI驱动的实时风格迁移系统 案例概述 本案例将实现一个基于深度学习的实时图像风格迁移系统,通过浏览器端神经网络推理实现以下高级特性: WebAssembly加速的ONNX模型推理 WebGL Shader实现的风格混合算法 WebRTC实时视频流处理 基于Web Workers的…...

基于STM32的阿里云智能农业大棚

目录 前言: 项目效果演示: 一、简介 二、硬件需求准备 三、硬件框图 四、CubeMX配置 4.1、按键、蜂鸣器GPIO口配置 4.2、ADC输入配置 4.3、IIC——驱动OLED 4.4、DHT11温湿度读取 4.5、PWM配置——光照灯、水泵、风扇 4.6、串口——esp8266模…...

80,【4】BUUCTF WEB [SUCTF 2018]MultiSQL

53,【3】BUUCTF WEB october 2019 Twice SQLinjection-CSDN博客 上面这个链接是我第一次接触二次注入 这道题也涉及了 对二次注入不熟悉的可以看看 BUUCTF出了点问题,打不开,以下面这两篇wp作为学习对象 [SUCTF 2018]MultiSQL-CSDN博客 …...

深入探索imi框架:PHP Swoole的高性能协程应用实践

摘要 本文将介绍 imi 框架,这是一个基于 PHP Swoole 的高性能协程应用开发框架。imi 支持 HttpApi、WebSocket、TCP 和 UDP 等多种服务类型,利用 Swoole 的优化技术,使得在处理请求时响应速度远超传统的 php-fpm 方式。通过丰富的代码示例&a…...

【算法篇·更新中】C++秒入门(附练习用题目)

一.二分 1.二分查找 我们来看这样一道题&#xff1a; 有一个保证有序的数组a&#xff0c;它的长度为n。现在我们需要知道这个序列是否含有x。 数据范围&#xff1a;保证n<1e9 我们看到这道题之后&#xff0c;第一时间想到的就是暴力枚举了&#xff0c;可是我们发现直接枚举…...

对神经网络基础的理解

目录 一、《python神经网络编程》 二、一些粗浅的认识 1&#xff09; 神经网络也是一种拟合 2&#xff09;神经网络不是真的大脑 3&#xff09;网络构建需要反复迭代 三、数字图像识别的实现思路 1&#xff09;建立一个神经网络类 2&#xff09;权重更新的具体实现 3&am…...

存储基础 -- SCSI命令格式与使用场景

SCSI命令格式与使用场景 1. SCSI命令描述符块&#xff08;CDB&#xff09; 1.1 CDB基本概念 SCSI命令通过**命令描述符块&#xff08;CDB, Command Descriptor Block&#xff09;**表示。 CDB长度&#xff1a;SCSI命令根据使用场景有不同长度的CDB&#xff0c;常见的有6字节…...

从崩溃难题看 C 标准库与 Rust:线程安全问题引发的深度思考

在软件开发的世界里&#xff0c;每一次技术的变革和尝试都伴随着未知的挑战。EdgeDB 团队在将部分网络 I/O 代码从 Python 迁移到 Rust 的过程中&#xff0c;就遭遇了一场棘手的问题&#xff0c;这个问题不仅暴露了 C 标准库的线程安全隐患&#xff0c;也让我们对 Rust 的 “安…...

【CSS入门学习】Flex布局设置div水平、垂直分布与居中

水平平均分布 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</title><style>…...

9. 神经网络(一.神经元模型)

首先&#xff0c;先看一个简化的生物神经元结构&#xff1a; 生物神经元有多种类型&#xff0c;内部也有复杂的结构&#xff0c;但是可以把单个神经元简化为3部分组成&#xff1a; 树突&#xff1a;一个神经元往往有多个树突&#xff0c;用于接收传入的信息。轴突&#xff1a;…...

R 语言 | future 包,非阻塞的执行耗时脚本

目的&#xff1a;有一段代码&#xff0c;后面暂时用不到&#xff0c;但是又很耗时&#xff0c;占了当前R session&#xff0c;难道只能等半个小时&#xff0c;等到它结束才能画图&#xff1f; 可以使用R多线程&#xff0c;在支线进程中执行耗时任务&#xff0c;同时不阻塞当前…...

UE学习日志#12 Niagara特效大致了解(水文,主要是花时间读了读文档和文章)

1 核心组件&#xff08;官方文档阅读&#xff09; 一些介绍主要来自官方文档Niagara概述 1.1 Systems 官方文档中的描述&#xff1a; Niagara系统是一种容器&#xff0c;可以放入你要构建该效果的所有内容。在这个系统中&#xff0c;你可以搭建不同的构建块来实现总体效果。…...

【数据结构】_链表经典算法OJ:合并两个有序数组

目录 1. 题目描述及链接 2. 解题思路 3. 程序 3.1 第一版 3.2 第二版 1. 题目描述及链接 题目链接&#xff1a;21. 合并两个有序链表 - 力扣&#xff08;LeetCode&#xff09; 题目描述&#xff1a; 将两个升序链表合并为一个新的 升序 链表并返回。 新链表是通过拼接给…...

Mongodb副本集群为什么选择3个节点不选择4个节点

一、容错能力的定义 在副本集中&#xff0c;容错能力是指系统能够容忍多少个节点故障而仍然能够保持服务可用性的能力。这通常与选举机制中的多数投票原则密切相关。 二、三节点副本集的容错能力 在三节点的副本集中&#xff0c;通常有一个主节点和两个从节点。当主节点故障…...

基于 WEB 开发的手机销售管理系统设计与实现内容

标题:基于 WEB 开发的手机销售管理系统设计与实现 内容:1.摘要 摘要&#xff1a;随着智能手机的普及和电子商务的快速发展&#xff0c;手机销售行业面临着越来越多的挑战和机遇。为了提高销售效率和管理水平&#xff0c;本文设计并实现了一个基于 WEB 的手机销售管理系统。该系…...

LeetCode - Google 大模型校招10题 第1天 Attention 汇总 (3题)

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/145368666 GroupQueryAttention(分组查询注意力机制) 和 KVCache(键值缓存) 是大语言模型中的常见架构&#xff0c;GroupQueryAttention 是注意力…...

Vue3 provide/inject用法总结

1. 基本概念 provide/inject 是 Vue3 中实现跨层级组件通信的方案&#xff0c;类似于 React 的 Context。它允许父组件向其所有子孙组件注入依赖&#xff0c;无论层级有多深。 1.1 基本语法 // 提供方&#xff08;父组件&#xff09; const value ref(hello) provide(key, …...

Linux——网络基础(1)

文章目录 目录 文章目录 前言 一、文件传输协议 应用层 传输层 网络层 数据链路层 数据接收与解封装 主机与网卡 数据传输过程示意 二、IP和MAC地址 定义与性质 地址格式 分配方式 作用范围 可见性与可获取性 生活例子 定义 用途 特点 联系 四、TCP和UDP协…...

【记录】日常|从零散记录到博客之星Top300的成长之路

文章目录 shandianchengzi 2024 年度盘点概述写作风格简介2024年的创作内容总结 shandianchengzi 2024 年度盘点 概述 2024年及2025年至今我创作了786即84篇文章&#xff0c;加上这篇就是85篇。 很荣幸这次居然能够入选博客之星Top300&#xff0c;这个排名在我之前的所有年份…...

【二分查找】力扣373. 查找和最小的 K 对数字

给定两个以 非递减顺序排列 的整数数组 nums1 和 nums2 , 以及一个整数 k 。 定义一对值 (u,v)&#xff0c;其中第一个元素来自 nums1&#xff0c;第二个元素来自 nums2 。 请找到和最小的 k 个数对 (u1,v1), (u2,v2) … (uk,vk) 。 示例 1: 输入: nums1 [1,7,11], nums2 …...

池化层Pooling Layer

1. 定义 池化是对特征图进行的一种压缩操作&#xff0c;通过在一个小的局部区域内进行汇总统计&#xff0c;用一个值来代表这个区域的特征信息&#xff0c;常用于卷积神经网络&#xff08;CNN&#xff09;中。 2. 作用 提取代表性信息的同时降低特征维度&#xff0c;具有平移…...

力扣算法题——11.盛最多水的容器

目录 &#x1f495;1.题目 &#x1f495;2.解析思路 本题思路总览 借助双指针探索规律 从规律到代码实现的转化 双指针的具体实现 代码整体流程 &#x1f495;3.代码实现 &#x1f495;4.完结 二十七步也能走完逆流河吗 &#x1f495;1.题目 &#x1f495;2.解析思路…...