js深拷贝和浅拷贝面试题 ,js 字符串复制

本文目录

js 字符串复制


这里其实就是深拷贝和浅拷贝的问题,你上边代码只是浅拷贝,当2种情况
1:涉及对象的赋值 obj = obj1
2: 涉及数组赋值 ary = ary1
那么他们是传地址的,如果改动其中一个,如改动obj.name ,那么obj1.name也是会发生变化的,自己验证一下。
你上边的情况不涉及对象的赋值,而仅仅是字符串的简单复制,用浅拷贝就行了,当你上边变动一下,
var a={name:'yy',age:26};
b=a;
var b=new Object();
那么就要考虑深拷贝了 a = b.clone() // Object.prototype.clone()重写
不然改动a.name ,b.name也发生变化
原理我解释一下: 涉及对象和数组, b = a ; 那么b 放弃了new Object的地址,替换成指向a创建的{name:'yy',age:26}存放的地址 ,而 b.name=a.name b和a的地址不一样,所以a.name改变不影响b.name
正确与否,自己写个例子验证

js深拷贝和浅拷贝面试题
,js 字符串复制图1

深拷贝和浅拷贝有什么区别?如何实现深拷贝?


1.深拷贝与浅拷贝 拷贝即是通常所说的复制(Copy)或克隆(Clone),对象的拷贝也就是从现有对象复制一个“一模一样”的新对象出来。虽然都是复制对象,但是不同的复制方法,复制出来的新对象却并非完全一模一样,对象内部存在着一些差异。通常的拷贝方法有两种,即深拷贝和浅拷贝,那二者之间有何区别呢?MSDN里对IClone接口的Clone方法有这样的说明:在深层副本中,所有的对象都是重复的;而在浅表副本中,只有顶级对象是重复的,并且顶级以下的对象包含引用。可以看出,深拷贝和浅拷贝之间的区别在于是否复制了子对象。这如何理解呢?下面我通过带有子对象的代码来验证二者的区别。 首先定义两个类型:Student和ClassRoom,其中Student类型里包含ClassRoom,并使这两个类型都分别实现自定义的深拷贝接口(IDeepCopy)和浅拷贝接口(IShallowCopy)。 类图如下: 定义代码如下: 定义代码 /// <summary> /// 深拷贝接口 /// </summary> interface IDeepCopy { object DeepCopy(); } /// <summary> /// 浅拷贝接口 /// </summary> interface IShallowCopy { object ShallowCopy(); } /// <summary> /// 教室信息 /// </summary> class ClassRoom : IDeepCopy, IShallowCopy { public int RoomID = 1; public string RoomName = "Room1"; public override string ToString() { return "RoomID=" + RoomID + "\tRoomName=" + RoomName; } public object DeepCopy() { ClassRoom r = new ClassRoom(); r.RoomID = this.RoomID; r.RoomName = this.RoomName; return r; } public object ShallowCopy() { //直接使用内置的浅拷贝方法返回 return this.MemberwiseClone(); } } class Student : IDeepCopy, IShallowCopy { //为了简化,使用public 字段 public string Name; public int Age; //自定义类型,假设每个Student只拥有一个ClassRoom public ClassRoom Room = new ClassRoom(); public Student() { } public Student(string name, int age) { this.Name = name; this.Age = age; } public object DeepCopy() { Student s = new Student(); s.Name = this.Name; s.Age = this.Age; s.Room = (ClassRoom)this.Room.DeepCopy(); return s; } public object ShallowCopy() { return this.MemberwiseClone(); } public override string ToString() { return "Name:" + Name + "\tAge:" + Age + "\t" + Room.ToString(); } } 测试代码: 测试代码 Student s1 = new Student("Vivi", 28); Console.WriteLine("s1=[" + s1 + "]"); Student s2 = (Student)s1.ShallowCopy(); //Student s2 = (Student)s1.DeepCopy(); Console.WriteLine("s2=[" + s2 + "]"); //此处s2和s1内容相同 Console.WriteLine("-----------------------------"); //修改s2的内容 s2.Name = "tianyue"; s2.Age = 25; s2.Room.RoomID = 2; s2.Room.RoomName = "Room2"; Console.WriteLine("s1=[" + s1 + "]"); Console.WriteLine("s2=[" + s2 + "]"); //再次打印两个对象以比较 Console.ReadLine(); 运行结果: a.ShallowCopy s1=[Name:Vivi Age:28 RoomID=1 RoomName=Room1] s2=[Name:Vivi Age:28 RoomID=1 RoomName=Room1] ------------------------------------------------------------- s1=[Name:Vivi Age:28 RoomID=2 RoomName=Room2] s2=[Name:tianyue Age:25 RoomID=2 RoomName=Room2] b.DeepCopy s1=[Name:Vivi Age:28 RoomID=1 RoomName=Room1] s2=[Name:Vivi Age:28 RoomID=1 RoomName=Room1] ----------------------------- s1=[Name:Vivi Age:28 RoomID=1 RoomName=Room1] s2=[Name:tianyue Age:25 RoomID=2 RoomName=Room2] 从以上结果可以看出,深拷贝时两个对象是完全“分离”的,改变其中一个,不会影响到另一个对象;浅拷贝时两个对象并未完全“分离”,改变顶级对象的内容,不会对另一个对象产生影响,但改变子对象的内容,则两个对象同时被改变。这种差异的产生,即是取决于拷贝子对象时复制内存还是复制指针。深拷贝为子对象重新分配了一段内存空间,并复制其中的内容;浅拷贝仅仅将指针指向原来的子对象。 示意图如下: 2.浅拷贝与赋值操作 大多数面向对象语言中的赋值操作都是传递引用,即改变对象的指针地址,而并没有复制内存,也没有做任何复制操作。由此可知,浅拷贝与赋值操作的区别是顶级对象的复制与否。当然,也有一些例外情况,比如类型定义中重载赋值操作符(assignment operator),或者某些类型约定按值传递,就像C#中的结构体和枚举类型。 赋值操作示意图如下: 3.C++拷贝构造函数 与其它面向对象语言不同,C++允许用户选择自定义对象的传递方式:值传递和引用传递。在值传递时就要使用对象拷贝,比如说按值传递参数,编译器需要拷贝一个对象以避免原对象在函数体内被破坏。为此,C++提供了拷贝构造函数用来实现这种拷贝行为,拷贝构造函数是一种特殊的构造函数,用来完成一些基于同一类的其它对象的构造和初始化。它唯一的参数是引用类型的,而且不可改变,通常的定义为X(const X&)。在拷贝构造函数里,用户可以定义对象的拷贝行为是深拷贝还是浅拷贝,如果用户没有实现自己的拷贝构造函数,那么编译器会提供一个默认实现,该实现使用的是按位拷贝(bitwise copy),也即本文所说的浅拷贝。构造函数何时被调用呢?通常以下三种情况需要拷贝对象,此时拷贝构造函数将会被调用。 1.一个对象以值传递的方式传入函数体 2.一个对象以值传递的方式从函数返回 3.一个对象需要通过另外一个对象进行初始化 4.C# MemberwiseClone与ICloneable接口 和C++里的拷贝构造函数一样,C#也为每个对象提供了浅拷贝的默认实现,不过C#里没有拷贝构造函数,而是通过顶级类型Object里的MemberwiseClone方法。MemberwiseClone 方法创建一个浅表副本,方法是创建一个新对象,然后将当前对象的非静态字段复制到该新对象。有没有默认的深拷贝实现呢?当然是没有,因为需要所有参与拷贝的对象定义自己的深拷贝行为。C++里需要用户实现拷贝构造函数,重写默认的浅拷贝;C#则不同,C#(确切的说是.NET Framework,而非C#语言)提供了ICloneable 接口,包含一个成员 Clone,它用于支持除 MemberwiseClone 所提供的克隆之外的克隆。C++通过拷贝构造函数无法确定子对象实现的是深拷贝还是浅拷贝,而C#在“强制”实现浅拷贝的基础上,提供ICloneable 接口由用户定义深拷贝行为,通过接口来强制约束所有参与拷贝的对象,个人觉得,这也算是一小点C#对C++的改进。 5.深拷贝策略与实现 深拷贝的要点就是确保所有参与拷贝的对象都要提供自己的深拷贝实现,不管是C++拷贝构造函数还是C#的ICloneable 接口,事实上都是一种拷贝的约定。有了事先的约定,才能约束实现上的统一,所以关键在于设计。 但偶尔也会在后期才想到要深拷贝,怎么办?总不能修改所有之前的实现吧。有没有办法能够通过顶级类而不关心内部的子对象直接进行深拷贝呢?能不能搞个万能的深拷贝方法,在想用的时候立即用,而不考虑前期的设计。这样“大包大揽”的方法,难点在于实现时必须自动获取子对象的信息,分别为子对象实现深拷贝。C++里比较困难,.NET的反射机制使得实现容易一些。不过这样的方法虽然通用,实则破坏了封装,也不符合“每个类对自己负责”的设计原则。 基于.NET的反射机制,以前写了一个通用的序列化方法,现在可以拿过来,先序列化,然后再反序列化回来,也即是一个深拷贝,示例代码如下: 深拷贝示例代码 #region ICloneable Members /// <summary> /// 此处的复制为深拷贝,在实现上,为了简化,采用序列化和反序列化。 /// </summary> /// <returns>深拷贝对象</returns> public object Clone() { Student stu = new Student(); XmlStorageHelper helper = new XmlStorageHelper(); string strXml = helper.ConvertToString(this); helper.LoadFromString(stu, strXml); //从XML字符串来赋值 return stu; } #endregion

