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

Java 泛型详细解析

泛型的定义

泛型类的定义

下面定义了一个泛型类 Pair,它有一个泛型参数 T

public class Pair<T> {private T start;private T end;
}

实际使用的时候就可以给这个 T 指定任何实际的类型,比如下面所示,就指定了实际类型为 LocalDate,泛型给了我们一个错觉就是通过个这个模板类 Pair<T>,我们可以在实际使用的时候动态的派生出各种实际的类型,比如这里的 Pair<LocalDate> 类。

Pair<LocalDate> period = new Pair<>();

泛型类的继承

子类是一个泛型类的定义方法如下:

public class Interval<T> extend Pair<T> {}

这里的 Interval<T> 类是一个泛型类,也可以像上面使用 Pair<T> 类一样给它指定实际的类型。

子类是一个具体类的定义方法如下:

public class DateInterval extends Pair<LocalDate> {}

这里的 DateInterval 类就是一个具体的类,而不再是一个泛型类了。这里的语义是 DateInteral 类继承了 Pair<LocalDate> 类,这里的 Pair<LocalDate> 类也是一个具体类。但是由于 Java 的泛型实现机制,这里会带来多态上的一个问题,见下面的分析。

而像下面的这种定义具体类的写法是错误的:

public class DateInterval<LocalDate> extends Pair<LocalDate> {}

泛型方法的定义

泛型方法定义时,类型变量放在修饰符的后面,返回值的前面。泛型方法既可以泛型类中定义,在普通类中定义。

public static <T> T genericMethod(T a) {}

这里顺便记录一下,因为是使用擦除来实现的泛型,因此字节码中的方法的签名是不会包含泛型信息的。对于泛型方法会多生成一个 Signature 的属性,用于记录方法带泛型信息的签名,反编译器也可以根据这个信息将泛型方法还原回来。
image.png
请添加图片描述

构造函数泛型

下面的代码定义了一个泛型类 ConstructorGeneric,它的泛型参数是 T,这个类的构造函数也是泛型的,它有一个泛型参数 X

class ConstructorGeneric<T> {public <X> ConstructorGeneric(X a) {}
}

创建该对象的代码如下:

ConstructorGeneric<Number> t = new <String>ConstructorGeneric<Number>("123");

这里 new 后面的 String 是传给构造器的泛型 X 的,即 X 的实际类型为 String;类的范型参数是由 Number 传递的,即 T 的实际类型是 Number。这里两个都是省略,写在这里是为了显示区分出两个参数传递的位置。

类型变量的限定

带单个上界限定
下面的代码定义了一个 NatualNumber 类,它的泛型参数 T 限制为 Integer 或者 Integer 的子类。

public class NaturalNumber<T extends Integer> {private T n;public NaturalNumber(T n)  { this.n = n; }public boolean isEven() {return n.intValue() % 2 == 0;}
}

调用代码如下:

// 正常
NaturalNumber<Integer> natural1 = new NaturalNumber<>(1);  // 无法编译,因为这里和泛型类定义的上界不符合
NaturalNumber<Double> natualral2 = new NaturalNumber<>(1.0);

带多个上界的限定
多个上界之间使用 & 符号进行分隔,如果多个限定中有类,则类需要排在接口后面(因为 Java 不支持多继承,所以不存在有多个限定的类的情况)。使用时需要满足所有的限定条件才能执行,这个校验应该是在编译时期做的,因为擦除之后,只会保留第一个限定界。

class A {}
interface B {}
class C extends A implements B {}
public static <T extends A & B> void test(T a) {}public static void main(String[] args) {// 编译错误,A 只能满足一个上界test(new A());// 正常test(new C());
}

通配符

在泛型中使用 ? 表示通配符,它的语义是表示未知的类型。通配符可以用作方法的形参、字段的定义、局部变量的定义,以及有的时候作为函数的返回值。通配符不能作为实参调用泛型方法,不能创建对象,或者派生子类型。

上界通配符

当你想定义一个普通方法,这个普通方法可以处理某一类的 List 中的元素时,比如像:List<Number>List<Integer>List<Double> 时,这个时候如果你把方法的入参定义为 List<Number> 是不行的,因为在 Java 中 List<Integer> 不是 List<Number> 的子类。

public static void process(List<Number> numbers) {}// 编译错误
List<Number> numbers = new ArrayList<>();
proess(numbers);

假设 List 是 List 的子类,则可以实现如下的代码:

List<Integer> integers = new ArrayList<>();// 假设下面是成立的
List<Number> numbers = integers;// 下面这句也应该是合法的,但是这违背了 intergers 只能存放 Integer 的语义
numbers.add(new Double());

从上面的例子可以看出,如果允许 List<Integer>List<Number> 的子类型,则会破坏泛型的语义,因此这在 Java 中是不允许的。

但是又实际存在上面描述的这种需求,因此 Java 提供了上界通配符的语法,则方法定义可以定义为如下:

public static void process(List<? extends Number> numbers) {for (Number num : numbers) {// do something}
}// 下面的调用都是能够正常编译通过的
List<Number> numbers = new ArrayList<>();
process(numbers);List<Integer> integers = new ArrayList<>();
process(integers);List<Double> doubles = new ArrayList<>();
process(doubles);

这里的 ? extends Number 的语义就是可以匹配 Number 或者 Number 子类的 List,需要注意的是在 Java 中的继承(extends)和实现(implements)在这里都用关键字 extends 来表示。

从这里也可以看出,List<? extends Number> 的返回值是可以赋值给 Number 类型的。这里可以想象一下 Listget() 方法的泛型参数 E 就变成了 ? extends Number 这个实际类型,而它表达的语义是 Number 以及 Number 的子类,因此赋值给一个 Number 类型的变量是合法的。

但是下面的代码是不合法的:

public static void process(List<? extends Number> numbers) {numbers.add(new Integer());
}

这里同样可以想象一下 Listadd() 方法的入参的泛型参数 E 就变成了 ? extends Number 这个实际类型,它表达的语义是 Number 以及 Number 的子类,但是具体是哪个子类是无法确定的。上面的例子也解释了它可能是 NumberIntegerDouble 等,假设它是 Double 类型,这里放一个 Integer 类型,又违背了泛型只能放 Double 的语义,因此这里的赋值是不合法的。

无界通配符

下面的代码就是定义了一个 List<?> 形参的方法,这里的 List<?> 语义是一个未知类型的 List

public static void printList(List<?> list) {}

无界通配符定义的 List 里面的元素只能赋值给 Object 类型。这里可以想象一下 Listget() 方法的泛型参数 E 就变成了 ? 这个实际类型,它的语义是一个未知的类型,既然是一个未知的类型那么我只能赋值给 Object 类型的变量了。

public static void printList(List<?> list) {for (Object obj : list) {// do something}
}

无界通配符定义的 List 里面只能添加 null,不能添加其它的任何类型的元素,即使是 Object 也不行,因为添加了之后就会违背泛型的语义了。

无界通配符的主要使用场景是:

  • 需要使用 Object 类中的方法
  • 使用了泛型类中不用关心泛型的方法,比如 List 中的 size()clear() 方法

下界通配符

在使用上面的上界通配时,发现了一个问题,如果一个 List 类型形参声明为了上界通配符,是没有办法往这个 List 里面添加元素的,为了解决这个问题,可以使用下界通配符,可以定义如下的方法:

public static void addNumbers(List<? super Number> list) {list.add(new Integer());list.add(new Double());
}

这里可以想象一下这个时候 Listadd() 方法的入参的泛型参数 E 就变成了 ? super Integer 类型,它的语义是匹配 Number 以及 Number 类型的超类。根据 Java 多态的原理,这里实际可以传递的类型为 Integer 以及 Integer 的子类型,因为形参声明的是超类,实际传递子类的引用当然是合法的。

泛型继承关系

泛型的继承关系如下图所示:

image.png

通配符捕获

假设定义了一个无界限通配符的方法如下,这个方法会编译错误,因为按照之前分析的 List<?> 中不能添加任何类型的对象,而这里 list.get(0) 返回的是 Object 类型的对象,肯定是无法放入进去的。代码如下:

public void foo(List<?> list) {list.set(0, list.get(0)); // 编译报错
}

为了解决这个问题这个时候就可以通过新建一个私有的泛型方法来帮助捕获通配符的类型,这个私有的泛型方法名称通常是原有方法加上Helper后缀,这种技巧称为通配符捕获。代码如下:

pulic void foo(List<?> list) {// 调用这个方法的语义是告诉编译器我不知道具体类型是什么,// 但是取出来和放进去的元素类型是相同的fooHelper(list);
}private <T> void fooHelper(List<T> list) {// 合法T temp = list.get(0);// 合法list.set(0, temp);
}

对于泛型方法,因为 add() 方法的入参,get() 方法返回值的泛型参数都是 T,当传入一个 List 进来,虽然这个 List 里面的对象实际类型不知道,但是通过泛型参数可以判断 get() 方法返回类型和 add() 方法的入参类型都是一样的,都是 T 捕获到的一个实际类型 X

对于带通配符参数的方法,因为方法的声明没有一个泛型参数,不能捕获到实际的参数类型 X。那么对于每次方法的调用编译器都会认为是一个不同的类型。比如编译器编译的时候 list.set(0, xxx),这里的入参的类型就会是 CAP#1list.get(0) 返回的类型就是 CAP#2,因为没有一个泛型参数来告诉编译器说 CAP#1CAP#2 是一样的类型,因此编译器就会认为这两个是不同的类型,从而拒绝编译。下图是编译器实际的提示信息:
image.png

image.png
从上面的图也可以看出,第二次调用方法时,类型又变成 CAP#3CAP#4 了,这也证明了每次编译器都会认为是一个新的类型。

实际上这里也可以将这个私有的 Helper 方法定义为公共的,然后去掉通配符的方法。这两种定义实际上是达到了相同的效果,但是 Java 语言规范 5.1.10 章节中更推荐采用通配符的方式定义,但它上面阐述的原因没太看懂,但是在另外一篇博客里面看到一个观点感觉有点道理。
image.png

它说如果定义成一个泛型方法,那么老的遗留的没有用泛型的代码调用这个方法就会产生一个警告,但是如果是使用通配符则不会有警告产生。

public static void foo1(List<?>) {}public static <T> void foo2(List<T>) {}// 假设老的代码没有用泛型
List rawList = Arrays.asList("1", "2");
// 不会产生告警
foo1(rawList);
// 会产生告警,提示未经检查的转换
foo2(rawList);

然而实际上 JDK 中真正的实现并没有采用这种方式,而是直接用注解忽略了异常,直接用的原生类型来实现的。Collections 中的 reverse() 方法内部实现逻辑如下:

@SuppressWarnings({"rawtypes", "unchecked"})  
public static void reverse(List<?> list) {  int size = list.size();  if (size < REVERSE_THRESHOLD || list instanceof RandomAccess) {  for (int i=0, mid=size>>1, j=size-1; i<mid; i++, j--)  swap(list, i, j);  } else {  // instead of using a raw type here, it's possible to capture // the wildcard but it will require a call to a supplementary         // private method ListIterator fwd = list.listIterator();  ListIterator rev = list.listIterator(size);  for (int i=0, mid=list.size()>>1; i<mid; i++) {  Object tmp = fwd.next();  fwd.set(rev.previous());  rev.set(tmp);  }  }  
}

桥接方法

假设定义了如下代码:

public class Node<T> {public T data;public Node(T data) { this.data = data; }public void setData(T data) {System.out.println("Node.setData");this.data = data;}
}public class MyNode extends Node<Integer> {public MyNode(Integer data) { super(data); }public void setData(Integer data) {System.out.println("MyNode.setData");super.setData(data);}
}

泛型擦除后的实际代码如下,注意看 MyNode 里面的 setData() 方法并没有重写 Node 里面的 setData() 方法了,因为方法签名不一样。这就违背了 Java 多态的语义。
Java 编译器在编译的时候会自动给 MyNode 生成一个桥接方法,这个方法的签名和 Node 类里面的一样,然后在这个方法里面去调用真正的 setData() 方法。
通过查看 MyNode.class 文件可以看到真的有两个 setData() 方法存在。
image.png

方法的形参类型是 Object 类型,和 Node 类中泛型擦除后的类型相同,说明这个方法才是真正重载了 Node 类中的方法。
image.png

方法实现中调用了 MyNode 类中形参为 Integer 类型的 setData() 方法。
image.png

同时在 MyNode 类中不允许自己定义形参为 Object 类型的 setData() 方法了,如果定义了则无法编译:
image.png

经过编译器编译后的代码等效为如下的代码:

public class Node {public Object data;public Node(Object data) { this.data = data; }public void setData(Object data) {System.out.println("Node.setData");this.data = data;}
}public class MyNode extends Node<Integer> {public MyNode(Integer data) { super(data); }public void setData(Integer data) {System.out.println("MyNode.setData");super.setData(data);}// 由编译器生成的桥接方法// 如果手动定义了这个方法编译器就会报错了public void setData(Object data) {setData((Integer) data);}
}

泛型的局限性

泛型不能用于基本类型

泛型是通过擦除实现的,擦除之后 ArrayList 内部是 Object[] 类型的数组,是不能存放基本类型的,因为基本类型不是 Object 类型的子类。

List<int> list = new ArrayList<>();

不能创建泛型类型的实例

泛型是通过擦除来实现的,所以擦除之后都会变成 new Object() (没有指定上界的情况),而实际上我们是要创建 T 类型的实例的。

public static <T> void test(List<T> list) {E ele = new E();list.add(ele);
}// 可以通过如下方式
public static <T> void test(List<T> list, Class<T> clazz) {E ele = clazz.newInstance();list.add(ele);
}
// 调用
List<String> list = new ArrayList<>();
test(list, String.class);

不能声明静态的泛型变量

泛型相当于是类的工厂,可以创建不同类型的实例。而静态变量是所有实例共享的,如果允许声明静态的泛型变量,那么不同类型的实例之间就会存在矛盾。

public class MobileDevice<T> {private static T os;
}// 这两个实例的静态变量就会存在矛盾
MobileDevice<Smartphone> phone = new MobileDevice<>();
MobileDevice<TabletPC> pc = new MobileDevice<>();

不能使用 instanceof 判断泛型类型

泛型是通过擦除实现的,因此 List<T>.class 在内存中是不存在的,只有 List.class,这个类型也被称为原生类型。

// 错误
if (list instanceof List<String>) {
}// 正确
if (list instanceof List) {
}

不能创建泛型数组

泛型是通过擦除实现的,如果允许声明泛型数组,则无法实现数组在存放时会校验数组的元素类型这个语义。

// 假设允许创建,这个数组的每个元素只允许存放 List<String> 类型的元素
Object[] stringLists = new List<String>[2]; 
// 正确执行
stringLists[0] = new ArrayList<String>(); 
// 这行应该抛出 ArrayStoreException 异常,
// 但是由于擦除,实际上和上面是一样的,这里违背了数组的语义
stringLists[1] = new ArrayList<Integer>();

不能创建、捕获、抛出带泛型的异常

// 编译报错
class MathException<T> extends Exception {}    // 编译报错
class QueueFullException<T> extends Throwable {}// 编译报错
public static <T extends Exception, J> void execute(List<J> jobs) {try {for (J job : jobs)} catch (T e) {  // 编译报错}
}class Parser<T extends Exception> {// 这样是允许的// 我觉得允许的原因是声明了抛出父类,而实际抛出子类也是合法的public void parse(File file) throws T {  }
}

不能使用擦除后原生类型相同的泛型参数方法来重载

public class Example {// 这两个方法擦除后的参数是一样的,所以不能算重载public void print(Set<String> strSet) { }public void print(Set<Integer> intSet) { }
}

堆污染

当定义变长的泛型参数时,如果尝试把一个原生类型赋值给变成泛型参数就有可能发生堆污染。堆污染的本质原因就是可以通过语法糖变长参数列表来创建泛型的的数组导致的。例如下面的代码:

public class ArrayBuilder {public static <T> void addToList (List<T> listArg, T... elements) {for (T x : elements) {listArg.add(x);}}public static void faultyMethod(List<String>... l) {// 这里编译应该会有告警,如果忽略这个告警,则有可能带来堆污染Object[] objectArray = l;   objectArray[0] = Arrays.asList(42);String s = l[0].get(0);     }
}

编译告警中就会提示有堆污染,如下图所示:
image.png

当编译器遇到一个变长参数方法时,它会把它转换为一个数组。对于 T... elements 这种参数声明就会转为 T[] elements,因为泛型的擦除,最终会被转换为 Object[] elements,这里编译器就会认为有可能发生堆污染。

可以通过以下三种方式抑制这种警告:

  • @SuppressWarnings({"unchecked", "varargs"})
    这种方式只能抑制方法声明时候的告警,方法调用处还是会产生告警;
    image.png
  • @SafeVarargs
    不会产生任何警告
    image.png
  • 增加 -Xlint:varags 编译选项
    不会产生任何警告
    image.png

JVM 控制参数

显示所有告警信息

给编译器增加 -Xlint:unchecked ,在 Idea 中可以参考如下图配置:
image.png

显示更详细的诊断信息

给编译增加 -Xdiags: verbose 选项
image.png

显示所有告警信息为英文

增加如下环境变量:
image.png
Idea 中可以将配置放在 vmproperties 文件中,如下图所示:
image.png

参考

Java Generic Tutorial
Java核心技术·卷 I(原书第10版)
深入理解Java虚拟机(第3版)
When to use generic methods and when to use wild-card?
Why use a wild card capture helper method?
Capture Conv: rev/reverse - what’s the point?
Difference between <? super T> and <? extends T> in Java
What is PECS (Producer Extends Consumer Super)?
Differences between copy(List<? super T> dest, List<? extends T> src) and copy(List dest, List<? extends T> src)

相关文章:

Java 泛型详细解析

泛型的定义 泛型类的定义 下面定义了一个泛型类 Pair&#xff0c;它有一个泛型参数 T。 public class Pair<T> {private T start;private T end; }实际使用的时候就可以给这个 T 指定任何实际的类型&#xff0c;比如下面所示&#xff0c;就指定了实际类型为 LocalDate…...

「Mac畅玩鸿蒙与硬件33」UI互动应用篇10 - 数字猜谜游戏

本篇将带你实现一个简单的数字猜谜游戏。用户输入一个数字&#xff0c;应用会判断是否接近目标数字&#xff0c;并提供提示“高一点”或“低一点”&#xff0c;直到用户猜中目标数字。这个小游戏结合状态管理和用户交互&#xff0c;是一个入门级的互动应用示例。 关键词 UI互…...

自然语言处理期末试题汇总

建议自己做&#xff0c;写完再来对答案。答案可能存在极小部分错误&#xff0c;不保证一定正确。 一、选择题 1-10、C A D B D B C D A A 11-20、A A A C A B D B B A 21-30、B C C D D A C A C B 31-40、B B B C D A B B A A 41-50、B D B C A B B B B C 51-60、A D D …...

记录Threadlocal使用

编写ThreadLocal工具类 package com.jjking.jplan.context;public class BaseContext<T> {public static final ThreadLocal threadLocal new ThreadLocal();//存储用户public static void set(Object t) {threadLocal.set(t);}//获取用户public static <T> T ge…...

利用 SpringBoot 开发的新冠密接者跟踪系统:医疗机构疫情防控辅助方案

摘 要 信息数据从传统到当代&#xff0c;是一直在变革当中&#xff0c;突如其来的互联网让传统的信息管理看到了革命性的曙光&#xff0c;因为传统信息管理从时效性&#xff0c;还是安全性&#xff0c;还是可操作性等各个方面来讲&#xff0c;遇到了互联网时代才发现能补上自古…...

vue 2 父组件根据注册事件,控制相关按钮显隐

目标效果 我不注册事件&#xff0c;那么就不显示相关的按钮 注册了事件&#xff0c;才会显示相关内容 实现思路 组件在 mounted 的时候可以拿到父组件注册监听的方法 拿到这个就可以做事情了 mounted() {console.log(this.$listeners, this.$listeners);this.show.search !…...

【深度学习基础】一篇入门模型评估指标(分类篇)

&#x1f308; 个人主页&#xff1a;十二月的猫-CSDN博客 &#x1f525; 系列专栏&#xff1a; &#x1f3c0;深度学习_十二月的猫的博客-CSDN博客 &#x1f4aa;&#x1f3fb; 十二月的寒冬阻挡不了春天的脚步&#xff0c;十二点的黑夜遮蔽不住黎明的曙光 目录 1. 前言 2. 模…...

hls视频流学习

hls格式播放的依赖安装&#xff1a; <!-- 新增hls播放库 -->npm install hls.js 组件封装&#xff1a; <template><div class"hls-player-cls"><video ref" video" controls style"width: 100%; max-width: 800px;">…...

【electron-vite】搭建electron+vue3框架基础

一、拉取项目 electron-vite 中文文档地址&#xff1a; https://cn-evite.netlify.app/guide/ 官网网址&#xff1a;https://evite.netlify.app/ 版本 vue版本&#xff1a;vue3 构建工具&#xff1a;vite 框架类型&#xff1a;Electron JS语法&#xff1a;TypeScript &…...

第三方Express 路由和路由中间件

文章目录 1、Express 应用使用回调函数的参数&#xff1a; request 和 response 对象来处理请求和响应的数据。2、Express路由1.路由方法2.路由路径3.路由处理程序 3. 模块化路由4. Express中间件1.中间件简介2.中间件分类3.自定义中间件 1、Express 应用使用回调函数的参数&am…...

WPF 常用的5个布局容器控件介绍

1. Grid Grid 是最常用的布局容器之一&#xff0c;它允许开发者以表格的方式对控件进行组织和布局。Grid 使用行和列来划分区域&#xff0c;可以精确控制控件的位置和大小。 特点&#xff1a; 行列定义&#xff1a;Grid 使用 RowDefinitions 和 ColumnDefinitions 来定义行和…...

【JAVA] 杂谈: java中的拷贝(克隆方法)

这篇文章我们来介绍什么是拷贝&#xff0c;并且实现浅拷贝到深拷贝。 目录 一、浅拷贝 1.1 clone 方法 1.2 实现浅拷贝&#xff1a; 1.2.1 重写 clone方法 1.2.2 实现接口 Cloneable 1.2.3 调用克隆方法 1.2.4 原理图&#xff1a;​ 1.3 浅拷贝的不足 1.3.1 增加引用…...

同时多平台git配置:GitHub和Gitee生成不同的SSH Key

文章目录 GitHub和Gitee生成不同的SSH Key步骤1&#xff1a;生成SSH Key步骤2&#xff1a;配置SSH配置文件步骤3&#xff1a;查看SSH公钥步骤4&#xff1a;将SSH公钥添加到GitHub和Gitee步骤5&#xff1a;测试SSH连接步骤6&#xff1a;添加remote远程库 GitHub和Gitee生成不同的…...

flink1.6集成doris,并从mysql同步数据到doris

使用 Apache Flink 1.6 集成 Doris&#xff0c;并从 MySQL 同步数据到 Doris 是一个复杂的任务&#xff0c;但可以通过以下步骤实现。Doris 是一个现代化的 MPP&#xff08;大规模并行处理&#xff09;SQL 数据库&#xff0c;支持实时分析和交互式查询。Flink 可以作为实时数据…...

手搓一个不用中间件的分表策略

场景&#xff1a;针对一些特别的项目&#xff0c;不用中间件&#xff0c;以月为维度进行分表&#xff0c;代码详细设计方案 1. 定义分片策略 首先&#xff0c;定义一个分片策略类&#xff0c;用于决定数据存储在哪个分表中 import java.time.LocalDate; import java.time.fo…...

AI前景分析展望——GPTo1 SoraAI

引言 人工智能&#xff08;AI&#xff09;领域的飞速发展已不仅仅局限于学术研究&#xff0c;它已渗透到各个行业&#xff0c;影响着从生产制造到创意产业的方方面面。在这场技术革新的浪潮中&#xff0c;一些领先的AI模型&#xff0c;像Sora和OpenAI的O1&#xff0c;凭借其强大…...

损失函数Hinge Loss介绍

Hinge Loss 是一种损失函数,广泛用于 支持向量机(SVM, Support Vector Machine) 和一些分类问题中。它特别适合用于 二分类问题,主要目标是让模型的预测值(通常是经过线性变换的原始分数)与真实标签之间的间隔尽可能大,从而提高分类的鲁棒性。 Hinge Loss 的定义 Hinge…...

多维高斯分布(Multivariate Gaussian Distribution)以及协方差矩阵:解析与应用

多维高斯分布&#xff1a;全面解析及其应用 1. 什么是多维高斯分布&#xff1f; 多维高斯分布&#xff08;Multivariate Gaussian Distribution&#xff09;&#xff0c;也称多元正态分布&#xff0c;是高斯分布在高维空间中的推广。它描述了随机向量 ( x ( x 1 , x 2 , … ,…...

前端开发常用快捷键

浏览器 ctrl e 光标定位在搜索框ctrl r 刷新ctrl t 新打开tabctrl tab 向右切换tabctrl shift tab 向左切换tab vscode ctrl p 全局搜索文件ctrl f 当前文件搜索alt 光标左键向下拖动&#xff1a;竖向选中多行文本ctrl b 切换侧边栏显示隐藏ctrl shift p 显示命…...

用MATLAB符号工具建立机器人的动力学模型

目录 介绍代码功能演示拉格朗日方法回顾求解符号表达式数值求解 介绍 开发机器人过程中经常需要用牛顿-拉格朗日法建立机器人的动力学模型&#xff0c;表示为二阶微分方程组。本文以一个二杆系统为例&#xff0c;介绍如何用MATLAB符号工具得到微分方程表达式&#xff0c;只需要…...

全面解析 MySQL 常见问题的排查与解决方法

目录 前言1. 查看 MySQL 日志信息1.1 日志文件的种类与路径1.2 查看日志内容的方法1.3 日志分析的关键点 2. 查看 MySQL 服务状态2.1 查看服务状态2.2 检查进程运行情况2.3 常见启动失败问题与解决 3. 检查 MySQL 配置信息3.1 配置文件的路径与内容3.2 验证配置文件的正确性 4.…...

泷羽Sec-星河飞雪-BurpSuite之解码、日志、对比模块基础使用

免责声明 学习视频来自 B 站up主泷羽sec&#xff0c;如涉及侵权马上删除文章。 笔记的只是方便各位师傅学习知识&#xff0c;以下代码、网站只涉及学习内容&#xff0c;其他的都与本人无关&#xff0c;切莫逾越法律红线&#xff0c;否则后果自负。 泷羽sec官网&#xff1a;http…...

【小白学机器学习34】基础统计2种方法:用numpy的方法np().mean()等进行统计,pd.DataFrame.groupby() 分组统计

目录 1 用 numpy 快速求数组的各种统计量&#xff1a;mean, var, std 1.1 数据准备 1.2 直接用np的公式求解 1.3 注意问题 1.4 用print() 输出内容&#xff0c;显示效果 2 为了验证公式的背后的理解&#xff0c;下面是详细的展开公式的求法 2.1 均值mean的详细 2.2 方差…...

【C++】stack和queue

目录 1. stack的介绍和使用 1.1 stack的介绍 1.2 stack的使用 2. queue的介绍和使用 2.1 queue的介绍 2.2 queue的使用 3. 容器适配器 3.1 什么是适配器 3.2 STL标准库中stack和queue的底层结构 3.3 deque的简单介绍(了解) 3.3.1 deque的原理介绍 3.3.2 deque优势与…...

向量的内积和外积 为什么这样定义

向量的内积和外积 为什么这样定义 flyfish 定义、公理与证明的区别 定义&#xff1a; 定义是人为规定的&#xff0c;用于描述概念的含义。例如&#xff0c;内积和外积是根据实际需求定义的&#xff0c;目的是描述几何和代数性质。定义不需要证明。 公理&#xff1a; 公理是数…...

简述循环神经网络RNN

1.why RNN CNN&#xff1a;处理图像之间没有时间/先后关系 RNN&#xff1a;对于录像&#xff0c;图像之间也许有时间/先后顺序&#xff0c;此时使用CNN效果不会很好&#xff0c;同理和人类的语言相关的方面时间顺序就更为重要了 2.RNN和CNN之间的关联 RNN和CNN本质上其实一…...

【Electron学习笔记(四)】进程通信(IPC)

进程通信&#xff08;IPC&#xff09; 进程通信&#xff08;IPC&#xff09;前言正文1、渲染进程→主进程&#xff08;单向&#xff09;2、渲染进程⇌主进程&#xff08;双向&#xff09;3、主进程→渲染进程 进程通信&#xff08;IPC&#xff09; 前言 在Electron框架中&…...

APP自动化测试框架的开发

基于appium的APP自动化测试框架的开发流程概览 1. 环境搭建 安装Appium Server 下载与安装&#xff1a;可以从Appium官方网站&#xff08;Redirecting&#xff09;下载安装包。对于Windows系统&#xff0c;下载.exe文件后双击安装&#xff1b;对于Mac系统&#xff0c;下载.dmg…...

【深度学习】各种卷积—卷积、反卷积、空洞卷积、可分离卷积、分组卷积

在全连接神经网络中&#xff0c;每个神经元都和上一层的所有神经元彼此连接&#xff0c;这会导致网络的参数量非常大&#xff0c;难以实现复杂数据的处理。为了改善这种情况&#xff0c;卷积神经网络应运而生。 一、卷积 在信号处理中&#xff0c;卷积被定义为一个函数经过翻转…...

pytorch 融合 fuse 学习笔记

目录 fuse_lora 作用是什么 fuse_modules源码解读 fuse_lora 作用是什么 在深度学习模型微调场景下&#xff08;与 LoRA 相关&#xff09; 参数融合功能 在使用 LoRA&#xff08;Low - Rank Adaptation&#xff09;对预训练模型进行微调后&#xff0c;fuse_lora函数的主要作…...

41 基于单片机的小车行走加温湿度检测系统

目录 一、主要功能 二、硬件资源 三、程序编程 四、实现现象 一、主要功能 基于51单片机&#xff0c;采样DHT11温湿度传感器检测温湿度&#xff0c;滑动变阻器连接数码转换器模拟电量采集传感器&#xff0c; 电机采样L298N驱动&#xff0c;各项参数通过LCD1602显示&#x…...

GitLab: You cannot create a branch with a SHA-1 or SHA-256 branch name

最近在迁移git库&#xff0c;把代码从gerrit迁移到gitlab&#xff0c;有几个库报错如下&#xff1a; GitLab: You cannot create a branch with a SHA-1 or SHA-256 branch name ! [remote rejected] refs/users/73/373/edit-95276/1 -> refs/users/73/373/edit-95276/1 (p…...

YOLOv9改进,YOLOv9引入TransNeXt中的ConvolutionalGLU模块,CVPR2024,二次创新RepNCSPELAN4结构

摘要 由于残差连接中的深度退化效应,许多依赖堆叠层进行信息交换的高效视觉Transformer模型往往无法形成足够的信息混合,导致视觉感知不自然。为了解决这个问题,作者提出了一种聚合注意力(Aggregated Attention),这是一种基于仿生设计的token混合器,模拟了生物的中央凹…...

TorchMoji使用教程/环境配置(2024)

TorchMoji使用教程/环境配置&#xff08;2024&#xff09; TorchMoji简介 这是一个基于pytorch库&#xff0c;用于将文本分类成不同的多种emoji表情的库&#xff0c;适用于文本的情感分析 配置流程 从Anaconda官网根据提示安装conda git拉取TorchMoji git clone https://gi…...

uniapp运行时,同步资源失败,未得到同步资源的授权,请停止运行后重新运行,并注意手机上的授权提示。

遇到自定义基座调试时安装无效或无反应&#xff1f;本文教你用 ADB 工具快速解决&#xff1a;打开 USB 调试&#xff0c;连接设备&#xff0c;找到应用包名&#xff0c;一键卸载问题包&#xff0c;清理干净后重新运行调试基座&#xff0c;轻松搞定&#xff01; 问题场景&#…...

uniapp中父组件调用子组件方法

实现过程&#xff08;setup语法糖形式下&#xff09; 在子组件完成方法逻辑&#xff0c;并封装。在子组件中使用defineExpose暴露子组件的该方法。在父组件完成子组件ref的绑定。通过ref调用子组件暴露的方法。 子组件示例 <template> </template><script se…...

腾讯云 AI 代码助手:单元测试应用实践

引言 在软件开发这一充满创造性的领域中&#xff0c;开发人员不仅要构建功能强大的软件&#xff0c;还要确保这些软件的稳定性和可靠性。然而&#xff0c;开发过程中并非所有任务都能激发创造力&#xff0c;有些甚至是重复且乏味的。其中&#xff0c;编写单元测试无疑是最令人…...

ArcGIS栅格影像裁剪工具

1、前言 在最近的栅格转矢量处理过程中&#xff0c;发现二值化栅格规模太大&#xff0c;3601*3601&#xff0c;并且其中的面元太过细碎&#xff0c;通过arcgis直接栅格转面有将近几十万的要素&#xff0c;拿这样的栅格数据直接运行代码&#xff0c;发现速度很慢还难以执行出来结…...

VueWordCloud标签云初实现

文章目录 VueWordCloud学习总结安装初使用在组件中注册该组件简单使用项目中实现最终实现效果 VueWordCloud学习总结 本次小组官网的项目中自己要负责标签模块&#xff0c;想要实现一个标签云的效果&#xff0c;搜索了很多&#xff0c;发现vue有一个VueWordCloud库&#xff0c…...

AI数据分析工具(二)

豆包-免费 优点 强大的数据处理能力&#xff1a; 豆包能够与Excel无缝集成&#xff0c;支持多种数据类型的导入&#xff0c;包括文本、数字、日期等&#xff0c;使得数据整理和分析变得更加便捷。豆包提供了丰富的数据处理功能&#xff0c;如数据去重、填充缺失值、转换格式等…...

sizeof和strlen区分,(好多例子)

sizeof算字节大小 带\0 strlen算字符串长度 \0之前...

求100之内的素数-多语言

目录 C 语言实现 方法 1: 使用 for 循环 方法 2: 使用埃拉托斯特尼筛法 方法 3: 使用自定义判断素数 Python 实现 方法 1: 使用自定义函数判断素数 方法 2: 使用埃拉托斯特尼筛法&#xff08;Sieve of Eratosthenes&#xff09; 方法 3: 使用递归方法 Java 实现 方法…...

0.shell 脚本执行方式

1.脚本格式要求 &#x1f951;脚本以 #!/bin/bash 开头 &#x1f966; 脚本要有可执行权限 2.执行脚本的两种方式 &#x1f96c; 方式1&#xff1a;赋予x执行权限 &#x1f952; ​​​​​​​方式2&#xff1a; sh执行 ​​​​​​​...

Web实时通信@microsoft/signalr

概要说明 signalr 是微软对 websocket技术的封装; build() 与后端signalR服务建立链接&#xff1b;使用 on 方法监听后端定义的函数&#xff1b;ps&#xff1a;由后端发起&#xff0c;后端向前端发送数据使用 invoke 主动触发后端的函数&#xff1b;ps&#xff1a;由前端发起&a…...

智截违规,稳保安全 | 聚铭视频专网违规外联治理系统新品正式发布

“千里之堤&#xff0c;毁于蚁穴”。 违规外联作为网络安全的一大隐患&#xff0c; 加强防护已刻不容缓。 这一次&#xff0c; 我们带着全新的解决方案来了 ——聚铭视频专网违规外联治理系统&#xff0c; 重磅登场&#xff01;...

博弈论算法详解与Python实现

目录 博弈论算法详解与Python实现第一部分:博弈论简介与算法概述1.1 博弈论概述1.2 博弈论算法概述第二部分:纳什均衡算法2.1 纳什均衡的定义2.2 纳什均衡算法的实现2.2.1 算法思路2.2.2 Python实现2.2.3 设计模式分析第三部分:囚徒困境问题的博弈论算法3.1 囚徒困境的定义3…...

Python学习笔记之IP监控及告警

一、需求说明 作为一名运维工程师&#xff0c;监控系统必不可少。不过我们的监控系统往往都是部署在内网的&#xff0c;如果互联网出口故障&#xff0c;监控系统即使发现了问题&#xff0c;也会告警不出来&#xff0c;这个时候我们就需要补充监控措施&#xff0c;增加从外到内的…...

2024/11/30 RocketMQ本机安装与SpringBoot整合

目录 一、RocketMQ简介 1.1、核心概念 1.2、应用场景 1.3、架构设计 2、RocketMQ Server安装 3、RocketMQ可视化控制台安装与使用 4、SpringBoot整合RocketMQ实现消息发送和接收 4.1、添加maven依赖 4.2、yaml配置 4.3、生产者 4.4、消费者 4.5、接口 4.6、接口测试 一、R…...

解决“磁盘已插上,但Windows系统无法识别“问题

电脑上有2块硬盘&#xff0c;一块是500GB的固态硬盘&#xff0c;另一块是1000GB的机械硬盘&#xff0c;按下开机键&#xff0c;发现500G的固态硬盘识别了&#xff0c;但1000GB的机械硬盘却无法识别。后面为了描述方便&#xff0c;将"500GB的固态硬盘"称为X盘&#xf…...

解决vue3,动态添加路由,刷新页面出现白屏或者404

解决vue3&#xff0c;动态添加路由&#xff0c;刷新页面出现白屏或者404 1.解决出现刷新页面&#xff0c;出现404的情况 1.问题的出现 在做毕设的时候&#xff0c;在权限路由得到时候&#xff0c;我问通过router**.**addRoute(item)的方式&#xff0c;在路由守卫动态添加路由…...