深入理解Java泛型及其在实际编程中的应用

news2025/1/10 23:28:42

第1章:泛型的起源与重要性

大家好,我是小黑,在Java里,泛型(Generics)是一种不可或缺的特性,它允许咱们在编码时使用类型(Type)作为参数。这听起来可能有点绕,但其实就像是给方法传递参数一样,只不过这次传递的是数据类型而不是数据值。这样一来,咱们就能写出更加通用、更加安全的代码。想象一下,如果有一个容器,这个容器可以装任何类型的数据,不管是整数、字符串还是自定义的对象,这岂不是非常方便?但如果在取出数据的时候,咱们不知道它是什么类型,那就得转换类型,这时候很容易出错。泛型就是来解决这个问题的。

泛型最早在其他编程语言中出现,Java直到JDK 5.0版本才引入泛型,这一改进大大增强了Java的表达能力,同时也提高了代码的安全性和可读性。通过泛型,编译器可以在编译期间检查类型,避免了运行时的ClassCastException,这对于提升大型应用程序的稳定性和健壮性有着不言而喻的好处。

简单来说,泛型就像是一种严格的门卫,确保咱们在代码中严格遵守类型安全,不会不小心把猫当成狗来养。这样一来,就可以大大减少运行时出现问题的可能性,让咱们的程序更加健壮。

第2章:泛型的基本概念

泛型的基础概念围绕着类型参数(Type Parameters)和类型变量(Type Variables)。让小黑来举个栗子,假设咱们要写一个可以存储任意类型元素的容器类。在不使用泛型的世界里,可能会用Object类型来实现,但这样做既不安全也不方便。引入泛型后,情况就大不相同了。

public class GenericContainer<T> {
    private T element;

    public void setElement(T element) {
        this.element = element;
    }

    public T getElement() {
        return this.element;
    }
}

在这个例子中,T就是一个类型参数,它代表着任何类型。当创建GenericContainer实例的时候,可以指定T的具体类型:

GenericContainer<String> stringContainer = new GenericContainer<>();
stringContainer.setElement("泛型真好玩");
String element = stringContainer.getElement(); // 不需要类型转换

这样一来,咱们就可以在编译期间确保类型的安全性,避免了运行时的类型转换错误。此外,泛型不仅仅可以用在类上,还可以用在接口和方法上。比如,咱们可以有一个泛型接口,表示任何可以比较自己的类型:

public interface Comparable<T> {
    int compareTo(T o);
}

或者是一个泛型方法,用来交换数组中两个元素的位置:

public class ArrayUtil {
    public static <T> void swap(T[] array, int i, int j) {
        T temp = array[i];
        array[i] = array[j];
        array[j] = temp;
    }
}

通过这个swap方法的例子,可以看到泛型方法的强大之处:它不依赖于类或接口的泛型。这个方法可以用于任何类型的数组,增加了代码的复用性。

小黑偷偷告诉你一个买会员便宜的网站: 小黑整的视頻会园优惠站

第3章:泛型的使用场景与实例

泛型在Java编程中的应用非常广泛,尤其是在集合框架中。在没有泛型之前,咱们处理集合时常常需要进行类型转换,这既繁琐又容易出错。有了泛型之后,这一切都变得简单且安全多了。让小黑来带大家看看泛型在实际编程中是如何发挥作用的。

集合框架中的泛型

在Java的集合框架中,泛型的引入让集合的操作变得类型安全且易于管理。比如说,咱们来看看如何使用泛型创建一个只存储字符串的列表:

List<String> stringList = new ArrayList<>();
stringList.add("Java");
stringList.add("泛型");
// stringList.add(123); // 这行代码会编译错误,因为列表只接受字符串类型

这样,编译器就能在编译期间帮助咱们检查类型错误,避免了运行时的类型转换异常。咱们再也不用担心不小心把整数加入到字符串列表中了。

自定义泛型类和方法的使用

泛型不仅仅局限于集合框架,咱们还可以在自己的类和方法中使用泛型。比如说,小黑想要实现一个可以对任意类型的两个元素进行比较的工具类:

public class Pair<T extends Comparable<T>> {
    private T first;
    private T second;

    public Pair(T first, T second) {
        this.first = first;
        this.second = second;
    }

    public T getBigger() {
        return (first.compareTo(second) > 0) ? first : second;
    }
}

