Skip to content

Latest commit

 

History

History
101 lines (82 loc) · 5.06 KB

entity-clone.md

File metadata and controls

101 lines (82 loc) · 5.06 KB

克隆

简介

节点克隆是运行时的常用功能,同时节点克隆也会附带克隆其绑定的组件。例如在初始化阶段根据配置动态创建一定数量相同的实体,然后根据逻辑规则摆放到场景不同的位置。这里会对脚本的克隆细节进行详细讲解。

使用

实体的克隆

非常简单,直接调用实体的 clone() 方法即可完成实体以及附属组件的克隆。

const cloneEntity = entity.clone();

脚本的克隆

脚本的本质也是组件,所以当我们调用实体的 clone() 函数时,引擎不仅会对引擎内置组件进行克隆,还是对自定义脚本进行克隆。引擎内置组件的克隆规则官方已经完成定制,同样我们也将脚本的克隆能力和规则定制开放给了开发者。脚本字段默认的克隆方式为浅拷贝,例如我们对脚本的字段值进行修改后再克隆,克隆后的脚本将保持修改后的值,无需增加任何额外的编码。以下为自定义脚本的克隆案例:

// define a custom script
class CustomScript extends Script{
  /** boolean type.*/
  a:boolean = false;
  
  /** number type.*/
  b:number = 1;
  
  /** class type.*/
  c:Vector3 = new Vector3(0,0,0);
}

//init entity and script.
const entity = engine.createEntity();
const script = entity.addComponent(CustomScript);
script.a = true;
script.b = 2;
script.c.setValue(1,1,1);

// clone logic.
const cloneEntity = entity.clone();
const cloneScript = cloneEntity.getComponent(CustomScript);
console.log(cloneScript.a);// output is true.
console.log(cloneScript.b);// output is 2.
console.log(cloneScript.c);// output is (1,1,1).

克隆装饰器

除了默认的克隆方式外,引擎还提供了“克隆装饰器“对脚本字段的克隆方式进行定制。引擎内置四种克隆装饰:

装饰器名称 装饰器释义
ignoreClone 克隆时对字段进行忽略。
assignmentClone ( 默认值,和不添加任何克隆装饰器等效) 克隆时对字段进行赋值。如果是基本类型则会拷贝值,如果是引用类型则会拷贝其引用地址。
shallowClone 克隆时对字段进行浅克隆。克隆后会保持自身引用独立,并使用赋值的方式克隆其内部所有字段(如果内部字段是基本类型则会拷贝值,如果内部字段是引用类型则会拷贝其引用地址)。
deepClone 克隆时对字段进行深克隆。克隆后会保持自身引用独立,并且其内部所有深层字段均保持完全独立。

我们将上面的案例稍加修改,分别对 CustomScript 中的四个字段添加了不同的“克隆装饰器“,由于 shallowClone 和 deepCone  较复杂,我们对字段 c 和 d 增加了额外的打印输出进行进一步讲解。

// define a custom script
class CustomScript extends Script{
  /** boolean type.*/
  @ignoreClone
  a:boolean = false;
  
  /** number type.*/
  @assignmentClone
  b:number = 1;
  
  /** class type.*/
  @shallowClone
  c:Vector3[] = [new Vector3(0,0,0)];
  
  /** class type.*/
  @deepClone
  d:Vector3[] = [new Vector3(0,0,0)];
}

//init entity and script.
const entity = engine.createEntity();
const script = entity.addComponent(CustomScript);
script.a = true;
script.b = 2;
script.c[0].setValue(1,1,1);
script.d[0].setValue(1,1,1);

// clone logic.
const cloneEntity = entity.clone();
const cloneScript = cloneEntity.getComponent(CustomScript);
console.log(cloneScript.a);// output is false,ignoreClone will ignore the value.
console.log(cloneScript.b);// output is 2,assignmentClone is just assignment the origin value.
console.log(cloneScript.c[0]);// output is Vector3(1,1,1),shallowClone clone the array shell,but use the same element.
console.log(cloneScript.d[0]);// output is Vector3(1,1,1),deepClone clone the array shell and also clone the element.

cloneScript.c[0].setValue(2,2,2);// change the field c[0] value to (2,2,2).
cloneScript.d[0].setValue(2,2,2);// change the field d[0] value to (2,2,2).

console.log(script.c[0]);// output is (2,2,2). bacause shallowClone let c[0] use the same reference with cloneScript's c[0].
console.log(script.d[0]);// output is (1,1,1). bacause deepClone let d[0] use the different reference with cloneScript's d[0].

注意:

  • shallowClone 和 deepClone 适用于 ObectArrayClass 类型。
  • shallowClone 克隆后会保持自身引用独立,并使用赋值的方式克隆其内部所有字段(如果内部字段是基本类型则会拷贝值,如果内部字段是引用类型则会拷贝其引用地址)。
  • deepClone 如果在深克隆过程中遇到 Class 则会调用对象的 cloneTo() 实现克隆,需要对象实现 IClone 接口。