Java -- 类加载机制

类加载机制

 
  使用某个类时,如果该类的class文件没有加载到内存时,则系统会通过以下三个步骤来对该类进行初始化.
 
  类的加载(Load) → 2.类的连接(Link) → 3.类的初始化(Initialize)

  • 类的加载(Load):将类的class文件读入内存,并为之创建一个java.lang.Class的对象,此过程由类加载器(ClassLoader )完成
  • 类的连接(Link):将类中的数据加载到各个内存区域中
  • 类的初始化(Initialize):JVM负责对类进行初始化
     

深入类加载过程

 
  类的完整生命周期 :加载、连接(验证、准备、解析)、初始化、使用、卸载.
 

加载

  1. 通过一个类的全限定名来获取其定义的二进制字节流 (全限定名: 包名+class文件名)

  2. 将这个字节流所代表的的静态存储结构转化为方法区的运行时数据结构

  3. 在堆中生成一个代表这个类的Class对象,作为方法区中这些数据的访问入口

  4. 注意: 相对于类加载过程的其他阶段而言,加载阶段是可控性最强的阶段,因为程序员可以使用系统的类加载器加载,还可以使用自己的类加载器加载,在这里我们只需要知道类加载器的作用就是上面虚拟机需要完成的三件事.
     

连接

 

验证

 
  1. 文件格式的验证:验证.class文件字节流是否符合class文件的格式的规范,并且能够被当前版本的虚拟机处理

  1. 元数据验证:主要是对字节码描述的信息进行语义分析,以保证其描述的信息符合java语言规范的要求,比如说验证这个类是不是有父类,类中的字段方法是不是和父类冲突等等。

  2. 字节码验证:这是整个验证过程最复杂的阶段,主要是通过数据流和控制流分析,确定程序语义是合法的、符合逻辑的。在元数据验证阶段对数据类型做出验证后,这个阶段主要对类的方法做出分析,保证类的方法在运行时不会做出威胁虚拟机安全的事。

  3. 符号引用验证:它是验证的最后一个阶段,发生在虚拟机将符号引用转化为直接引用的时候。主要是对类自身以外 (常量池中的各种符号引用) 的信息进行校验。目的是确保解析动作能够完成。

  4. 注意: 对整个类加载机制而言,验证阶段是一个很重要但是非必需的阶段,如果我们的代码能够确保没有问题,那么我们就没有必要去验证,毕竟验证需要花费一定的的时间。当然我们可以使用

    -Xverfty:none来关闭大部分的验证。

准备

 
  准备阶段主要为类变量(static)分配内存并设置初始值。这些内存都在方法区分配。在这个阶段我们只需要注意两点就好了,类变量和初始值两个关键词:

  1. 类变量(static):会分配内存,但不会对应的分配值,其次实例变量不会分配空间,因为实例变量主要随着对象的实例化一块分配到java堆内存中

  2. 初始值:这里的初始值指的是数据类型默认值,而不是代码中被显示赋予的值

    比如1:public static int value = 1;

       在这里准备阶段过后的value值为0,而不是1赋值为1的动作在初始化阶段
     
    比如2:public static final int value = 1;

      同时被final和static修饰准备阶段之后就是1了,因为static final在编译器就将结果放入调用它的类的常量池中

解析

 
  解析阶段主要是虚拟机将常量池中的符号引用转化为直接引用的过程 .

  1. 符号引用:以一组符号来描述所引用的目标,可以是任何形式的字面量,只要是能无歧义的定位到目标就好,就好比在班级中,老师可以用张三来代表你,也可以用你的学号来代表你,但无论任何方式这些都只是一个代号(符号),这个代号指向你(符号引用)
  2. 直接引用:直接引用是可以指向目标的指针、相对偏移量或者是一个能直接或间接定位到目标的句柄。和虚拟机实现的内存有关,不同的虚拟机直接引用一般不同
  3. 补充: 解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符7类符号引用进行

初始化

 
  这是类加载机制的最后一步,在这个阶段,java程序代码才开始真正执行。在准备阶段已经为类变量赋过一次值。在初始化阶端,程序员可以根据自己的需求来赋值了。一句话描述这个阶段就是执行类构造器clinit()方法的过程。

