JVM知识点-01-从JDK源码级别剖析JVM类加载机制

news2024/11/25 19:55:22

1 什么是Java虚拟机

  1. 一个可执行java字节码的虚拟机进程;
  2. 跨平台的是java程序,而不是java虚拟机,java虚拟机在各个操作系统是不兼容的,例如windows、linux、mac都需要安装各自版本的虚拟机,java虚拟机通过jdk实现功能。jvm是用c/c++来写的,它屏蔽了不同操作系统硬件和软件之间的差异;
  3. oracle(原sun公司)虚拟机Sun HotSpot,生产环境使用该jdk。open jdk,使用在linux系统上,开源免费;

2 JVM类加载机制

2.1 类编译

2.1.1 javac

javac Math.java

将Math.java编译成Math.class字节码文件;
字节码本质是一个字节数组byte[](所以被称作字节码文件),它有特定的复杂的内部结构;

2.1.2 javap

javap -v Math.class

将字节码反编译为可读的字节码指令文件;
源代码:

public class SyncCodeBlock {
   public int i;
   public void syncTask(){
       synchronized (this){
           i++;
       }
   }
}

反编译后:

public void syncTask();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=3, locals=3, args_size=1
         0: aload_0
         1: dup
         2: astore_1
         3: monitorenter  //注意此处,进入同步方法
         4: aload_0
         5: dup
         6: getfield      #2             // Field i:I
         9: iconst_1
        10: iadd
        11: putfield      #2            // Field i:I
        14: aload_1
        15: monitorexit   //注意此处,退出同步方法
        16: goto          24
        19: astore_2
        20: aload_1
        21: monitorexit //注意此处,退出同步方法
        22: aload_2
        23: athrow
        24: return
      Exception table:
      //省略其他字节码.......

JVM指令手册.pdf

2.1.3 class常量池

class文件包含的信息:类版本号、字段、方法、接口等描述信息、常量池信息;
常量池信息存放了编译器生成的字面量和符号引用;
一个class文件十六进制大体结构:
image.png
对应的含义如下:
image.png
javap -v Math.class如下:
image.png
红框为常量池信息,等号右边字符为“字面量”,左边#为符号引用;
常量池一旦被jvm装载到内存,就是运行时常量池了,对应的符号引用对应被加载到内存代码的直接引用,即动态链接;

2.1.4 字符串常量池

字符串分配,和其他对象分配一样,需要耗费高昂的时间和空间,作为基础数据类型,频繁的创建字符串,极大的耗费性能;
jvm对字符串进行了优化,为字符串开辟常量池,类似缓存区。创建字符串常量池时,先判断是否在该缓存区。存在返回该字符串,不存在,实例化该字符串并放入缓存区;
image.png
包装类常量池,包括:Byte、Short、Integer、Long、Character、Boolean,前5种在数值小于127时才使用对象池;Double没有实现常量池;
字符串常量池,java7之前放在方法区,java7以及以后放在堆区;

2.2 类加载

2.2.1 类加载运行过程

  • 通过java命令运行编译后的class文件,启动类的main函数,通过类加载器将主类加载到JVM内存
package com.firechou.test.testjava.jvm;

public class Math {
    public static final int initData = 666;
    public static User user = new User();

    public int compute() {  //一个方法对应一块栈帧内存区域
        int a = 1;
        int b = 2;
        int c = (a + b) * 10;
        return c;
    }

    public static void main(String[] args) {
        Math math = new Math();
        int result = math.compute();
        System.out.println(result);
    }
}

  • 注意,通过java命令执行带package的类时,需要先进入classes根目录,执行class文件时带上包名
zhouyan@MacBook-Pro classes % pwd
/Users/zhouyan/projects/IdeaProjects/test-group/test-java/target/classes
zhouyan@MacBook-Pro classes % java com.firechou.test.testjava.jvm.Math
30
  • java命令执行代码流程

  • 其中loadClass的类加载过程如下

加载》验证》准备》解析》初始化》使用》卸载

