服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - 编程技术 - 「字符串」存在「栈内存」?那我可要杠你了哦!

「字符串」存在「栈内存」?那我可要杠你了哦!

2024-03-27 15:29前端之神 编程技术

当我们新建一个字符串时,V8会从内存中查找一下是否已经有存在的一样的字符串,找到的话直接复用。如果找不到的话,则开辟一块新的内存空间来存这个字符串,并把地址赋给变量。

前言

大家好,我是林三心,用最通俗易懂的话讲最难的知识点是我的座右铭,基础是进阶的前提是我的初心。

在我们的认知里:基础类型存栈内存,引用数据类型存堆内存。

const a = '林三心'
const b = {
    age: 18,
    height: 180
}

「字符串」存在「栈内存」?那我可要杠你了哦!图片

超长字符串

大家都知道,字符串属于基础类型,所以大家会觉得字符串是存在栈内存中的,但是大家要知道,V8默认栈内存是984Kib,那如果一个超长字符串 > 984Kib能装的进栈内存吗?这也就是一个比较经典的问题——大象装箱问题,试问:一头大象能装进一个小箱子里吗?

「字符串」存在「栈内存」?那我可要杠你了哦!图片

一探究竟

堆快照

先来看一段代码

const func = function() {
  this.str1 = '林三心'
  this.str2 = 'Sunshine_Lin'
}

const a = new func()
const b = new func()

然后咱们来看看堆快照的详情

「字符串」存在「栈内存」?那我可要杠你了哦!图片

上面的结果可以看出:

  • a 和 b的str1都指向同一个地址
  • a 和 b的str2都指向同一个地址

那我们可不可以猜测出一个结论:字符串的内容存于堆内存中,指针存于栈内存中,且相同的字符串指向同一个堆内存地址

修改和新增字符串

我们稍微修改下代码

const func = function() {
  this.str1 = '林三心'
  this.str2 = 'Sunshine_Lin'
}

const a = new func()
const b = new func()

// 修改str1
a.str1 = '哈哈哈哈哈哈哈哈哈哈'
// 新增str3,跟str2一样
a.str3 = 'Sunshine_Lin'

再来看看现阶段的堆快照的详情

「字符串」存在「栈内存」?那我可要杠你了哦!图片

上面的结果可以看出:

  • str1修改成一个新的字符串后,重新开辟了一个内存空间(新地址)
  • str3新增之后,指针指向已有的Sunshine_Lin的内存空间

那我们可不可以猜测出一个结论:新增或者修改字符串后,如果是一个之前不存在的字符串,则新开辟内存空间,如果是已有的,则直接使用已有的内存空间

源码分析

当我们声明一个字符串时:

  • 1、v8内部有一个名为stringTable的hashmap缓存了所有字符串,在V8阅读我们的代码,转换抽象语法树时,每遇到一个字符串,会根据其特征换算为一个hash值,插入到hashmap中。在之后如果遇到了hash值一致的字符串,会优先从里面取出来进行比对,一致的话就不会生成新字符串类。
  • 2、缓存字符串时,根据字符串不同采取不同hash方式。

「字符串」存在「栈内存」?那我可要杠你了哦!图片

源码

「字符串」存在「栈内存」?那我可要杠你了哦!图片

image.png

「字符串」存在「栈内存」?那我可要杠你了哦!图片

通俗易懂总结

字符串的数据存于 堆内存 中,栈内存 中只是存其 地址指针

当我们新建一个字符串时,V8会从内存中查找一下是否已经有存在的一样的字符串,找到的话直接复用。如果找不到的话,则开辟一块新的内存空间来存这个字符串,并把地址赋给变量。

大家有没有想过,为什么字符串不能通过下标索引来进行修改呢?因为字符串的修改本质上只能是通过整个的修改,而不能局部修改。

原文地址:https://mp.weixin.qq.com/s/fBQ41_HgGHSAufoT1IDy2Q

延伸 · 阅读

精彩推荐