在初始化阶段,主要为类的静态(stitic)变量赋予正确的初始值,JVM负责对类进行初始化,主要对类变量(stitic)进行初始化。在Java中对类变量进行初始值设定有两种方式:

  1. 声明类变量是指定初始值
  2. 使用静态代码块为类变量指定初始值;

clinit() 方法具有以下特点

  1. 由编译器自动收集类中所有类变量(static)的赋值动作和静态语句块(static{} 块)中的语句合并产生的,编译器收集的顺序由语句在源文件中出现的顺序决定。特别注意的是,静态语句块只能访问到定义在它之前的类变量,定义在它之后的类变量只能赋值,不能访问。例如以下代码
class Test {static {i = 0;                // 给变量赋值可以正常编译通过System.out.print(i);  // 这句编译器会报错,提示“非法向前引用”}static int i = 1;
}
  1. 与类的构造函数(或者说实例构造器 init())不同,不需要显式的调用父类的构造器。虚拟机会自动保证在子类的 clinit() 方法运行之前,父类的 clinit() 方法已经执行结束。因此虚拟机中第一个执行 clinit() 方法的类肯定为 java.lang.Object。由于父类的 clinit() 方法先执行,也就意味着父类中定义的静态语句块要优于子类的变量赋值操作。例如以下代码:
public class Test {public static void main(String[] args) {System.out.println(Son.B);//输出结果是父类中的静态变量A的值,也就是2}
}
class Father{public static int A = 1;static {System.out.println("a");A = 2;}
}
class son extends Father {public static int B = A;}
  1. clinit() 方法对于类或接口不是必须的,如果一个类中不包含静态语句块,也没有对类变量的赋值操作,编译器可以不为该类生成 clinit() 方法。

  2. 接口中不可以使用静态语句块,但仍然有类变量初始化的赋值操作,因此接口与类一样都会生成 clinit() 方法。但接口与类不同的是,执行接口的 () 方法不需要先执行父接口的 clinit() 方法。只有当父接口中定义的变量使用时,父接口才会初始化。另外,接口的实现类在初始化时也一样不会执行接口的 clinit() 方法。

  3. 虚拟机会保证一个类的 clinit() 方法在多线程环境下被正确的加锁和同步,如果多个线程同时初始化一个类,只会有一个线程执行这个类的 clinit() 方法,其它线程都会阻塞等待,直到活动线程执行 clinit() 方法完毕。如果在一个类的 clinit() 方法中有耗时的操作,就可能造成多个线程阻塞,在实际过程中此种阻塞很隐蔽。

  4. JVM初始化步骤:

  1. 假如这个类还没有被加载和连接,则程序先加载并连接该类
  2. 假如该类的直接父类还没有被初始化,则先初始化其直接父类
  3. 假如类中有初始化语句,则系统依次执行这些初始化语句
  1. 类初始化时机:
    只有当对类的主动使用的时候才会导致类的初始化,类的主动使用包括以下六种:
  1. 创建类的实例,也就是new的方式
  2. 访问某个类或接口的静态变量,或者对该静态变量赋值
  3. 调用类的静态方法
  4. 反射
  5. 初始化某个类的子类,则其父类也会被初始化
  6. Java虚拟机启动时被标明为启动类的类,直接使用 java.exe命令来运行某个主类
    比如:测试类Test

  使用: 当 JVM 完成初始化阶段之后,JVM 便开始从入口方法开始执行用户的程序代码.

  卸载: 当用户程序代码执行完毕后,JVM 便开始销毁创建的 Class 对象,最后负责运行的 JVM 也退出内存

利用类加载过程理解面试题

public class Test {public static void main(String[] args) {A a = A.getInstance();System.out.println("A value1:" + a.value1);//1System.out.println("A value2:" + a.value2);//0B b = B.getInstance();System.out.println("B value1:" + b.value1);//1System.out.println("B value2:" + b.value2);//1}
}
class A{private static A a = new A();public static int value1;	public static int value2 = 0;private A(){value1++;value2++;}public static A getInstance(){return a;}
}
class B{public static int value1;public static int value2 = 0;private static B b = new B();private B(){value1++;value2++;}public static B getInstance(){return b;}}

 

类加载器

类加载器实现的功能是即为加载阶段获取二进制字节流的时候,在 Java 虚拟机外部实现,以便让应用程序自己决定如何去获取所需要的类。

分类

从 Java 虚拟机的角度来讲,只存在以下两种不同的类加载器:

  1. 启动类加载器(Bootstrap ClassLoader):这个类加载器用 C++ 实现,是虚拟机自身的一部分
  2. 所有其他类的加载器,这些类由 Java 实现,独立于虚拟机外部,并且全都继承自抽象类 java.lang.ClassLoader

从 Java 开发人员的角度看,类加载器可以划分得更细致一些:

  1. 启动类加载器(Bootstrap ClassLoader): 最顶层的类加载器,负责加载 JAVA_HOME\lib 目录中的,或通过-Xbootclasspath参数指定路径中的,且被虚拟机认可(按文件名识别,如rt.jar)的类。
  2. 扩展类加载器(Extension ClassLoader)这个类加载器是由 ExtClassLoader(sun.misc.Launcher$ExtClassLoader)实现的。 负责加载 JAVA_HOME\lib\ext 目录中的,或通过java.ext.dirs系统变量指定路径中的类库
  3. 应用程序类加载器(Application ClassLoader)这个类加载器是由 AppClassLoader(sun.misc.Launcher$AppClassLoader)实现的。 也叫做系统类加载器,可以通过getSystemClassLoader()获取,负责加载用户路径(classpath)上的类库。如果没有自定义类加载器,一般这个就是默认的类加载器。

类加载器之间的层次关系

 
启动类加载器 > 扩展类加载器 > 应用程序类加载器 > 自定义类加载器

public class ClassLoaderTest {public static void main(String[] args) {Thread thread = new Thread();ClassLoader appClassLoader = thread.getContextClassLoader();ClassLoader extClassLoader = appClassLoader.getParent();ClassLoader booClassLoader = extClassLoader.getParent();//sun.misc.Launcher$AppClassLoader@73d16e93System.out.println("应用程序类加载器:" + appClassLoader);//sun.misc.Launcher$ExtClassLoader@15db9742System.out.println("扩展类加载:" + extClassLoader);//nullSystem.out.println("启动类加载器:" + booClassLoader);/*没有获取到ExtClassLoader的父Loader,原因是Bootstrap Loader(引导类加载器)是用C语言实现的,找不到一个确定的返回父Loader的方式,于是就返回null。*/}
}

双亲委派模型

概念

  类加载器之间的这种层次关系叫做双亲委派模型。

  双亲委派模型要求除了顶层的启动类加载器(Bootstrap ClassLoader)外,其余的类加载器都应当有自己的父类加载器。这里的类加载器之间的父子关系一般不是以继承关系实现的,而是用组合实现的。

工作过程

  一个类加载器接受到类加载请求,他自己不会去加载这个请求,而是将这个类加载请求委派给父类加载器,这样一层一层传送,直到到达启动类加载器(Bootstrap ClassLoader)。
  只有当父类加载器无法加载这个请求时,子加载器才会尝试自己去加载。

好处

  使得 Java 类随着它的类加载器一起具有一种带有优先级的层次关系,从而使得基础类得到统一。

  双亲委派模型很好的解决了各个类加载器加载基础类的统一性问题。即越基础的类由越上层的加载器进行加载.

  比如: java.lang.Object 存放在 rt.jar 中,如果编写另外一个 java.lang.Object 的类并放到 ClassPath 中,程序可以编译通过。因为双亲委派模型的存在,所以在 rt.jar 中的 Object 比在 Clas sPath 中的 Object 优先级更高,因为 rt.jar 中的 Object 使用的是启动类加载器,而 ClassPath 中的 Object 使用的是应用程序类加载器。正因为 rt.jar 中的 Object 优先级更高,因为程序中所有的 Object 都是这个 Object。

双亲委派原则

  1. 可以避免重复加载,父类已经加载了,子类就不需要再次加载
  2. 更加安全,很好的解决了各个类加载器的基础类的统一问题,如果不使用该种方式,那么用户可以随意定义类加载器来加载核心api,会带来相关隐患。

双亲委派模型的代码实现