// 使用示例
Pair<Integer> intPair = new Pair<>(1, 2);
System.out.println("较大的数字是:" + intPair.getBigger());

Pair<String> stringPair = new Pair<>("apple", "banana");
System.out.println("字典序较后的是:" + stringPair.getBigger());

在这个例子中,Pair类使用了泛型T,并且限制了T必须是实现了Comparable接口的类型,这样就可以保证T类型的对象是可以比较的。这种方式不仅提高了代码的复用性,也保证了类型安全。

泛型的使用场景远不止这些,咱们在实际编程中会发现,几乎所有需要类型参数的地方都可以用泛型来解决。它不仅可以让代码更加灵活和安全,还可以大大提高代码的可读性和可维护性。随着对泛型理解的加深,咱们会发现它在设计模式、API开发等高级应用中的巨大潜力。

第4章:类型擦除与泛型的局限性

泛型在Java中的实现方式是通过类型擦除(Type Erasure)来完成的,这个概念听起来可能有点抽象,但实际上它对咱们使用泛型有着直接的影响。类型擦除意味着在编译时期,所有的泛型信息都会被擦除掉,换句话说,泛型类型参数在编译后的字节码中都会被替换成它们的限定类型(Bounding Type),如果没有指定限定类型,则默认为Object。这个设计决策带来了一些特别的限制,但也使得Java的泛型能够与之前版本的代码兼容。

类型擦除的实例

让小黑用代码示例来说明类型擦除是怎么一回事:

List<String> stringList = new ArrayList<>();
List<Integer> intList = new ArrayList<>();
System.out.println(stringList.getClass() == intList.getClass()); // 输出true

虽然stringListintList是不同类型的泛型实例,但在运行时,它们的类都是ArrayList,没有任何泛型信息。这就是类型擦除的结果。这意味着在运行时,咱们无法获取泛型的具体类型信息,因为它们都被擦除了。

泛型的局限性

由于类型擦除,泛型在Java中有以下几个局限性:

  1. 不能实例化泛型类型的数组

    // T[] array = new T[10]; // 编译错误
    

    这是因为在运行时,JVM需要知道数组的确切类型,而由于类型擦除,这个信息是不可知的。

  2. 不能实例化泛型类的类型参数

    // public class GenericClass<T> {
    //     T obj = new T(); // 编译错误
    // }
    

    同样是因为在运行时,T的具体类型是未知的。

  3. 不能创建具体类型的泛型数组

    // List<Integer>[] arrayOfLists = new List<Integer>[10]; // 编译错误
    

    这违反了Java的类型安全原则,因为泛型类型在运行时会被擦除,导致数组的实际类型只能是List[]

  4. 泛型类不能扩展Throwable

    // public class GenericException<T> extends Exception { } // 编译错误
    

    这是因为异常处理是在运行时进行的,需要知道异常的确切类型。

尽管有这些局限性,泛型仍然是Java编程中非常强大的工具。理解类型擦除和泛型的局限性对于编写健壮和高效的Java代码是非常重要的。通过这些知识,咱们可以更好地利用泛型的优点,同时规避可能遇到的问题。这就需要咱们在使用泛型时既要充分利用其提供的便利和类型安全,又要了解其背后的原理和限制,以便在实际开发中作出恰当的设计和编码决策。

第5章:泛型的继承与通配符

泛型在Java中不仅提高了代码的可读性和安全性,还引入了继承和通配符的概念,这让泛型的应用更加灵活。但是,泛型的继承规则与Java中的类继承有所不同,这经常让初学者感到困惑。让小黑来慢慢道来,希望能让咱们更清晰地理解这个概念。

泛型的继承规则

首先,咱们得明白一个基本概念:在Java泛型中,两个具有相同的泛型类型的类或接口之间,并不存在继承关系。也就是说,List<String>并不是List<Object>的子类型,即使StringObject的子类型。这听起来可能有点违反直觉,但它是有其原因的。这样设计主要是为了确保类型安全,防止咱们在运行时遇到不期望的类型转换错误。

List<Object> objList = new ArrayList<>();
List<String> strList = new ArrayList<>();
// objList = strList; // 编译错误,因为List<String>不是List<Object>的子类型
通配符的使用