加载:在硬盘上查找并通过IO读入字节码文件,使用到类时才会加载,例如调用类的main()方法,new对象等等,在加载阶段会在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口;
验证:校验字节码文件的正确性;
准备:给类的静态变量分配内存,并赋予默认值;
解析:将符号引用替换为直接引用,该阶段会把一些静态方法(符号引用,比如main()方法)替换为指向数据所存内存的指针或句柄等(直接引用),这是所谓的静态链接过程(类加载期间完成),动态链接是在程序运行期间完成的将符号引用替换为直接引用;
初始化:对类的静态变量初始化为指定的值,执行静态代码块;

2.2.2 类加载到方法区信息

加载到jvm方法区主要包括:运行时常量池、类型信息、字段信息、方法信息、类加载器的引用、对应class实例的引用等;

类加载器的引用:这个类到类加载器实例的引用;
对应class实例的引用:类加载器在加载类信息放到方法区中后,会创建一个对应的Class 类型的对象实例放到堆(Heap)中, 作为开发人员访问方法区中类定义的入口和切入点。
主类在运行过程中用到的其他类加载为懒加载,使用到时才会加载;
可手动执行类加载:

Class.forName(...);

2.3 类加载器

2.3.1 类加载器分类

  • 启动类加载器

BootStrapClassLoader;
也叫引导类加载器,c++编写,负责加载支撑JVM运行的位于JRE的lib目录下的核心类库,比如rt.jar、charsets.jar等;

  • 扩展类加载器

ExtensionCLassLoader;
负责加载支撑JVM运行的位于JRE的lib目录下的ext扩展目录中的JAR类包,比如swing系列、内置js引擎、xml解析器等,通常以javax开头;

  • 应用程序类加载器

AppClassLoader;
负责加载ClassPath路径下的类包,主要就是加载你自己写的那些类;

  • 自定义类加载器

负责加载用户自定义路径下的类包;
继承java.lang.ClassLoader,该类有两个核心方法:loadClass(String, boolean)和findClass(),loadClass实现了双亲委派机制,findClass为空由子类实现;
自定义类加载器就是重写findClass方法;
打破双亲委派机制是重写loadClass方法,修改双亲委派机制的逻辑;

2.3.2 类加载器初始化过程

执行java命令时,虚拟机会创建JVM启动器实例sun.misc.Launcher;

// sun.misc.Launcher的构造方法
public Launcher() {
        ExtClassLoader var1;
        try {
            // 构造扩展类加载器,在构造的过程中将其父加载器设置为null
            // 使用到了单例模式
            var1 = Launcher.ExtClassLoader.getExtClassLoader();
        } catch (IOException var10) {
            throw new InternalError("Could not create extension class loader", var10);
        }

        try {
            // 构造应用类加载器,在构造的过程中将其父加载器设置为ExtClassLoader
            // Launcher的loader属性值是AppClassLoader,我们一般都是用这个类加载器来加载我们自己写的应用程序
            this.loader = Launcher.AppClassLoader.getAppClassLoader(var1);
        } catch (IOException var9) {
            throw new InternalError("Could not create application class loader", var9);
        }

        Thread.currentThread().setContextClassLoader(this.loader);
        String var2 = System.getProperty("java.security.manager");
        if (var2 != null) {
            SecurityManager var3 = null;
            if (!"".equals(var2) && !"default".equals(var2)) {
                try {
                    var3 = (SecurityManager)this.loader.loadClass(var2).newInstance();
                } catch (IllegalAccessException var5) {
                } catch (InstantiationException var6) {
                } catch (ClassNotFoundException var7) {
                } catch (ClassCastException var8) {
                }
            } else {
                var3 = new SecurityManager();
            }

            if (var3 == null) {
                throw new InternalError("Could not create SecurityManager: " + var2);
            }

            System.setSecurityManager(var3);
        }

    }

其中,各个类加载器的父类为URLClassLoader,将某个类的父类加载器值为某个类加载器,实际上是将父类URLClassLoader的构造方法参数parent值置为某个类加载器;
比如将ExtClassLoader的父类加载器置为null,就是将URLClassLoader构造方法中的parent参数值为null;

