【JVM详解二】常量池
一、常量池概述
- class文件常量池
- 运行时常量池
- 字符串常量池
- 基本类型包装类常量池
- 每个 class 的字节码文件中都有一个常量池,里面是编译后即知的该 class 会用到的字面量与符号引用,这就是 class 文件常量池。JVM加载 class ,会将其类信息,包括 class 文件常量池置于方法区中。
- class 类信息及其 class 文件常量池是字节码的二进制流,它代表的是一个类的静态存储结构,JVM加载类时,需要将其转换为方法区中的 java.lang.Class 类的对象实例;同时,会将 class 文件常量池中的内容导入运行时常量池。
- 运行时常量池中的常量对应的内容只是字面量,比如一个"字符串",它还不是 String 对象;当 Java 程序在运行时执行到这个"字符串"字面量时,会去字符串常量池里找该字面量的对象引用是否存在,存在则直接返回该引用,不存在则在Java堆里创建该字面量对应的 String 对象,并将其引用置于字符串常量池中,然后返回该引用。
- Java的基本数据类型中,除了两个浮点数类型,其他的基本数据类型都在各自内部实现了常量池,但都在[-128~127]这个范围内。
二、class文件常量池
java的源代码 .java 文件在编译之后会生成 .class 文件,class 文件需要严格遵循 JVM 规范才能被 JVM 正常加载,它是一个二进制字节流文件,里面包含了class文件常量池的内容。
class文件常量池诞生于编译时,存在于class文件中,存放符号引用和字面量。
2.1 查看一个class文件内容
jdk 提供了 javap 命令,用于对class文件进行反汇编,输出类相关信息。该命令用法如下:
用法: javap <options> <classes>
其中, 可能的选项包括:-help --help -? 输出此用法消息-version 版本信息-v -verbose 输出附加信息-l 输出行号和本地变量表-public 仅显示公共类和成员-protected 显示受保护的/公共类和成员-package 显示程序包/受保护的/公共类和成员 (默认)-p -private 显示所有类和成员-c 对代码进行反汇编-s 输出内部类型签名-sysinfo 显示正在处理的类的系统信息 (路径, 大小, 日期, MD5 散列)-constants 显示最终常量-classpath <path> 指定查找用户类文件的位置-cp <path> 指定查找用户类文件的位置-bootclasspath <path> 覆盖引导类文件的位置 例如,我们可以编写一个简单的类,如下:
public class Student {private final String name = "张三";private final int entranceAge = 18;private String evaluate = "优秀";private int scores = 95;private Integer level = 5;public String getEvaluate() {return evaluate;}public void setEvaluate(String evaluate) {String tmp = "+";this.evaluate = evaluate + tmp;}public int getScores() {return scores;}public void setScores(int scores) {final int base = 10;System.out.println("base:" + base);this.scores = scores + base;}public Integer getLevel() {return level;}public void setLevel(Integer level) {this.level = level;}
} 对其进行编译和反汇编:
javac Student.java
javap -v Student.class
# over 得到以下反汇编结果:
Classfile /home/work/sources/open_projects/lib-zc-crypto/src/test/java/Student.classLast modified 2021-1-4; size 1299 bytesMD5 checksum 06dfdad9da59e2a64d62061637380969Compiled from "Student.java"
public class Studentminor version: 0major version: 52flags: ACC_PUBLIC, ACC_SUPER
Constant pool:#1 = Methodref #19.#48 // java/lang/Object."<init>":()V#2 = String #49 // 张三#3 = Fieldref #18.#50 // Student.name:Ljava/lang/String;#4 = Fieldref #18.#51 // Student.entranceAge:I#5 = String #52 // 优秀#6 = Fieldref #18.#53 // Student.evaluate:Ljava/lang/String;#7 = Fieldref #18.#54 // Student.scores:I#8 = Methodref #55.#56 // java/lang/Integer.valueOf:(I)Ljava/lang/Integer;#9 = Fieldref #18.#57 // Student.level:Ljava/lang/Integer;#10 = String #58 // +#11 = Class #59 // java/lang/StringBuilder#12 = Methodref #11.#48 // java/lang/StringBuilder."<init>":()V#13 = Methodref #11.#60 // java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;#14 = Methodref #11.#61 // java/lang/StringBuilder.toString:()Ljava/lang/String;#15 = Fieldref #62.#63 // java/lang/System.out:Ljava/io/PrintStream;#16 = String #64 // base:10#17 = Methodref #65.#66 // java/io/PrintStream.println:(Ljava/lang/String;)V#18 = Class #67 // Student#19 = Class #68 // java/lang/Object#20 = Utf8 name#21 = Utf8 Ljava/lang/String;#22 = Utf8 ConstantValue#23 = Utf8 entranceAge#24 = Utf8 I#25 = Integer 18#26 = Utf8 evaluate#27 = Utf8 scores#28 = Utf8 level#29 = Utf8 Ljava/lang/Integer;#30 = Utf8 <init>#31 = Utf8 ()V#32 = Utf8 Code#33 = Utf8 LineNumberTable#34 = Utf8 getEvaluate#35 = Utf8 ()Ljava/lang/String;#36 = Utf8 setEvaluate#37 = Utf8 (Ljava/lang/String;)V#38 = Utf8 getScores#39 = Utf8 ()I#40 = Utf8 setScores#41 = Utf8 (I)V#42 = Utf8 getLevel#43 = Utf8 ()Ljava/lang/Integer;#44 = Utf8 setLevel#45 = Utf8 (Ljava/lang/Integer;)V#46 = Utf8 SourceFile#47 = Utf8 Student.java#48 = NameAndType #30:#31 // "<init>":()V#49 = Utf8 张三#50 = NameAndType #20:#21 // name:Ljava/lang/String;#51 = NameAndType #23:#24 // entranceAge:I#52 = Utf8 优秀#53 = NameAndType #26:#21 // evaluate:Ljava/lang/String;#54 = NameAndType #27:#24 // scores:I#55 = Class #69 // java/lang/Integer#56 = NameAndType #70:#71 // valueOf:(I)Ljava/lang/Integer;#57 = NameAndType #28:#29 // level:Ljava/lang/Integer;#58 = Utf8 +#59 = Utf8 java/lang/StringBuilder#60 = NameAndType #72:#73 // append:(Ljava/lang/String;)Ljava/lang/StringBuilder;#61 = NameAndType #74:#35 // toString:()Ljava/lang/String;#62 = Class #75 // java/lang/System#63 = NameAndType #76:#77 // out:Ljava/io/PrintStream;#64 = Utf8 base:10#65 = Class #78 // java/io/PrintStream#66 = NameAndType #79:#37 // println:(Ljava/lang/String;)V#67 = Utf8 Student#68 = Utf8 java/lang/Object#69 = Utf8 java/lang/Integer#70 = Utf8 valueOf#71 = Utf8 (I)Ljava/lang/Integer;#72 = Utf8 append#73 = Utf8 (Ljava/lang/String;)Ljava/lang/StringBuilder;#74 = Utf8 toString#75 = Utf8 java/lang/System#76 = Utf8 out#77 = Utf8 Ljava/io/PrintStream;#78 = Utf8 java/io/PrintStream#79 = Utf8 println
{方法信息,此处省略...
}
SourceFile: "Student.java" 其中的 Constant pool 就是 class 文件常量池,使用 # 加数字标记每个“常量”。
2.2 class文件常量池的内容
class 文件常量池存放的是该 class 编译后即知的,在运行时将会用到的各个“常量”。注意这个常量不是编程中所说的 final 修饰的变量,而是字面量和符号引用,如下图所示:
2.2.1 字面量
字面量大约相当于Java代码中的双引号字符串和常量的实际的值,包括:
1)文本字符串
即代码中用双引号包裹的字符串部分的值。
例如刚刚的例子中,有三个字符串:"张三","优秀","+",它们在class文件常量池中分别对应:
#49 = Utf8 张三
#52 = Utf8 优秀
#58 = Utf8 +
2)用final修饰的成员变量的值
例如,private static final int entranceAge = 18;
这条语句定义了一个final常量entranceAge,它的值是18,对应在class文件常量池中就有:
#25 = Integer 18 注意,只有final修饰的成员变量如entranceAge,才会在常量池中存在对应的字面量。而非final的成员变量scores,以及局部变量base(即使使用final修饰了),它们的字面量都不会在常量池中定义。
2.2.2 符号引用
1)类和接口的全限定名
例如:
#11 = Class #59 // java/lang/StringBuilder
#59 = Utf8 java/lang/StringBuilder
2)方法的名称和描述符
例如:
#38 = Utf8 getScores
#39 = Utf8 ()I#40 = Utf8 setScores
#41 = Utf8 (I)V
以及这种对其他类的方法的引用:
#8 = Methodref #55.#56 // java/lang/Integer.valueOf:(I)Ljava/lang/Integer;#55 = Class #69 // java/lang/Integer
#69 = Utf8 java/lang/Integer#56 = NameAndType #70:#71 // valueOf:(I)Ljava/lang/Integer;
#70 = Utf8 valueOf
#71 = Utf8 (I)Ljava/lang/Integer;
3)字段的名称和描述符
例如:
#3 = Fieldref #18.#50 // Student.name:Ljava/lang/String;#18 = Class #67 // Student
#67 = Utf8 Student#50 = NameAndType #20:#21 // name:Ljava/lang/String;
#20 = Utf8 name
#21 = Utf8 Ljava/lang/String;
以及这种局部变量:
#64 = Utf8 base:10
三、运行时常量池
JVM在加载某个 class 的时候,需要完成以下任务:
- 通过该class的全限定名来获取它的二进制字节流,即读取其字节码文件。其内容包括在上文class文件常量池中介绍的class文件常量池。
- 将读入的字节流从静态存储结构转换为方法区中的运行时的数据结构。
- 在Java堆中生成该 class 对应的类对象,代表该 class 原信息。这个类对象的类型是 java.lang.Class,它与普通对象不同的地方在于,普通对象一般都是在 new 之后创建的,而类对象是在类加载的时候创建的,且是单例。
上述过程的第二步,就包含了将 class 文件常量池内容导入运行时常量池。class 文件常量池是一个 class 文件对应一个常量池,而运行时常量池只有一个,多个 class 文件常量池中的相同字符串只会对应运行时常量池中的一个字符串。
运行时常量池除了导入 class 文件常量池的内容,还会保存符号引用对应的直接引用(实际内存地址)。这些直接引用是JVM在类加载之后的链接(验证、准备、解析)阶段从符号引用翻译过来的。
此外,运行时常量池具有动态性的特征,它的内容并不是全部来源与编译后的 class 文件,在运行时也可以通过代码生成常量并放入运行时常量池。比如 String.intern() 方法(String.intern()方法的分析见下文)。
要注意的是,运行时常量池中保存的“常量”依然是字面量和符号引用。比如字符串,这里放的仍然是单纯的文本字符串,而不是 String 对象。
四、字符串常量池
字符串常量池,是 JVM 用来维护字符串实例的一个引用表。在 HotSpot 虚拟机中,它被实现为一个全局的 StringTable ,底层是一个 c++ 的 hashtable 。它将字符串的字面量作为 key ,实际堆中创建的 String 对象的引用作为 value 。
字符串常量池在逻辑上属于方法区,但 JDK1.7 开始,就被挪到了堆区。
String 的字面量被导入 JVM 的运行时常量池时,并不会马上试图在字符串常量池加入对应 String 的引用,而是等到程序实际运行时,要用到这个字面量对应的 String 对象时,才会去字符串常量池试图获取或者加入 String 对象的引用。因此它是懒加载的。
如前所述,class 文件常量池和运行时常量池中,都没有直接存储字面量对应的实际对象,比如 String 对象。那么 String 对象到底是什么时候在哪里创建的呢?
4.1 创建字符串对象的方式
4.1.1 字面量赋值
直接用双引号字面值的方式创建字符串。
String str1 = "test";
String str2 = "test";
//str1 == str2 ? true
System.out.println("str1 == str2 ? "+ (str1 == str2));
当Java虚拟机启动成功后,上面的字符串"test"的字面量已经进入运行时常量池;
然后主线程开始运行,执行到 String str1 = “test"; 这条语句时,JVM会根据运行时常量池中的这个字面量去字符串常量池寻找其中是否有该字面量对应的 String 对象的引用,注意是引用。 如果没找到,就会去 Java 堆创建一个值为 “test" 的 String 对象,并将该对象的引用保存到字符串常量池,然后返回该引用;如果找到了,说明之前已经有其他语句通过相同的字面量赋值创建了该 String 对象,直接返回引用即可。
接着执行到 String str2 = “test"; 这条语句,JVM 成功找到上一步生成的 String 对象的引用。所以 str1 == str2 返回 true。
4.1.2 new String 创建字符串
使用new String 创建字符串,会先在常量池中创建,再在堆中创建。
String str3 = new String("tony");
String str4 = "tony";
//str3 == str4 ? false
System.out.println("str3 == str4 ? " + (str3 == str4));
- 第一句中JVM拿字面量 “tony" 去字符串常量池试图获取其对应 String 对象的引用,因为是首次执行,所以没找到,于是在堆中创建了一个 “tony" 的 String 对象,并将其引用保存到字符串常量池中,然后返回;
- 返回之后,因为new的存在,JVM又在堆中创建了与 “tony" 等值的另一个 String 对象。
- 因此这条语句创建了两个 String 对象,它们值相等,都是 “tony" ,但是引用(内存地址)不同,所以 str3 == str4 结果是false。
4.1.3 String.intern() 方法
String.intern() 是一个 native (本地) 方法,用来处理字符串常量池中的字符串对象引用。它的工作流程可以概括为以下两种情况:
- 常量池中已有相同内容的字符串对象:如果字符串常量池中已经有一个与调用 intern() 方法的字符串内容相同的 String 对象,intern() 方法会直接返回常量池中该对象的引用。
- 常量池中没有相同内容的字符串对象:如果字符串常量池中还没有一个与调用 intern() 方法的字符串内容相同的对象,intern() 方法会将当前字符串对象的引用添加到字符串常量池中,并返回该引用。
总结:
- intern() 方法的主要作用是确保字符串引用在常量池中的唯一性。
- 当调用 intern() 时,如果常量池中已经存在相同内容的字符串,则返回常量池中已有对象的引用;否则,将该字符串添加到常量池并返回其引用。
// 创建一个新的 "Java" 对象,并把该"Java" 对象的引用添加到字符串常量池,s1 指向该引用
String s1 = "Java";
// s2 也指向字符串常量池中 "Java" 对象的引用,和 s1 是同一个对象
String s2 = s1.intern();
// 在堆中创建一个新的 "Java" 对象,s3 指向它
String s3 = new String("Java");
// s4 指向字符串常量池中的 "Java" 对象的引用,和 s1 是同一个对象
String s4 = s3.intern();
// s1 和 s2 指向的是同一个常量池中的对象
System.out.println(s1 == s2); // true
// s3 指向堆中的对象,s4 指向字符串常量池中的对象引用,所以不同
System.out.println(s3 == s4); // false
// s1 和 s4 都指向常量池中的同一个对象
System.out.println(s1 == s4); // true 对于第14行 s1 == s4 要注意,JDK1.6和JDK1.7开始,String.intern()的执行逻辑是不一样的。
- JDK1.6 以前 intern() 可以查找调用该 intern 的对象是否存在字符串常量池,如果存在 返回其引用;如果 不存在,则添加进去并返回引用;
- JDK1.7及以后 intern() 查找是否存在 如果存在 则返回其引用;但是,如果字符串常量池不存在(堆区一定存在,不存在就无法调用该方法),那么就将该堆区对象的引用加入字符串常量池。
因此对于第14行 s1 == s4 ,如果是JDK1.6及以前的版本,结果就是false;而如果是JDK1.7开始的版本,结果就是true。
4.1.4 new String 相加
String str5 = new String("good") + new String("morning");
str5.intern();
String str6 = "goodmorning";
//str5 == str6 ? true str5 == str5.intern() ? true
System.out.println("str5 == str6 ? " + (str5 == str6) + " str5 == str5.intern() ? " + (str5 == str5.intern()));
第一句会分别把不在常量池的"good"和"morning"两个常量对象加入常量池,拼接起来"goodmorning"不在常量池,而"goodmorning"并不会在常量池中创建常量或引用。str5.intern()检查常量池是否有字符串"goodmorning",没有,把引用添加到常量池中,str6检查常量池有字符串"goodmorning",返回它的引用,所以两个判断的结果都是true。
String str5 = new String(“good”) + new String(“morning”) 被JVM编译优化后后如下:
String str5 = (new StringBuilder()).append(new String("good")).append(new String("morning")).toString();
底层是一个StringBuilder在进行把两个对象拼接在一起,最后栈中str5指向堆中的"goodmorning",其实是StringBuilder对象:
4.2 字符串常量池是否会被GC
字符串常量池本身不会被GC,但其中保存的引用所指向的 String 对象们是可以被回收的。否则字符串常量池总是"只进不出",那么很可能会导致内存泄露。
在 HotSpot 的字符串常量池实现 StringTable 中,提供了相应的接口用于支持GC,不同的GC策略会在适当的时候调用它们。一般实在Full GC的时候,额外调用StringTable的对应接口做可达性分析,将不可达的String对象的引用从StringTable中移除掉并销毁其指向的String对象。
五、封装类常量池
除了字符串常量池,Java 的基本类型的封装类大部分也都实现了常量池。包括 Byte、Short、Integer、Long、Character、Boolean,注意,浮点数据类型Float、Double是没有常量池的。
封装类的常量池是在各自内部类中实现的,比如 IntegerCache(Integer的内部类),自然也位于堆区。
要注意的是,这些常量池是有范围的:
- Byte , Short , Integer , Long : [-128~127]
- Character : [0~127]
- Boolean : [True, False]
例如下面的代码,注意其结果:
Integer i1 = 127;
Integer i2 = 127;
System.out.println(i1 == i2); //trueInteger i3 = 128;
Integer i4 = 128;
System.out.println(i3 == i4); //falseInteger i5 = -128;
Integer i6 = -128;
System.out.println(i5 == i6); //trueInteger i7 = -129;
Integer i8 = -129;
System.out.println(i7 == i8); //false
相关文章:
【JVM详解二】常量池
一、常量池概述 JVM的常量池主要有以下几种: class文件常量池运行时常量池字符串常量池基本类型包装类常量池 它们相互之间关系大致如下图所示: 每个 class 的字节码文件中都有一个常量池,里面是编译后即知的该 class 会用到的字面量与符号引…...
Leetcode - 149双周赛
目录 一、3438. 找到字符串中合法的相邻数字二、3439. 重新安排会议得到最多空余时间 I三、3440. 重新安排会议得到最多空余时间 II四、3441. 变成好标题的最少代价 一、3438. 找到字符串中合法的相邻数字 题目链接 本题有两个条件: 相邻数字互不相同两个数字的的…...
Java爬虫:打造高效的数据抓取利器——详解详情接口设计与实现
在当今数字化时代,数据如同黄金般珍贵。无论是企业进行市场调研、竞争对手分析,还是研究人员收集信息,数据的需求无处不在。而爬虫技术,作为一种高效的数据抓取手段,成为了众多开发者手中的利器。本文将深入探讨如何使…...
蓝桥杯K倍区间(前缀和与差分,取模化简)
输入 5 2 1 2 3 4 5 输出 6 思路:首先由连续子串和可以想用前缀和,由于加减法总和取模和分别取模结果不受影响,所以我们前缀和之后直接取模方便观察性质,本题前缀和:1,3,6,10&#…...
CEF132 编译指南 MacOS 篇 - depot_tools 安装与配置 (四)
1. 引言 在 CEF132(Chromium Embedded Framework)的编译过程中,depot_tools 扮演着举足轻重的角色。这套由 Chromium 项目精心打造的脚本和工具集,专门用于获取、管理和更新 Chromium 及其相关项目(包括 CEFÿ…...
Ubuntu 20.04 上安装 qBittorrent
qBittorrent 通过终端安装 系统更新系统升级在 Ubuntu 20.04 上添加 Qbittorent PPA系统更新Qbittorent 安装 Qbittorent 是一个开源且可免费使用的点对点比特流客户端。它体积小,不加载内存盘。众所周知,此应用程序可以在许多操作系统(例如…...
iPhone 在华销量大幅下挫
iPhone在乔布斯时代缔造的神话在中国正逐渐走向没落,挤牙膏式的升级方式类似于诺基亚的N70系列,毫无新意的创新能力,求稳着陆的经营理念,工艺和美学不再独领风骚,甚至拍照领域和AI增强计算,折叠屏等技术领域…...
【Ubuntu VScode Remote SSH 问题解决】Resolver error: Error: XHR failed
1. 问题描述 VScode使用remote ssh 远程服务器,报错类似: [12:06:01.219] Downloading VS Code server locally... [12:06:01.310] Resolver error: Error: XHR failedat k.onerror (vscode-file://vscode-app/private/var/folders/g1/cvs2rnpx60qc3b4…...
在JVM的栈(虚拟机栈)中,除了栈帧(Stack Frame)还有什么?
在JVM的栈(虚拟机栈)中,除了栈帧(Stack Frame),还有其他一些与方法调用相关的重要信息。栈的主要作用是存储方法调用的执行过程中的上下文信息,栈帧是其中最关键的组成部分。 栈的组成 栈帧&am…...
docker发布自己的镜像
官方node-red镜像: nodered/node-red - Docker Image 拉取v3版本: docker pull nodered/node-red:3.1.14 运行镜像: docker run --restartalways --privilegedtrue -d -p 1880:1880 -v node_red_data:/data --name mynodered nodered/n…...
【人工智能】解码语言之谜:使用Python构建神经机器翻译系统
《Python OpenCV从菜鸟到高手》带你进入图像处理与计算机视觉的大门! 解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 神经机器翻译(NMT)是近年来机器翻译领域的一项重大突破。它利用深度学习模型,特别是循环神经网络(RNN)和Transformer网络,以端到端的…...
JavaScript数组类型详解
目录 一、数组的基本概念 二、数组的类型 1. 基本数组类型: 2. 数字数组: 3. 字符串数组: 4. 对象数组: 5. 类型数组(TypedArray): 6. ArrayBuffer数组: 7. 类数组ÿ…...
【实战AI】利用deepseek 在mac本地部署搭建个人知识库
之前的文章中实现了本地通过ollma 部署deepseek R1:14b 模型,这里我想继续实现个人知识库,方便自己文档,数据的检索; 下载anythingLLM 地址: https://anythingllm.com/desktop 下载安装即可;…...
Spring Boot 3.4 中 MockMvcTester 的新特性解析
引言 在 Spring Boot 3.4 版本中,引入了一个全新的 MockMvcTester 类,使 MockMvc 测试可以直接支持 AssertJ 断言。本文将深入探讨这一新特性,分析它如何优化 MockMvc 测试并提升测试的可读性。 Spring MVC 示例 为了演示 MockMvcTester 的…...
Express 中间件
在构建 Web 应用程序时,中间件(Middleware)扮演着至关重要的角色。它允许你定义一系列的函数来处理 HTTP 请求和响应过程中的各种任务。Express.js 是 Node.js 上最流行的框架之一,以其简洁且强大的中间件机制著称。本文将深入探讨…...
PyCharm结合DeepSeek-R1
PyCharm结合DeepSeek-R1,打造专属 AI 编程助手 在程序员的日常工作中,提高编程效率、快速解决代码问题是重中之重。今天给大家分享一个强强联合的组合 ——PyCharm 插件 Continue 与 DeepSeek-R1,它们能帮你打造出强大的个人 AI 编程助手。 …...
AJAX XML技术详解
AJAX XML技术详解 引言 随着互联网技术的不断发展,前端与后端之间的交互需求日益增长。AJAX(Asynchronous JavaScript and XML)技术应运而生,成为实现前后端分离、提高页面响应速度的关键技术之一。本文将详细介绍AJAX XML技术,包括其原理、应用场景、优缺点等内容。 A…...
【openresty服务器】:源码编译openresty支持ssl,增加service系统服务,开机启动,自己本地签名证书,配置https访问
1,openresty 源码安装,带ssl模块 https://openresty.org/cn/download.html (1)PCRE库 PCRE库支持正则表达式。如果我们在配置文件nginx.conf中使用了正则表达式,那么在编译Nginx时就必须把PCRE库编译进Nginx…...
Java+vue前后端分离项目集群部署
一、项目概述 假设我们有一个前后端分离的项目,前端使用React或Vue框架,后端使用Spring Boot或Node.js。我们将分别部署前端和后端到集群环境中。 二、准备工作 1. 代码准备:确保前端和后端代码已经开发完成,并通过本地测试。 2…...
3. CSS中@scope
说说你对 CSS 中scope 的了解 <style>/* scope规则 */scope (#app) {.box {width: 100px;height: 100px;background-color: red;}} </style> <div id"app"><div class"box"></div> </div>CSS 中的scope 是一个相对较新…...
互联网大厂面试高频题-操作系统部分
前言 哈喽各位小伙伴们,本期小梁给大家带来了互联网大厂面试中操作系统部分的高频题,本文会以通俗易懂的语言以及图解形式描述,希望能给大家的面试带来一点帮助,祝大家offer拿到手软!!! 话不多说,我们立刻进入本期正题! 1 说说什么是操作系统吧。 答…...
Sentinel——Spring Boot 应用接入 Sentinel 后内存开销增长计算方式
接入 Sentinel 对 Spring Boot 应用的内存消耗影响主要取决于 规则数量、资源数量、监控粒度、并发量 等因素。 1. 核心内存消耗来源 (1) Sentinel 核心库 默认依赖:Sentinel Core 本身占用较小,通常在 10~50MB(取决于资源数量和规则复杂度…...
redis之数据库
文章目录 服务器中的数据库切换数据库数据库键空间读写键空间时的维护操作 设置键的生存时间或过期时间保存过期时间过期键的判定过期键删除策略清性删除策略的实现定期删除策略的实现 总结 服务器中的数据库 Redis服务器将所有数据库都保存在服务器状态redis.h/redisServer结…...
Vue3(1)
一.create-vue // new Vue() 创建一个应用实例 > createApp() // createRouter() createStore() // 将创建实例进行了封装,保证每个实例的独立封闭性import { createApp } from vue import App from ./App.vue// mount 设置挂载点 #app (id为app的盒子) createA…...
01.Docker 概述
Docker 概述 1. Docker 的主要目标2. 使用Docker 容器化封装应用程序的意义3. 容器和虚拟机技术比较4. 容器和虚拟机表现比较5. Docker 的组成6. Namespace7. Control groups8. 容器管理工具9. docker 的优缺点10. 容器的相关技术 docker 官网: http://www.docker.com 帮助文档…...
从零搭建:Canal实时数据管道打通MySQL与Elasticsearch
Canal实时同步Mysql Binlog至 Elasticsearch 文章目录 Canal实时同步Mysql **Binlog**至**Elasticsearch** 一. 环境准备1.环境检查检查Mysql是否开启BinLog开启Mysql BinlogJava环境检查 2.新建测试库和表3.新建Es索引 二.**部署 Canal Server****2.1 解压安装包****2.2 配置 …...
PyArmor:一个超级厉害的 Python 库!
在 Python 的世界里,如何保护我们的代码不被轻易盗用或者破解,一直是开发者们关注的问题。尤其是在发布软件时,如何有效防止源代码泄漏或者被逆向工程分析,成为了一个重要课题。 PyArmor 作为一款强大的 Python 加密工具ÿ…...
《战神:诸神黄昏》游戏闪退后提示弹窗“d3dx9_43.dll缺失”“找不到d3dx11_43.d”该怎么处理?
宝子们,是不是在玩《战神:诸神黄昏》的时候,突然弹出一个提示:“找不到d3dx9_43.dll”或者“d3dx11_43.dll缺失”?这可真是让人着急上火!别慌,今天就给大家唠唠这个文件为啥会丢,还有…...
Ollama本地部署DeepSeek(Mac)
准备工作 DeepSeek对比 DeepSeek-r1 DeepSeek-R1的多个版本:加上2个原装671B的,总计8个参数版本 DeepSeek-R1 671B DeepSeek-R1-Zero 671B DeepSeek-R1-Distill-Llama-70B DeepSeek-R1-Distill-Qwen-32B DeepSeek-R1-Distill-Qwen-14B DeepSeek-R1-Di…...
mysql8 从C++源码角度看sql生成抽象语法树
在 MySQL 8 的 C 源码中,SQL 语句的解析过程涉及多个步骤,包括词法分析、语法分析和抽象语法树(AST)的生成。以下是详细的解析过程和相关组件的描述: 1. 词法分析器(Lexer) MySQL 使用一个称为…...