  1. 检查类是否被加载,没有则调用父类加载器的loadClass()方法;
  2. 若父类加载器为空,则默认使用启动类加载器作为父加载器;
  3. 若父类加载失败,抛出ClassNotFoundException 异常后,再调用自己的findClass() 方法。
//loadClass()源代码
protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {//首先检查类是否被加载Class c = findLoadedClass(name);if (c == null) {//如果没有被加载try {if (parent != null) {//存在父类//则调用父类加载器的loadClass()方法;c = parent.loadClass(name, false);} else {//不存在你父类//则默认使用启动类加载器作为父加载器;c = findBootstrapClass0(name);}} catch (ClassNotFoundException e) {//若父类加载失败,抛出ClassNotFoundException异常后//调用自身的加载功能,一般自定义类重写此方法c = findClass(name);}}if (resolve) {//是否初始化//再调用自己的findClass() 方法。resolveClass(c);}return c;
}

自定义类加载器

首先,我们定义一个待加载的普通Java类:Test.java。放在com.dream.test包下:

package com.dream.test;public class Person {private String name;public Person() {}public Person(String name) {this.name = name;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "Person [name=" + name + "]";}
}

注意:

  如果你是直接在当前项目里面创建,待Test.java编译后,请把Test.class文件拷贝走,再将Test.java`删除。因为如果Test.class存放在当前项目中,根据双亲委派模型可知,会通过sun.misc.Launcher$AppClassLoader类加载器加载。为了让我们自定义的类加载器加载,我们把Test.class文件放入到其他目录。

在本例中,我们Person.class文件存放的目录如下:
C:\code\com\dream\test\Person.class

接下来就是自定义我们的类加载器 :

package com.dream.test;import java.io.BufferedInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;public class ClassLoadTest {public static void main(String[] args) throws Exception {//创建自定义类加载器对象MyClassLoader classLoader = new MyClassLoader("C:/code");//通过类的全路径获取该类的字节码文件对象Class<?> c = classLoader.loadClass("com.dream.test.Person");//创建对象Object obj = c.newInstance();System.out.println(obj);System.out.println(obj.getClass().getClassLoader());//com.dream.test.MyClassLoader@6d06d69c}
}class MyClassLoader extends ClassLoader{private String classPath;public MyClassLoader(){}public MyClassLoader(String classPath) {this.classPath = classPath;}//将"com.dream.test.Person"转换为 Class对象	 @Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {String replaceAll = name.replaceAll("\\.", "/");//com\dream\test\PersonFile file = new File(classPath,replaceAll + ".class");//C:\code\com\dream\test\Person.classtry {//将文件转换为字节数组byte[] bytes = getClassBytes(file);//将字节数组转换为Class对象Class<?> c = this.defineClass(name,bytes, 0, bytes.length);return c;} catch (Exception e) {e.printStackTrace();}return super.findClass(name);}//读取xxx.class文件,把该文件的内容以字节数组返回public byte[] getClassBytes(File file) throws Exception{BufferedInputStream bis = new BufferedInputStream(new FileInputStream(file));ByteArrayOutputStream baos = new ByteArrayOutputStream();byte[] buf = new byte[1024];int len;while((len=bis.read(buf)) != -1){baos.write(buf, 0, len);}bis.close();return baos.toByteArray();}
}

自定义类加载器的应用场景

  引入:Tomcat容器,每个WebApp有自己的ClassLoader,加载每个WebApp的ClassPath路径上的类,一旦遇到Tomcat自带的Jar包就委托给CommonClassLoader加载.

  1. 加密:Java代码可以轻易的被反编译,如果你需要把自己的代码进行加密以防止反编译,可以先将编译后的代码用某种加密算法加密,类加密后就不能再用Java的ClassLoader去加载类了,这时就需要自定义ClassLoader在加载类的时候先解密类,然后再加载。
  2. 从非标准的来源加载代码:如果你的字节码是放在数据库、甚至是在云端,就可以自定义类加载器,从指定的来源加载类。

  以上两种情况在实际中的综合运用:比如你的应用需要通过网络来传输 Java 类的字节码,为了安全性,这些字节码经过了加密处理。这个时候你就需要自定义类加载器来从某个网络地址上读取加密后的字节代码,接着进行解密和验证,最后定义出在Java虚拟机中运行的类。

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

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

相关文章

深入理解 MySQL 索引:优化查询性能的关键

在数据库领域中&#xff0c;索引是优化查询性能的关键之一。MySQL 作为一个广泛使用的关系型数据库&#xff0c;其索引的设计和使用对于提高查询效率和加速数据检索非常重要。在本文中&#xff0c;我们将深入探讨 MySQL 索引的原理、类型以及如何在实际应用中优化查询性能。 1…

Python快速入门系列-5(Python标准库与常用模块)

第五章:Python标准库与常用模块 5.1 常用内置模块介绍5.1.1 os模块5.1.2 datetime模块5.1.3 random模块5.2 文件操作与IO处理5.2.1 文件读写5.2.2 文件复制与移动5.2.3 文件遍历与递归操作5.3 正则表达式5.3.1 匹配字符串5.3.2 替换字符串5.3.3 高级匹配结语Python作为一门强大…

工厂方法模式与抽象工厂模式的深度对比

​&#x1f308; 个人主页&#xff1a;danci_ &#x1f525; 系列专栏&#xff1a;《设计模式》 &#x1f4aa;&#x1f3fb; 制定明确可量化的目标&#xff0c;坚持默默的做事。 &#x1f680; 转载自&#xff1a;设计模式深度解析&#xff1a;工厂方法模式与抽象工厂模式的深…

深度学习论文中结构A+B效果很好,怎么讲故事写成一篇优质的论文?

牛牛牛肉饭&#xff1a; AB的故事不一定好讲 但是可以包装成&#xff1a;A族 B族有效成分 C杂质 D微量杂质 我们创新性的提出了全新的算法Pipeline&#xff0c;涵盖ABCD&#xff0c;极大的改进了该领域的算法范式&#xff0c;除此之外我们系统分析了 A以及其相似算法对 新…

P23—P25:标识符和关键字

标识符 什么是标识符&#xff1f; 在java源程序中&#xff0c;程序员有权自己命名的单词都是标识符在EditPlus编译器中&#xff0c;表示符以黑色高亮字体显示 标识符可以标识什么元素&#xff1f; 类名方法名变量名接口名常量名 … 标识符的命名规则&#xff1a; 只能由**数…

详解 Java多线程带来的的风险-线程安全

目录 一、什么是线程安全&#xff1f; 二、线程不安全的原因 1、线程调度是随机的 2、修改共享数据&#xff1a;多个线程修改同⼀个变量 3、原⼦性 ​编辑 &#xff08;1&#xff09;什么是原⼦性 &#xff08;2&#xff09;⼀条 java 语句不⼀定是原⼦的&#xff0c;也不⼀定…

文心一言指令词宝典之求职招聘篇

作者&#xff1a;哈哥撩编程&#xff08;视频号、抖音、公众号同名&#xff09; 新星计划全栈领域优秀创作者博客专家全国博客之星第四名超级个体COC上海社区主理人特约讲师谷歌亚马逊演讲嘉宾科技博主极星会首批签约作者 &#x1f3c6; 推荐专栏&#xff1a; &#x1f3c5;…

C语言函数递归调用

在C语言中&#xff0c;函数可以直接或间接地调用自身&#xff0c;这种函数调用自身的过程称为递归调用。递归是一种强大的编程技巧&#xff0c;能够简化程序结构、提高代码的可读性和可维护性。本文将介绍C语言函数递归调用的原理、应用场景以及注意事项。 以下是我整理的关于…

PinkysPalaceV2靶场详解IDA逆向查看缓存区溢出漏洞原理以及使用kali gdb使用超详细三次提权字典生成

下载链接: Pinkys Palace: v2 ~ VulnHub 安装&#xff1a; 正常用vm虚拟机打开即可&#xff0c;注意导入时所选择的硬盘存储目录应为空目录&#xff0c;否则会导入失败 根据下载链接提示我们需要更改host文件&#xff0c;以便于我们可以正常访问 kali中的host文件位置为 /etc/h…

理解虚拟dom

什么是虚拟dom&#xff1f; 虚拟 DOM是由一系列的 JavaScript 对象组成的树状结构&#xff0c;每个对象代表着一个DOM元素&#xff0c;包括元素的标签名、属性、子节点等信息。虚拟 DOM中的每个节点都是一个 JavaScript 对象&#xff0c;它们可以轻松地被创建、更新和销毁&…

弧形导轨在自动化设备中的传动原理

在自动化机械系统中&#xff0c;弧形导轨是一种常见的轨道结构&#xff0c;用于支撑和引导物体沿着指定的弧线运动。其工作原理基于几何学和物理学的原理。 弧形导轨通常由一个弧形的轨道和一个移动部件组成。轨道一般呈弧形&#xff0c;其几何形状可以是圆弧、椭圆弧等&#x…

Java(2)之变量,数据类型,运算符

1、标识符 在java中标识符是由字母数字下划线&#xff0c;美元符号组成&#xff08;其中数字没办法作首&#xff09;在java中是区分大小写的。 2、关键字 关键字是在java中赋予了特殊的含义的字符&#xff0c;只能用于特定地方 例如包package 类class 接口interface 引用 &a…

智慧工地整体解决方案(3)

塔吊安全管理系统 需求分析 塔式起重机是现代施工中必不可少的关键设备,是施工企业装备水平的标志性重要装备之一。随着近年来建筑行业塔机的大量使用,由于塔机违规超限作业和塔机群干涉碰撞等引发的各类塔机运行安全事故频繁发生,造成了巨大的生命财产损失。 典型事故现…

Linux 设备树: 设备树节点与属性在 dtb 文件中的存储

前言 当前新版本的 Linux 内核 设备驱动框架&#xff0c;与设备树&#xff08;Device Tree&#xff09;结合密切&#xff0c;整体 设备树的设备驱动框架&#xff0c;比较的庞大&#xff0c;但又非常的经典。 一个个的 设备树解析函数&#xff0c;都是前人【智慧】的结晶&#…

【Golang星辰图】解密数据序列化:探索Go语言中的数据序列化库,提升开发效率

提升效率&#xff1a;了解Go语言中的高效数据序列化解决方案 前言 Go语言作为一门流行的编程语言&#xff0c;其生态系统中涵盖了许多强大且高效的数据序列化库。本文将深入探索Go语言的几个著名数据序列化库&#xff0c;包括go-msgpack、go-bson、go-base58、go-json、go-xm…

【pysurvival Python 安装失败】

这个错误与 sklearn 包的名称更改有关&#xff0c;导致 pysurvival 在构建元数据时失败。现在&#xff0c;你需要修改 pysurvival 的安装文件以使用正确的 scikit-learn 包名 编辑安装文件&#xff1a;找到 pysurvival 的安装文件&#xff0c;可能是 setup.py 或 pyproject.to…

2024系统架构师---隐式调用架构风格的概念与应用

隐式调用架构风格&#xff0c;也被称为事件驱动架构&#xff0c;是一种软件架构模式&#xff0c;其中组件间的交互是基于事件的&#xff0c;而不是直接的调用。这种架构风格侧重于系统中的事件生成、事件监听和事件处理&#xff0c;使组件间的耦合度降低&#xff0c;提高了模块…

智慧公厕:改变城市生活的革命性基础设施

在现代城市的高品质生活要求背景下&#xff0c;公共厕所成为了不可或缺的基础设施。然而&#xff0c;传统的公厕在服务质量、管理效率方面存在一定的问题。为了提供更好的公厕服务&#xff0c;智慧公厕应运而生。通过智能化的管理模式&#xff0c;智慧公厕实现了公厕使用与管理…

玩转ChatGPT:Suno制作音乐

AI开始进军音乐领域了。 一款音乐AI神器——Suno V3发布&#xff0c;它能够处理从间奏到主歌、副歌、桥段直至尾奏的完整结构&#xff0c;零门槛创作音乐。 需要科学上网&#xff0c;官方网站&#xff1a;https://app.suno.ai/ 使用GPT写个歌词&#xff0c;然后丢进Suno生成…

Web APIs知识点讲解(阶段七)

正则表达式 1.能够利用正则表达式校验输入信息的合法性2. 具备利用正则表达式验证小兔鲜注册页面表单的能力 一.正则表达式 1.正则表达式 正则表达式&#xff08;Regular Expression&#xff09;是用于匹配字符串中字符组合的模式。在 JavaScript中&#xff0c;正则表达式也…