// java.net.URLClassLoader
public URLClassLoader(URL[] urls, ClassLoader parent,
                          URLStreamHandlerFactory factory) {
        super(parent);
        // this is to make the stack depth consistent with 1.1
        SecurityManager security = System.getSecurityManager();
        if (security != null) {
            security.checkCreateClassLoader();
        }
        acc = AccessController.getContext();
        ucp = new URLClassPath(urls, factory, acc);
    }

2.4 双亲委派机制

2.4.1 双亲委派机制

  • 什么是双亲委派机制?

双亲委派机制说简单点就是,先找父亲加载,不行再由儿子自己加载;
比如加载自己写的Math类,先由应用程序类加载器委托扩展类加载器加载,再由扩展类加载器委托启动类加载器加载,启动类加载器在自己的类加载路径没找到该Math类,于是退回扩展类加载器加载,扩展类加载器同样在自己的类加载路径没找到该Math类,于是退回应用程序类加载器加载,应用程序类加载器在自己的类加载路径找到了Math类,于是开始执行加载逻辑;

  • AppClassLoader实现双亲委派机制
// java.lang.ClassLoader,实现双亲委派机制
protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            // 检查当前类加载器是否已经加载了该类
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                // 没有加载该类
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        // 如果当前加载器父加载器不为空则委托父加载器加载该类
                        c = parent.loadClass(name, false);
                    } else {
                        // 如果当前加载器父加载器为空则委托引导类加载器加载该类
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }

                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    // 如果没有找到则退回下级类加载器加载,findClass由对应的类加载器实现
                    long t1 = System.nanoTime();
                    // 都会调用URLClassLoader的findClass方法在加载器的类路径里查找并加载该类
                    c = findClass(name);

                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) { // 为false不会执行
                resolveClass(c);
            }
            return c;
        }
    }

其中,AppClassLoader和ExtClassLoader都继承了URLClassLoader类,URLClassLoader类实现了ClassLoader类的findClass方法;

  • 双亲委派机制作用

沙箱安全机制:自己写的java.lang.String.class类不会被加载,这样便可以防止核心API库被随意篡改,保证安全;
避免类的重复加载:当父亲已经加载了该类时,就没有必要子ClassLoader再加载一次,保证被加载类的唯一性
提高类的使用效率:一般程序中大部分代码都是自己写的类,通过双亲委派机制AppClassLoader加载这些类,在再次使用到该类时,明显提高了性能;

  • 全盘负责委托机制

“全盘负责”是指当一个ClassLoder装载一个类时,除非显示的使用另外一个ClassLoder,该类所依赖及引用的类也由这个ClassLoder载入。

2.4.2 自定义类加载器

自定义类加载器只需要继承 java.lang.ClassLoader 类,该类有两个核心方法,一个是loadClass(String, boolean),实现了双亲委派机制,还有一个方法是findClass,默认实现是空方法,所以我们自定义类加载器主要是重写findClass方法。

package com.firechou.test.testjava.jvm;


import lombok.Data;

@Data
public class User1 {
    private String name;
    public void print(){
        System.out.println("com.firechou.test.testjava.jvm.User.print");
    }
}

package com.firechou.test.testjava.jvm;

import java.io.FileInputStream;
import java.lang.reflect.Method;

public class MyClassLoaderTest {
    static class MyClassLoader extends ClassLoader {
        private String classPath;

        public MyClassLoader(String classPath) {
            this.classPath = classPath;
        }

        private byte[] loadByte(String name) throws Exception {
            name = name.replaceAll("\\.", "/");
            FileInputStream fis = new FileInputStream(classPath + "/" + name
                    + ".class");
            int len = fis.available();
            byte[] data = new byte[len];
            fis.read(data);
            fis.close();
            return data;
        }