为了解决上述问题,Java提供了通配符(Wildcard),用?表示。通配符有两种形式:无界通配符(?),表示任何类型;有界通配符,包括上界通配符(? extends T)和下界通配符(? super T)。

  • 无界通配符(?:当咱们不关心集合中元素的具体类型时,可以使用无界通配符。它主要用于读取操作,因为咱们可以安全地从集合中读取Object类型的数据。
public void printList(List<?> list) {
    for (Object item : list) {
        System.out.println(item);
    }
}
  • 上界通配符(? extends T:表示参数化类型的可能是T或T的某个子类型。它限制了未知类型的上限。上界通配符是为了安全地读取T类型数据而设计的。
public <T> T getFirst(List<? extends T> list) {
    return list.get(0); // 安全地返回T类型
}
  • 下界通配符(? super T:表示参数化类型是T或T的某个父类型。下界通配符让咱们可以安全地写入T和T的子类型的对象。
public <T> void addToList(List<? super T> list, T element) {
    list.add(element); // 安全地添加元素
}

通配符使得泛型更加灵活,但同时也增加了泛型的复杂性。理解和正确使用通配符,对于编写健壮的泛型代码来说非常重要。通过上界和下界通配符的使用,咱们可以在保持类型安全的同时,提高代码的灵活性和可用性。

泛型的继承和通配符是Java泛型中非常强大的特性,它们为处理泛型集合提供了更多的灵活性。掌握这些概念,能够帮助咱们更好地设计和实现泛型接口和方法,使代码既安全又灵活。

第6章:泛型方法的深入分析

泛型方法是Java泛型编程中的一个核心概念,它允许在方法级别上指定泛型类型,使得方法能够在不同类型的上下文中重用。这种方法不仅能提升代码的复用性,还能保持代码的清晰度和类型安全。让小黑来带大家深入了解泛型方法的定义、使用以及它的强大之处。

定义泛型方法

泛型方法可以定义在普通类中,也可以定义在泛型类中。它的特点是,在方法返回类型之前有一个类型参数声明部分(由尖括号<>包围的部分)。这告诉编译器,这个特定的方法将会使用一个或多个类型参数。

public class GenericMethodDemo {

    // 定义一个泛型方法,它可以打印不同类型数组的内容
    public static <T> void printArray(T[] inputArray) {
        for (T element : inputArray) {
            System.out.printf("%s ", element);
        }
        System.out.println();
    }
}

在这个例子中,<T>就是类型参数声明,它告诉编译器,T是一个类型参数,printArray方法可以接受T类型的数组,并遍历打印每个元素。

泛型方法的类型推断

调用泛型方法时,大多数情况下不需要显式指定类型参数,因为编译器能够根据方法参数和调用上下文推断出具体的类型。这种特性称为类型推断。

Integer[] intArray = {1, 2, 3, 4, 5};
String[] stringArray = {"Hello", "World"};

GenericMethodDemo.printArray(intArray); // 类型推断为Integer
GenericMethodDemo.printArray(stringArray); // 类型推断为String

在这个例子中,当调用printArray方法时,不需要指明T代表的是Integer还是String,编译器会自动根据传入参数的类型进行推断。

泛型方法的使用场景

泛型方法非常适用于需要在多种类型之间进行操作的场景。例如,考虑一个交换数组中两个元素位置的方法,这个方法应该能够处理任意类型的数组:

public class ArrayUtil {

    // 泛型方法,用于交换数组中两个元素的位置
    public static <T> void swap(T[] array, int i, int j) {
        T temp = array[i];
        array[i] = array[j];
        array[j] = temp;
    }
}

这个swap方法就是一个典型的泛型方法使用示例。它不依赖于数组的具体类型,可以用于整数数组、字符串数组或任何其他类型的数组。

通过上面的示例和讲解,咱们可以看到,泛型方法提供了极大的灵活性和强大的类型安全性。掌握泛型方法的使用,能够让咱们在面对需要广泛重用的方法时,写出更加清晰、简洁且类型安全的代码。这也是Java泛型编程中一个非常重要的概念,理解并掌握它,对于提升咱们的编程技能至关重要。

第7章:泛型的最佳实践

泛型不仅仅是Java编程中的一个高级特性,它还是一种编程思想,帮助咱们写出更加通用、安全和易于维护的代码。但是,要充分利用泛型的优势,避免其陷阱,就需要遵循一些最佳实践。让小黑来跟大家分享一些使用泛型时的建议和技巧。

明确泛型的使用目的

在引入泛型到你的代码中之前,先明确泛型能为你解决什么问题。泛型主要用于以下几个方面:

  • 类型安全:确保你的集合、方法或类在处理数据时不会遇到类型不匹配的问题。
  • 代码复用:通过类型参数化,使得代码可以用于多种数据类型。
  • API清晰:泛型使得方法签名直接表明使用的类型,提高代码的可读性。
避免泛型警告

当你使用泛型时,可能会遇到编译器的警告,比如未经检查的转换警告。这些警告是有其意义的,它们提示可能存在的类型安全问题。不要忽视这些警告,尽可能地解决它们。如果确信代码是类型安全的,可以使用@SuppressWarnings("unchecked")注解来抑制警告,但这应当是最后的选择。

使用有界通配符来增加API的灵活性

当你设计接受泛型参数的方法时,考虑是否可以通过有界通配符(? extends T? super T)来增加方法的通用性。例如,如果你的方法只从泛型参数类型的集合中读取数据,而不会写入,那么使用? extends T可以让你的方法接受更广泛的类型,比如子类型的集合。

优先使用泛型集合

在可能的情况下,总是使用泛型集合而不是原始类型(raw types)的集合。泛型集合不仅可以提供编译时的类型检查,还可以避免在使用集合时的类型转换,使代码更加清晰。

泛型类和方法的设计

在设计泛型类或方法时,要考虑到泛型的类型参数在实际使用中可能带来的限制。例如,如果一个泛型类的方法中需要创建类型参数的实例,那么这个泛型类就只能用于那些具有无参构造函数的类。这时,你可能需要提供一个工厂方法来解决这个问题。

谨慎使用泛型数组

由于泛型和数组的规则有所不同,创建泛型数组是不合法的。如果需要,可以使用泛型集合,如ArrayList<T>,作为替代。泛型集合提供了数组的大部分功能,同时还增加了类型安全性。

遵循这些最佳实践,并不意味着泛型的使用会变得复杂或限制性增强,相反,它们可以帮助咱们更有效地利用泛型带来的好处。随着对泛型更深入的理解和应用,咱们会发现,泛型不仅能提高代码的质量,还能让编程工作变得更加愉快和有成效。通过这些指导原则和技巧,咱们可以避免常见的泛型相关问题,编写出既强大又灵活的Java代码。

第8章:泛型在实际编程中的高级应用

泛型不仅仅是用来增强集合类的类型安全性,它在Java编程中有着更广泛的应用,尤其是在设计模式、API开发以及框架设计中。通过泛型,咱们可以写出更加灵活、可复用且类型安全的代码。

泛型和设计模式

设计模式是解决软件设计问题的通用解决方案。当咱们将泛型应用于设计模式时,会使得这些模式更加灵活和易于使用。以工厂模式为例,泛型可以使得工厂类能够产生多种类型的产品而不需要为每种产品写一个专门的工厂。

interface Product {}

class ProductA implements Product {}

class ProductB implements Product {}

class GenericFactory<T extends Product> {
    private Class<T> kind;

    public GenericFactory(Class<T> kind) {
        this.kind = kind;
    }

    public T createInstance() throws InstantiationException, IllegalAccessException {
        return kind.newInstance();
    }
}

// 使用示例
GenericFactory<ProductA> factoryA = new GenericFactory<>(ProductA.class);
Product a = factoryA.createInstance();

GenericFactory<ProductB> factoryB = new GenericFactory<>(ProductB.class);
Product b = factoryB.createInstance();

在这个例子中,通过泛型,GenericFactory可以用来创建任何Product的子类的实例,大大增加了代码的复用性和灵活性。

泛型在API开发中的应用

在开发泛型API时,泛型不仅可以提高API的灵活性,还可以增强类型安全性。例如,咱们可以设计一个泛型API来处理不同类型的数据转换:

public class DataConverter<T> {
    public <U> U convert(T data, Class<U> targetClass) throws Exception {
        // 实现数据转换逻辑
        // 这里仅为示例,具体实现会根据实际情况而定
        return targetClass.getDeclaredConstructor().newInstance();
    }
}

// 使用示例
DataConverter<String> converter = new DataConverter<>();
Integer convertedData = converter.convert("123", Integer.class);

这个DataConverter类使用泛型方法convert,可以将任意类型的数据转换成另一种类型。通过这种方式,咱们可以创建一个通用的数据转换工具,而不是为每种数据转换写一个单独的方法或工具。

泛型在框架设计中的应用

许多流行的Java框架,如Spring和Hibernate,广泛使用泛型来提供灵活且类型安全的编程接口。以ORM(对象关系映射)框架为例,泛型可以用来定义能够操作任意实体类型的DAO(数据访问对象)接口:

public interface GenericDao<T, ID> {
    T findById(ID id);
    List<T> findAll();
    void save(T entity);
    void update(T entity);
    void delete(T entity);
}

// 实现示例
public class UserDao implements GenericDao<User, Long> {
    // 实现具体方法
}

在这个例子中,GenericDao接口使用泛型定义了一组通用的数据访问方法。实现这个接口的类可以明确指定操作的实体类型和ID类型,这样就可以为不同的实体重用相同的数据访问逻辑,同时保持类型安全。


更多推荐

详解SpringCloud之远程方法调用神器Fegin

掌握Java Future模式及其灵活应用

小黑的视頻会园优惠站

使用Apache Commons Chain实现命令模式

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1480792.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

LeetCode 2581.统计可能的树根数目:换根DP(树形DP)

【LetMeFly】2581.统计可能的树根数目&#xff1a;换根DP(树形DP) 力扣题目链接&#xff1a;https://leetcode.cn/problems/count-number-of-possible-root-nodes/ Alice 有一棵 n 个节点的树&#xff0c;节点编号为 0 到 n - 1 。树用一个长度为 n - 1 的二维整数数组 edges…

文件基础和文件fd

文章目录 预备知识C语言的文件接口系统调用文件fd 正文开始前给大家推荐个网站&#xff0c;前些天发现了一个巨牛的 人工智能学习网站&#xff0c; 通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。 点击跳转到网站。 预备知识 我们平时说文件就是说文件里…

园区停车管理系统的设计与实现

** &#x1f345;点赞收藏关注 → 私信领取本源代码、数据库&#x1f345; 本人在Java毕业设计领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目希望你能有所收获&#xff0c;少走一些弯路。&#x1f345;关注我不迷路&#x1f345;** 一 、设计说明 1.1 选题…

第十四篇【传奇开心果系列】Python的文本和语音相互转换库技术点案例示例:深度解读Azure Cognitive Services个性化推荐系统

传奇开心果博文系列 系列博文目录Python的文本和语音相互转换库技术点案例示例系列 博文目录前言一、个性化推荐系统介绍和关键功能以及优势解说二、雏形示例代码三、个性化推荐示例代码四、实时推荐示例代码五、多种推荐算法示例代码六、易于集成示例代码七、数据安全和隐私保…

minio多版本

minio的多版本是怎么实现的 1、界面上传3次1M文件&#xff0c;可以显示3个版本的 2、在文件名1M目录下会有3个uuid为名的目录 3、其中的xl.meta文件可以使用xl-meta工具解析&#xff0c;这里会记录详细的版本信息 {"Versions": [{"Header": {"Flag…

快速下载Huggingface的大语言模型

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、Huggingface是什么&#xff1f;二、基于官方huggingface-cli下载&#xff08;基础&#xff0c;断线风险&#xff09;1.安装hf下载环境2.配置环境变量3.注册…

Container killed on request. Exit code is 143

Bug信息 WARN YarnAllocator: Container marked as failed: container_e33_1480922439133_0845_02_000002 on host: hdp4. Exit status: 143. Diagnostics: Container killed on request. Exit code is 143 Container exited with a non-zero exit code 143 Killed by externa…

css 面试 px,rem,em 区别

一、px是决定单位&#xff0c;一旦设置了就无法因为适应页面大小而改变。 二、em和rem 是相对长度单位&#xff0c; 相对于px更具有灵活性&#xff0c;更适用于响应式布局。 三、em是相对于其父元素来设置字体大小的&#xff0c;一般都是以<body>的“font-size”为基准…

Java核心-异常处理

上一次学完了Java的核心类与API&#xff0c;这次来学习异常处理。我们都知道&#xff0c;Java语言具有健壮性和安全性&#xff0c;而异常处理机制就是其重要保证。如下 一、类型 错误(Error)和异常(Exception)。这里只讨论 Exception 类型的异常处理。 1、错误(Error) Err…

浏览器卡顿解决办法 —— DNS设置

1.Windows - 设置 2.网络和Internet 3.以太网 - 更改适配器选项 4.点击你现在用的网络 (WLAN或以太网)&#xff0c;右键属性 5.双击IPv4协议&#xff0c;DNS地址首选114.114.114.114 南京信风网络科技有限公司&#xff0c;成立于2000年&#xff0c;是一家位于中国南京的高…

LeetCode215.数组中的第K个最大元素

题目 给定整数数组 nums 和整数 k&#xff0c;请返回数组中第 k 个最大的元素。 请注意&#xff0c;你需要找的是数组排序后的第 k 个最大的元素&#xff0c;而不是第 k 个不同的元素。 你必须设计并实现时间复杂度为 O(n) 的算法解决此问题。 示例 输入: [3,2,1,5,6,4], …

基于springboot + vue实现的前后端分离-在线旅游网站系统(项目 + 论文)

项目介绍 本旅游网站系统采用的数据库是MYSQL &#xff0c;使用 JSP 技术开发&#xff0c;在设计过程中&#xff0c;充分保证了系统代码的良好可读性、实用性、易扩展性、通用性、便于后期维护、操作方便以及页面简洁等特点。 技术选型 后端: SpringBoot Mybatis 数据库 : MyS…

大模型日报|今日必读的6篇大模型论文

大家好&#xff0c;今日必读的大模型论文来啦&#xff01; 1.中科院、国科大新研究&#xff1a;进行自我感知、更接近人类的AI “机器能思考吗&#xff1f;”这个问题和评估机器是否能达到人类智能水平的图灵测试&#xff0c;是人工智能&#xff08;AI&#xff09;的核心问题…

Stable Diffusion 模型分享:Realistic Stock Photo(真实的库存照片)

本文收录于《AI绘画从入门到精通》专栏&#xff0c;专栏总目录&#xff1a;点这里。 文章目录 模型介绍生成案例案例一案例二案例三案例四案例五案例六案例七案例八 下载地址 模型介绍 条目内容类型大模型基础模型SDXL 1.0来源CIVITAI作者PromptSharingSamaritan文件名称reali…

java 获取package下的所有类名

1、获取指定包的路劲 2、根据路劲获取所有实体类的.class文件 3、我们使用ClassLoader来获取资源的URL&#xff0c;然后根据URL获取对应的文件夹。最后&#xff0c;我们使用过滤器来筛选出所有以".class"结尾的文件 获取entity包下所有的实体进行映射 1.包的路劲…

android程序员面试笔试宝典,Android开发社招面试总结

部分面试常问的面试专题 一、Java篇 1.多线程并发&#xff1b; sleep 和 wait 区别join 的用法线程同步&#xff1a;synchronized 关键字等线程通信线程池手写死锁 2.Java 中的引用方式&#xff0c;及各自的使用场景 3.HashMap 的源码 4.GC(垃圾回收)是什么&#xff1f;如何…

浅谈 Linux 网络编程 - Server 端模型、sockaddr、sockaddr_in 结构体

文章目录 前言前置知识Server 端核心模型 【重点】相关函数 【重点】socket 函数bind 函数listen 函数accept 函数close 函数 sockaddr 数据结构 【重点】 前言 本文主要是对 Linux 网络编程中&#xff0c;Server 端的模型、相关函数 以及 sockaddr、sockaddr_in 结构体做介绍…

用node写后端环境运行时报错Port 3000 is already in use

解决方法:关闭之前运行的3000端口,操作如下 1.WindowR输入cmd确定,打开命令面板 2.查看本机端口详情 netstat -ano|findstr "3000" 3.清除3000端口 taskkill -pid 41640 -f 最后再重新npm start即可,这里要看你自己项目中package.joson的启动命令是什…

VBA_MF系列技术资料1-390

MF系列VBA技术资料1-390 为了让广大学员在VBA编程中有切实可行的思路及有效的提高自己的编程技巧&#xff0c;我参考大量的资料&#xff0c;并结合自己的经验总结了这份MF系列VBA技术综合资料&#xff0c;而且开放源码&#xff08;MF04除外&#xff09;&#xff0c;其中MF01-0…

阿里云能处!2024年ta真降价呀!附服务器优惠价格表

阿里云能处&#xff0c;关键时刻ta真降价啊&#xff01;2024新年伊始阿里云带头降价了&#xff0c;不只是云服务器&#xff0c;云数据库和存储产品都降价&#xff0c;阿里云新老用户均可购买99元服务器、199元服务器&#xff0c;续费不涨价&#xff0c;阿里云百科aliyunbaike.c…