js深拷贝和浅拷贝面试题
,js 字符串复制图2

深拷贝和浅拷贝有什么区别?如何实现深拷贝?


1、浅拷贝:默认的拷贝就是浅拷贝。 仅仅多了个指针指向原来的空间。
2、深拷贝:自己写的拷贝,自己申请了动态内存空间,用了new 或 malloc 。不但多了指针,而且多了空间。
3、用深拷贝的话,最好用自己写的析构,记得在里面释放内存,也可以用默认析构。
4.用浅拷贝(即默认隐藏的拷贝),最好用默认析构,若用自己写的析构里面 ,记得不要释放内存,不然会造成重复释放内存而报错。

js深拷贝和浅拷贝面试题
,js 字符串复制图3

浅拷贝和深拷贝的区别js常见笔试题


浅拷贝只是拷贝了引用, 数据在内存中还是一个, 如果引用的对象发生了变化, 该变量也会同步变化.


深拷贝相当于把内存上的数据拷贝了一份, 此时修改该变量不会影响原来的变量, 同理修改原来的变量也不会影响现在的变量.


浅拷贝:


const foo = { bar: 'bar', baz: 'baz' }
const ref_foo = foo

深拷贝(最简单的方法, 拷贝性能也不弱):


const foo = { bar: 'bar', baz: 'baz' }
const ano_foo = JSON.parse(JSON.stringify(foo))