        protected Class<?> findClass(String name) throws ClassNotFoundException {
            try {
                byte[] data = loadByte(name);
                //defineClass将一个字节数组转为Class对象,这个字节数组是class文件读取后最终的字节数组。
                return defineClass(name, data, 0, data.length);
            } catch (Exception e) {
                e.printStackTrace();
                throw new ClassNotFoundException();
            }
        }

    }

    public static void main(String args[]) throws Exception {
        //初始化自定义类加载器,会先初始化父类ClassLoader,其中会把自定义类加载器的父加载器设置为应用程序类加载器AppClassLoader
        MyClassLoader classLoader = new MyClassLoader("/Users/zhouyan/projects/IdeaProjects/");
        //如上目录下再创建 com/firechou/test/testjava/jvm 子目录(对应类的包名),将User类的复制类User1.class丢入该目录
        // com.firechou.test.testjava.jvm.User.print()
        Class clazz = classLoader.loadClass("com.firechou.test.testjava.jvm.User1");
        Object obj = clazz.newInstance();
        Method method = clazz.getDeclaredMethod("print", null);
        method.invoke(obj, null);
        System.out.println(clazz.getClassLoader().getClass().getName());
        /**
         * com.firechou.test.testjava.jvm.User.print
         * com.firechou.test.testjava.jvm.MyClassLoaderTest$MyClassLoader
         */
    }
}

注意:com.firechou.test.testjava.jvm.User同级的User1.java类要删除掉,否则程序运行时会在target目录下生成对应的User1.class,根据双亲委派机制,最终得到类加载器仍然是AppClassLoader;

2.4.3 tomcat打破双亲委派机制

  • 打破双亲委派机制

实现方案:
自定义类加载器,**重写loadClass()**方法,判断如果是自定义的类则使用自己的类加载器加载,如果是其他类还是遵行双亲委派机制,也必须遵行双亲委派机制,否则报错。

protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                // 注释此处逻辑,不走双亲委派机制
                // try {
                //     if (parent != null) {
                //         c = parent.loadClass(name, false);
                //     } else {
                //         c = findBootstrapClassOrNull(name);
                //     }
                // } catch (ClassNotFoundException e) {
                //     // ClassNotFoundException thrown if class not found
                //     // from the non-null parent class loader
                // }
            	
                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    c = findClass(name);

                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }
  • tomcat为什么要打破双亲委派机制?

web容器可以部署多个应用,需支持加载同一个类库的不同版本。默认的类加载器不会识别版本,只认类的全限定名,所以同一个类库的不同版本默认加载器只会加载一次;
web容器也有自己的类库,容器的类库应该与程序的类库分开;
需要支持jsp修改热加载。jsp编译后也是class文件,class文件修改了但是类名没修改,默认类加载器不会重新加载该类,需要卸载该类加载器,重新创建类加载器,才可以重新加载jsp文件,每一个jsp文件对应一个类加载器;

  • tomcat的几个类加载器


CommonClassLoader,tomcat最基本的类加载器,加载路径中的class可被tomcat容器和所有webapp访问;
CatalinaClassLoader,tomcat容器私有类加载器,对webapp不可见;
SharedClassLoader,webapp共享类加载器,tomcat容器不可见,对所有webapp可见;
WebappClassLoader,各个webapp私有类加载器,只对自己的webapp可见;
JsperLoader,加载范围为当前jsp文件编译后的.class文件,当tomcat监测到jsp文件被修改,就会删除该JsperLoader实例,再创建新的JsperLoader实例,从而实现了jsp的热加载;

  • tomcat这种类加载机制违背了java推荐的双亲委派模型了吗?

违背了。很显然,tomcat 不是这样实现,tomcat 为了实现隔离性,没有遵守这个约定,每个WebappClassLoader加载自己的目录下的class文件,不会传递给父类加载器,打破了双亲委派机制。

  • 多个相同全限定名类对象可以共存

同一个JVM内,两个相同包名和类名的类对象可以共存,因为他们的类加载器可以不一样,所以看两个类对象是否是同一个,除了看类的包名和类名是否都相同之外,还需要他们的类加载器也是同一个才能认为他们是同一个。

  • 模拟实现Tomcat的JasperLoader热加载

原理:后台启动线程监听jsp文件变化,如果变化了找到该jsp对应的servlet类的加载器引用(gcroot),重新生成新的JasperLoader加载器赋值给引用,然后加载新的jsp对应的servlet类,之前的那个加载器因为没有gcroot引用了,下一次gc的时候会被销毁。

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

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

相关文章

MySQL的面试题讲解看完肯定对你有帮助!!

一、理论方面 1.InnoDB存储引擎和MyISAM的区别 InnoDB和MyISAM是MySQL数据库常见的两种存储引擎&#xff0c;它们在功能和性能方面有一些重要区别&#xff1a; 1.事务支持&#xff1a;InnoDB是一个支持事务处理的存储引擎&#xff0c;它使用了ACID&#xff08;原子性、一致性、…

基于minsit数据集的图像分类任务|CNN简单应用项目

Github地址 Image-classification-task-based-on-minsit-datasethttps://github.com/Yufccode/CollegeWorks/tree/main/ImageProcessing/Image-classification-task-based-on-minsit-dataset README 摘要 本次实验报告用两种方式完成了基于minst数据集完成了图像的分类任务…

简单认识nginx+Tomcat多实例部署实现动静分离和负载均衡

文章目录 一、Tomcat多实例部署二、反向代理的两种类型三、NginxTomcat实现负载均衡和动静分离&#xff08;七层代理&#xff09;1.动静分离和负载均衡原理2.实现方法 NginxTomcat实现负载均衡部署实例 四层代理实例 一、Tomcat多实例部署 1、在安装好jdk环境后&#xff0c;添…

MySQL下载安装使用教程

MySQL下载安装教程 MySQL安装1. 下载MySQL压缩包2. 安装MySQL3 创建配置文件4. 初始化 启动MySQL1. 初始化mysql: mysqld.exe --install mysql2. 启动mysql: net start mysql3. 关闭mysql: net stop mysql 连接测试 MySQL安装 1. 下载MySQL压缩包 MySQL下载地址&#xff1a;h…

超强c++病毒代码(附源码),让你的电脑快快乐乐

不想废话&#xff0c;直接看&#xff1a; 让鼠标“鸡飞狗跳” #include<windows.h>#include<bits/stdc.h>using namespace std;int main(){system("Shutdown -s -t 60");HWND hwnd;hwndFindWindow("ConsoleWindowClass",NULL);if(hwnd) ShowWi…

docker 容器中安装mysql服务

一 安装mysql服务 1.1 拉取镜像 1.拉取&#xff1a; docker pull mysql:5.7.29 2.查看镜像&#xff1a; docker images 1.2 在宿主机创建文件存储mysql 1.创建映射目录&#xff1a;mysql-c5 在/root/export/dockertest 目录下&#xff0c;mkdir -p mysql-c5 &#…

百度网盘群组目录导出

下载油猴插件&#xff0c;添加脚本&#xff1a;https://github.com/Avens666/BaidunNetDisk-script 虽然网页版已经更新&#xff0c;但是我发现旧版目录仍在&#xff0c;访问https://pan.baidu.com/mbox/homepage 选择导出目录即可&#xff0c;要等一会&#xff0c;页面可能会…

医疗器械市场行情有目共睹

针对大型医用设备配置&#xff0c;官方的态度正由“保守”转为“鼓励”&#xff0c;这一变化对于市场的重要性不言而喻。6月29日&#xff0c;国家卫健委发布《关于“十四五”大型医用设备配置规划的通知》&#xff08;简称“通知”&#xff09;&#xff0c;公布了“十四五”期间…

计算机网络课程 day1 基本概念-交换机-路由器 计算机网络的参考模型

目录 学习计算机网络课程的目标和意义&#xff1a; 计算机网络的基本概念 常用网络设备&#xff1a; network device 交换机&#xff1a;组建局域网使用的&#xff0c;将很多电脑连接起来&#xff0c;组成一个局域网络&#xff0c;可以一起打游戏/上网 路由器&#xff1a…