js深拷贝和浅拷贝面试题
,js 字符串复制图4

以上就是关于js深拷贝和浅拷贝面试题 ,js 字符串复制的全部内容,以及js深拷贝和浅拷贝面试题 的相关内容,希望能够帮到您。

上一篇 2022年05月20日15时54分31秒
下一篇 2022年05月20日16时04分18秒

相关推荐

  • 高三英语语法知识点

    本文目录 1. 时态 2. 语态 3. 从句 4. 词汇 高三英语是中学英语学习阶段的最后一年,语法知识点是英语学习的重中之重。本文将介绍高三英语语法知识点的重点内容。 1. 时态 时态是英语中最基础的语法知识点之一,高考中也是最常考的内容。高三英语时态主要包括以下几种: 简单现在时 现在进行时 简单过去时 过去进行时 现在完成时 过去完成时 掌握时态的用法和变化规则对于高考阅读理解和写作都非常重…

    英语知识 2023年06月25日
    11
  • 表原因链接两个句子的词

    本文目录 1.因为 (Because) 2.由于 (Due to) 3.所以 (So) 4.因此 (Therefore) 5.由此可见 (Thus) 在英语写作中,经常需要使用一些词语来表达两个句子之间的因果关系。这些词语被称为表原因链接两个句子的词。下面将介绍几个常用的表原因链接两个句子的词。 因为 (Because) “因为”是最常用的表原因链接两个句子的词之一。它表示第一个句子是第二个句子的…

    英语知识 2023年06月25日
    11
  • 八年级下册重点短语重点句型

    英语作为一门国际通用语言,对于学生来说是一项非常重要的技能。在八年级下册的英语学习中,我们学习了很多重点短语和句型。在这篇文章中,我们将重点介绍其中的几个。 重点短语 Be used to 这个短语用于描述一个人对某种习惯或环境已经习惯了,通常会和-ing形式的动词一起使用。例如:“I am used to getting up early in the morning.” Look forwar…

    英语知识 2023年06月25日
    11
  • 双重转折句

    双重转折句是英语语法中的一种重要句式,它由两个转折词组成,常用的转折词包括"although"、"while"、"even though"等。双重转折句常用来表达对比、让步或者矛盾的关系。 例如,在句子"Although it was raining, we still went for a walk"中,"although"和"still"构成了一个典型的双重转折句。这个句子表达了尽管天在下雨,…

    英语知识 2023年06月25日
    11
  • 表示目的的英文连接词

    英语中,表示目的的连接词有很多种,常用的包括:in order to, so as to, for the purpose of, with the aim of, with the intention of, to name just a few. 首先是最常用的“in order to”。它的意思是“为了”,用来表达做某事的目的。例如,“I am studying hard in order …

    英语知识 2023年06月25日
    11
  • 表示结果的英语连接词

    本文目录 1.So 2.Therefore 3.As a result 4.Consequently 英语中有很多连接词可以用来表示结果。这些词汇可以帮助我们在表达时更加准确地传达信息。下面是几个常用的表示结果的英语连接词: So “So”是一个非常常用的表示结果的连接词。它可以用来连接两个简单句,表示因果关系。例如:“I studied hard, so I got an A on the ex…

    英语知识 2023年06月25日
    11
  • 必修一英语重点句子归纳

    本文目录 1. 一般现在时 2. 现在进行时 3. 一般过去时 4. 现在完成时 5.结论 英语作为世界上使用最广泛的语言之一,是我们必须掌握的一项基本技能。在学习英语时,掌握一些重点句子是非常重要的。本文将对必修一英语教材中的重点句子进行归纳总结,帮助读者更好地掌握英语知识。 1. 一般现在时 一般现在时是英语语法中最基础的时态之一。以下是一些常用的一般现在时句子: I like to play…

    英语知识 2023年06月25日
    11
  • 条件状语从句例句10句

    条件状语从句是英语语法中的一个重要概念,它用来表达一件事情发生的条件。下面是10个常见的条件状语从句例句: 如果我明天有时间,我会去看电影。 如果你不想迟到,你应该早点出门。 除非他改变主意,否则他不会参加聚会。 假如你学习更加刻苦,你的成绩将会有所提高。 如果我们不采取措施,气候变化将会带来很多问题。 如果你需要帮助,请随时和我联系。 如果明天下雨,我们就不去郊游了。 你只有努力学习,才能取得好…

    英语知识 2023年06月25日
    11