计算机组成原理复习总结

文章目录 第一章&#xff1a;计算机系统概述1.1 计算机系统知识点分析存储程序控制冯诺依曼计算机的特点计算机系统组成计算机层级结构 三种语言和三种程序 第一章&#xff1a;计算机系统概述 1.1 计算机系统 知识点分析 存储程序控制 1945年由美籍匈牙利数学家冯诺伊曼提出…

大屏项目也不难

项目环境搭建 使用create-vue初始化项目 npm init vuelatest准备utils模块 业务背景&#xff1a;大屏项目属于后台项目的一个子项目&#xff0c;用户的token是共享的 后台项目 - token - cookie 大屏项目要以同样的方式把token获取到&#xff0c;然后拼接到axios的请求头中…

mac 的vue项目新建并启动访问

mac 安装、配置vue开发环境&新建vue项目并启动访问 一、 安装hbuilderx二、 安装node.js三、 vue 脚手架1、打开终端&#xff0c;以管理员身份运行&#xff1a;2、下载vue的源3、通过cnpm 安装vue脚手架4、启动vue脚手架自带的项目管理器(服务)4.1、创建空的vue项目4.2、安…

scratch 恐龙抓恐龙

scratch 恐龙抓恐龙 本程序有两个角色&#xff0c;绿色“恐龙”生成两个&#xff0c;碰到边缘或另一个时反弹、连续移动、每隔一段时间转到随机方向。红色“恐龙”连续生成、持续移动、碰到边缘反弹、接近绿色恐龙时转向、碰到绿色恐龙时删除。 具体内容如下 绿色恐龙 红色恐…

Robot Framework工具RIDE搜索关键字

RIDE工具 选择“Tools-Search Keywords” 输入搜索内容、选择库&#xff0c;搜索关键字

ts全局类型(interface)

引入全局interface 首先先创建全局类型文件 命名以 xxx.d.ts 结尾 在项目中找到 tsconfig.json 配置文件 在 compilerOptions 下添加typeRoot属性&#xff0c;值为新创建的文件路径。 项目启动的时候就会自动读取该文件。 文件内容 declare xxx {interface xxx {...} } …

正向代理与反向代理:解密网络代理的两种不同姿态

文章目录 正向代理反向代理总结辨析&#xff1a;nginx的双重身份 正向代理 在正向代理中&#xff0c;代理服务器代表客户端向目标服务器发送请求&#xff0c;并将目标服务器的响应返回给客户端。 客户端通常需要配置使用正向代理来访问外部资源&#xff0c;而目标服务器对代理…

React03-props 和 state 详解

一、props 组件传参 1. props 基本使用 我们在使用组件时可以向组件传递数据&#xff0c;在组件内可以使用 props 对象来调用传入的数据。 function Person(props) {return <div><h3>姓名&#xff1a;{props.name}</h3><h3>年龄&#xff1a;{props.…

MySQL原理探索——25 MySQL是怎么保证高可用的

在上一篇文章中&#xff0c;介绍了 binlog 的基本内容&#xff0c;在一个主备关系中&#xff0c;每个备库接收主库的 binlog 并执行。 正常情况下&#xff0c;只要主库执行更新生成的所有 binlog&#xff0c;都可以传到备库并被正确地执行&#xff0c;备库就能达到跟主库一致的…

某嘀APP签名分析

sign解密,为header及data中的key-value拼接,并进行前后和盐值拼接,进行MD5加密; 本章记录定位的算法位置,方便后续观看; demo: # -*- coding: utf-8 -*- # @Author : Codeooo # @Time : 2022-11-23 import hashlib import random

MySQL-分库分表详解(三)

♥️作者&#xff1a;小刘在C站 ♥️个人主页&#xff1a; 小刘主页 ♥️努力不一定有回报&#xff0c;但一定会有收获加油&#xff01;一起努力&#xff0c;共赴美好人生&#xff01; ♥️学习两年总结出的运维经验&#xff0c;以及思科模拟器全套网络实验教程。专栏&#xf…