主頁 > 前端設計 > js深拷貝deepCopy教程:支持回圈參考、型別不丟失、可擴展、可定制

js深拷貝deepCopy教程:支持回圈參考、型別不丟失、可擴展、可定制

2021-02-25 16:24:57 前端設計

目錄

  • 1. 背景
  • 2. 簡介
  • 3. 安裝方式
    • 3.1. 方式1:通過 npm 安裝
    • 3.2. 方式2:直接下載原代碼
    • 3.3. 方式3:通過<script>標簽引入
  • 4. 教程
    • 4.1. API簡介
    • 4.2. 基本使用
    • 4.3. 拷貝函式
    • 4.4. 指定拷貝深度
    • 4.5. 回圈參考
    • 4.6. 保持型別資訊
    • 4.7. 拷貝不可列舉的屬性
    • 4.8. 自定義拷貝規則
      • 4.8.1. typeCopyers
        • 4.8.1.1. Types
        • 4.8.1.2. 拷貝者Copyer
      • 4.8.2. 預設presetTypeCopierMap
      • 4.8.3. 拷貝者的優先級
      • 4.8.4. getCopy
    • 4.9. 創建帶預設拷貝規則的拷貝函式

內容

1. 背景

開發中,經常需要對一個物件進行深拷貝操作,目前經常用到的深拷貝的方式有以下幾種:

  • 將物件序列化成 JSON 字串后,再反序化成物件 let copy = JSON.parse(JSON.stringify(value))
  • 第三方庫提供的深拷貝工具,如 Lodash 的 _.cloneDeep(value)_.cloneDeepWith(value, customizer)

但這些方法有以下缺點:

  • 不支持物件成員回圈參考,比如下面這種:
    回圈參考
  • 拷貝后會丟失型別資訊,變成了普通的物件
  • 拷貝后會丟失成員參考關系資訊
  • 不能根據型別自定義拷貝規則
  • 只能拷貝可列舉的屬性
  • 不能拷貝函式
  • 不能指定拷貝深度

為了解決這些問題,deep-copy 就出現了👏

2. 簡介

deep-copy 是一個深拷貝工具,可對任意資料進行深度拷貝,包括 函式 function、正則 RegExp、Map、Set、Date、Array、URL 等等;支持含回圈參考關系的物件的拷貝,并且不會丟失成員的參考關系資訊 和 型別資訊,支持擴展,可根據資料型別定制拷貝邏輯,也可指定拷貝深度;所以,通過它可實作對任意型別的資料進行任意想要的拷貝;

具有以下特性:

  • 支持物件成員回圈參考
  • 拷貝后不會丟失型別資訊 和 成員參考關系資訊
  • 可指定拷貝深度
  • 即能拷貝可列舉的成員,也可拷貝不可列舉的成員
  • 可拷貝函式
  • 可根據型別自定義拷貝規則
  • 支持預設拷貝規則
  • 支持創建多個不同預設拷貝規則的拷貝函式

詳情請看:

  • 主頁:https://github.com/GuoBinyong/deep-copy
  • GitHub倉庫
  • 碼云倉庫

如果您在使用的程序中遇到了問題,或者有好的建議和想法,您都可以通過以下方式聯系我,期待與您的交流:

  • 給該倉庫提交 issues
  • 給我 Pull requests
  • 郵箱:guobinyong@qq.com
  • QQ:guobinyong@qq.com
  • 微信:keyanzhe

3. 安裝方式

目前,安裝方式有以下幾種:

3.1. 方式1:通過 npm 安裝

npm install @gby/deep-copy

3.2. 方式2:直接下載原代碼

您可直接從專案的 發行地址 下載 原始碼 或 構建后包檔案;

您可以直接把 原始碼 或 構建后 的包拷貝到您的專案中去;然后使用如下代碼在您的專案中引入 deepCopy

import { deepCopy } from "path/to/package/deep-copy";

或者

import deepCopy from "path/to/package/deep-copy";

3.3. 方式3:通過<script>標簽引入

您可直接從專案的 發行地址 中下載以 .iife.js 作為綴的檔案,然后使用如下代碼參考 和 使用 deep-copy:

  1. 參考 deep-copy

    <script src="path/to/package/deep-copy.iife.js"></script>
    
  2. 使用全域的 deepCopy

    <script>
    // 使用全域的 deepCopy
        const copy = deepCopy.deepCopy(value);
    </script>
    

4. 教程

4.1. API簡介

deep-copy 匯出了兩個工具函式

import {deepCopy,createDeepCopy} from "deep-copy"
  • deepCopy<V>(value:V,options?:DeepCopyOptions|null|undefined,typeCopyers?:TypeRevivers<Copier>|null|undefined):V:對 value 進行深拷貝操作,并將深拷貝后的值回傳;引數如下:

    • value: any:被拷貝的值;
    • options?:DeepCopyOptions|null|undefined:描述拷貝配置的選項物件;
    • typeCopyers?:TypeRevivers<Copier>|null|undefined:可根據不同型別定制拷貝邏輯的配置物件
  • createDeepCopy(presetTypeCopierMap?:TypeReviverMap<Copier>):DeepCopy:創建并回傳帶有默認 typeCopyers 的 deepCopy() 函式;引數如下:

    • presetTypeCopierMap?:TypeReviverMap<Copier>:默認的 typeCopyers;

4.2. 基本使用

deep-copy 庫中 會匯出 deepCopy() 函式,用它可以直接對值進行深拷貝,如下所示:

import {deepCopy} from "../dist/deep-copy.es"

let value = {
    name:"root",
    sub:{
        name:"member1",
        fun:function () {
            console.log("這是函式")
        }
    }
};

let copy = deepCopy(value);

其中 copy 和 value 的資訊結構是完全一樣的,但 copy 和 value 及其成員(除了函式(方法)外)已經是完全兩份實體,占據著兩份不同的記憶體空間;

4.3. 拷貝函式

默認情況下,deepCopy() 不會對 函式物件 進行深拷貝,比如上例中的 value.sub.fun,所以,copy 中的函式(方法) 和 value 中的函式 是同一函式,占據同一份記憶體空間,即 copy.sub.fun === value.sub.fun

如果希望 deepCopy() 對函式也進行深拷貝,可以給 deepCopy() 傳遞第2個引數并設定選項 copyFuntrue,如下所示:

let copy = deepCopy(value,{
    copyFun:true
});

些時,copy 中的函式(方法) 和 value 中的函式 具有相同的代碼邏輯,但已經是兩個不同的函式實體,占據不同的記憶體空間,即 copy.sub.fun !== value.sub.fun

4.4. 指定拷貝深度

如果你只需要實體進行一定深度的深拷貝,比如:只對實體、實體的成員、實體的成員的成員 進行拷貝操作,再深入的不進行拷貝操作,像這樣的需求,可以通過給 deepCopy 傳遞 maxDepth 選項來實作,如下:

let copy = deepCopy(value,{
    maxDepth:2
});

maxDepth 是可選的,默認值為:Infinity;表示拷貝的最大深度;當值為 undefinednull 時,會使用默認值,表示無限深度;被拷貝的值(本例中是 value)本身的深度為 0 ,被拷貝值的成員的深度為 1 ,依次類推;

4.5. 回圈參考

當物件 與 物件 之間 或 物件 與 成員 之前 存在類似以下參考關系時,就會形成回圈參考
回圈參考

JSON的深拷貝方案是無法處理回圈參考關系 let copy = JSON.parse(JSON.stringify(value)),會造成記憶體溢位;

deepCopy() 支持拷貝這種帶回圈參考關系的物件,并且拷貝后的值,仍然會保持這種回圈參考關系,如下所示:

let root = {
    name:"root"
};

let member1 = {
    name:"member1",
    fun:function () {
        console.log("這是函式")
    }
};


let member2 = {
    name:"member2",
};

root.sub = member1;
member1.sub = member2;
member2.sub = root;

let rootCopy = DC.deepCopy(root);

拷貝前 和 拷貝后的 參考關系保持一樣:

root.sub.sub.sub === root;  // true
rootCopy.sub.sub.sub === rootCopy;  // true

4.6. 保持型別資訊

有些時候,被拷貝的物件并不是普通的物件,它可能是某個類的實體,比如:

class Person {
    constructor(){
        this.name = "";
        this.email = "";
    }
}

let p = new Person();
p.name = "郭斌勇";
p.email = "guobinyong@qq.com";

p 是 類 Person 的實體,但通過 JSON方案拷貝后

let pCopy = JSON.parse(JSON.stringify(p));

pCopy instanceof Person; // false
pCopy instanceof Object; // true

pCopy 卻變成了普通物件,即 Object 型別的實體;

然而,deepCopy() 會保持這種型別資訊,如下:

let pCopy = deepCopy(p);
pCopy instanceof Person;  // true

拷貝后的 pCopy 仍然是 Person 型別的實體;

注意:
deep-copy 在拷貝實體時,會用實體所屬的類創建一個新的實體,創建時并不會給建構式傳遞任何引數, 然后將原來實體本身的成員的副本重設 新實體為新實體的成員;這對于那些需要給建構式傳遞引數的類 可能會存在問題;如果某類創建實體時依賴建構式的引數,則您可以針對些類定制拷貝規則;

4.7. 拷貝不可列舉的屬性

deepCopy() 默認只拷貝物件自身的可列舉屬性,如果您也想拷貝物件自身的不可列舉的屬性,則可以給 deepCopy() 傳遞 allOwnProps 選項,并設定為 true,如下:

let copy = deepCopy(value,{
    allOwnProps:true
});
  • allOwnProps?:boolean | undefined | null:可選選項;默認值: undefined; 表示是否要拷貝所有自身的屬性,包不可列舉的,但不包括原型鏈上的屬性;
    • true:拷貝物件自身(不包括原型上的)的所有屬性(包括不可列舉的);
    • false|undefined|null : 只拷貝物件自身中(不包括原型上的)可列舉的屬性;

4.8. 自定義拷貝規則

對于下面這個 Person 類,在創建實體時(比如:p)需要傳入一個構造引數 sex,這個引數決定了模型的構建程序,如下:

class Person {
    constructor(sex){
        this.sex = sex;
        // 根據 sex  初始化模型
        switch(sex){
            case "男":{
                console.log("初始化男性特性");
                break;
            }
            case "女":{
                console.log("初始化女性特性");
                break;
            }
            default:{
                console.log("初始化人妖特性");
            }
        }

        this.name = "";
        this.email = "";
        this.idCard = null; //身份證,是一個物件
        this.mate = null;  //配偶,也是 Person 類的實體
        
    }
}


let p = new Person("男");
p.name = "郭斌勇";
p.email = "guobinyong@qq.com";
p.idCard = {
    id:4231232776886677,  //身份
    address:"中國的一個小農村里"
};

//設定配偶
p.mate = new Person("女");
p.mate.name = "簡愛";

像這種型別的類,如果使用 deepCopy() 的默認拷貝邏輯的話,新拷貝的 Person 實體可能會執行錯誤的構建程序,因為 deepCopy() 在創建新實體時,不會傳入任何構建引數;像這種型別,我們就需要提供自定義的拷貝規則了;

如果你想自定義某個型別實體的深拷貝規則,有以下幾種方案:

  • deepCopy() 函式提供 typeCopyers 引數 或 設定 presetTypeCopierMap 預設;
  • 給實體增加 getCopy 方法;

4.8.1. typeCopyers

可以通過 typeCopyers 引數自定義拷貝規則,如下:

let pCopy = deepCopy(p,null,{
    // 當需要拷貝 Person 型別的實體時,會回呼這個函式,這個函式需要回傳 Person 實體的副本;這個函式稱為 拷貝者 Copier
    "Person": function(value,copyMember,options){
        let copy = new Person(value.sex);  // 創建新的 Person 實體,作為 value 的副本

        // 給副本 copy 設定基本型別的成員;
        copy.name = value.name;
        copy.email = value.email;

        // 如果參考型別的成員也需要深拷貝,則需要通過 copyMember() 函式拷貝 參考型別的成員
        copy.idCard = copyMember(value.idCard);
        /**
         * copyMember() 函式會回傳 拷貝后的值;
         * 但如果被拷貝的值存在回圈參考的話,copyMember() 會回傳 undefined;這種情況下,可通過給 copyMember() 傳遞回呼函式(第2個引數),通過回呼函式來獲取拷貝后的值;
         * 無論是否存在回圈參考,通過回呼函式總能拿到拷貝后的值,回呼函式是一種保險的方法
         */
        copyMember(value.mate,function(mateCopy){
            copy.mate = mateCopy;
        });

        // 需要將副本回傳;
        return copy;
    }
});

deepCopy() 函式的 typeCopyers 引數是 TypeRevivers<Copier> 型別的,是用來描述 型別 和 拷貝者 Copier(提供拷貝實體回呼函式)的對應關系的,上例中 typeCopyers 使用的是 物件形式,它共以下幾種描述方式:

  • {[TypeName:ExactTypeName]:Copyer}:物件形式,屬性名字 是 型別的名字,屬性值是型別的 拷貝者(Copyer);
  • [Types, Copyer][]:陣列形式,陣列中的元素是 [Types, Copyer] 型別的元組,元組的第一個元素是 Types,第二個元素是 拷貝者;
  • Map<Types, Copyer>:Map形式,Map中的鍵是 Types 型別,值是 拷貝者;

4.8.1.1. Types

TypesExactTypeNameExactType 和 其陣列結構的聯合,定義如下:

type Types = DataType | DataTypeArray;
type DataType = ExactTypeName | ExactType;
type DataTypeArray = DataType[];
  • ExactType:精確型別,可更加細致地描述型別;各種型別的值(左側) 與 ExactType(右側) 的映射如下:

    • undefinedundefined
    • null : null
    • string : "string"
    • number : "number"
    • bigint : "bigint"
    • boolean : "boolean"
    • symbol : "symbol"
    • 普通函式 : Function
    • 異步函式 : AsyncFunction
    • 生成器函式 : GeneratorFunction
    • 沒有原型的物件(如:通過 Object.create(null) 創建的物件) : "object"
    • 其它任何型別的實體 : 該實體的建構式
  • ExactTypeName:精確型別的字串表示;各種型別的值(左側) 與 ExactTypeName(右側) 的映射如下:

    • undefined"undefined"
    • null : "null"
    • string : "string"
    • number : "number"
    • bigint : "bigint"
    • boolean : "boolean"
    • symbol : "symbol"
    • 普通函式 : "Function"
    • 異步函式 : "AsyncFunction"
    • 生成器函式 : "GeneratorFunction"
    • 沒有原型的物件(如:通過 Object.create(null) 創建的物件) : "object"
    • 其它任何型別的實體 : 該實體的建構式的名字

所以,Types 可以是具體的型別 ExactType ,比如:Person 類;也可以是 型別的名字字串 ExactTypeName ,比如:'Person';或者是包含多個型別的陣列,比如:[Person,'Map',Set]

4.8.1.2. 拷貝者Copyer

拷貝者 Copyer 是一個型別為 (value:T,copyMember:CopyMember,options:CopierOptions)=>T 函式,嚴格的定義如下:

type Copier<T = any,Host = any> = (this:T,value:T,copyMember:CopyMember,options:CopierOptions<Host>)=>T

當需要拷貝某個型別的實體時,就會呼叫這個型別的 拷貝者 Copyer ,然后將拷貝者回傳的值作為那個實體的副本;

在呼叫拷貝者 Copyer 時,會將 Copyer 的 this 設定為被拷貝的值(與 value 引數相同的值),并會給 Copyer 傳遞以下引數:

  • value:被拷貝的值;
  • copyMember:用于深拷貝 value 的成員的深拷貝函式;型別為 (member:T,completeCB?:CompleteCB<T>|null|undefined,key?:K,host?:H | null | undefined,options?:CopyMemberOptions)=>T|undefined;拷貝后副本會通過 copyMember() 的回傳值(當被拷貝的成員不存在回圈參考時) 和 completeCB 回呼函式 回傳;
  • options:包含了其它資訊的物件
     {
        allOwnProps:boolean;
        key:any;
        host:Host;
        type:string;
        depth:number;
        copyFun:boolean;
    }
    

拷貝者 Copyer 需要將自定創建的 value 的副本回傳;

在創建 value 的副本的程序中,如果需要對 value 的參考型別的成員 進行拷貝,則需要呼叫 copyMember(value) 函式,這個函式是專門用于 深拷貝 value 的成員的;拷貝后副本會通過 copyMember(value) 的回傳值(當被拷貝的成員不存在回圈參考時) 和 傳給 copyMember(value,completeCB)completeCB 回呼函式 傳回;例外的情況是:如果 value 的某個成員存在回圈參考,則 copyMember(value) 會回傳 undefined,這種情況下,只能通過 completeCB 回呼函式 傳回拷貝后的值;

4.8.2. 預設presetTypeCopierMap

如果經常需要給 deepCopy() 傳遞包含相同 型別和拷貝者 的 typeCopyers 引數,則可以將那些經常用到的 拷貝者 設定到 deepCopy() 的預設 deepCopy.presetTypeCopierMap 中;

deepCopy 函式物件有個屬性 presetTypeCopierMap 是用來設定常用的拷貝者的,它的型別是 Map<Types, Reviver>,所以上面的例子也可以如下設定預設:

// 當需要拷貝 Person 型別的實體時,會回呼些函式
deepCopy.presetTypeCopierMap.set(Person,function(value,copyMember,options){
    let copy = new Person(value.sex);  // 創建新的 Person 實體,作為 value 的副本

    // 給副本 copy 設定基本型別的成員;
    copy.name = value.name;
    copy.email = value.email;

    // 如果參考型別的成員也需要深拷貝,則需要通過 copyMember() 函式拷貝 參考型別的成員
    copy.idCard = copyMember(value.idCard);
    /**
     * copyMember() 函式會回傳 拷貝后的值;
     * 但如果被拷貝的值存在回圈參考的話,copyMember() 會回傳 undefined;這種情況下,可通過給 copyMember() 傳遞回呼函式(第2個引數),通過回呼函式來獲取拷貝后的值;
     * 無論是否存在回圈參考,通過回呼函式總能拿到拷貝后的值,回呼函式是一種保險的方法
     */
    copyMember(value.mate,function(mateCopy){
        copy.mate = mateCopy;
    });

    // 需要將副本回傳;
    return copy;
})

以后使用 deepCopy() 時,就不需要傳遞 Person 型別的拷貝者了;

4.8.3. 拷貝者的優先級

當需要拷貝某個型別的實體時,deepCopy() 會通過以下幾種方法獲取該實體的副本:

  • 首先使用 typeCopyers 引數中的拷貝者;
  • 如果沒有找到拷貝者,則再使用預設 presetTypeCopierMap 中的拷貝者;
  • 如果沒有找到拷貝者,則會將實體的 getCopy 方法作為拷貝者;
  • 如果實體沒有 getCopy 方法,則會使用 typeCopyers 引數中 "default" 對應的 拷貝者;
  • 如果沒有找到拷貝者,則會使用預設 presetTypeCopierMap"default" 對應的 拷貝者;
  • 如果沒有找到拷貝者,則會使用內置的默認拷貝邏輯;

其中,"default" 對應的拷貝者稱為 默認的拷貝者;

4.8.4. getCopy

通過上文的 [拷貝者的優先級][] 可以知道,我們也可以給物件增加 getCopy 方法 來自定義 該物件的拷貝邏輯,如果您想將拷貝者應用到某個型別的所有實體上,則只需要給該型別增加一個實體方法 getCopy 即可;

getCopy 方法的 和 拷貝者 Copier 的型別一樣,都是型別為 (value:T,copyMember:CopyMember,options:CopierOptions)=>T 函式;

所以,上例中,也可以如下自定義 實體 p 的拷貝邏輯:

p.getCopy = function(value,copyMember,options){
    let copy = new Person(value.sex);  // 創建新的 Person 實體,作為 value 的副本

    // 給副本 copy 設定基本型別的成員;
    copy.name = value.name;
    copy.email = value.email;

    // 如果參考型別的成員也需要深拷貝,則需要通過 copyMember() 函式拷貝 參考型別的成員
    copy.idCard = copyMember(value.idCard);
    /**
     * copyMember() 函式會回傳 拷貝后的值;
     * 但如果被拷貝的值存在回圈參考的話,copyMember() 會回傳 undefined;這種情況下,可通過給 copyMember() 傳遞回呼函式(第2個引數),通過回呼函式來獲取拷貝后的值;
     * 無論是否存在回圈參考,通過回呼函式總能拿到拷貝后的值,回呼函式是一種保險的方法
     */
    copyMember(value.mate,function(mateCopy){
        copy.mate = mateCopy;
    });

    // 需要將副本回傳;
    return copy;
};

或,如下自定義 Person 型別 的拷貝邏輯:

Person.prototype.getCopy = function(value,copyMember,options){
    let copy = new Person(value.sex);  // 創建新的 Person 實體,作為 value 的副本

    // 給副本 copy 設定基本型別的成員;
    copy.name = value.name;
    copy.email = value.email;

    // 如果參考型別的成員也需要深拷貝,則需要通過 copyMember() 函式拷貝 參考型別的成員
    copy.idCard = copyMember(value.idCard);
    /**
     * copyMember() 函式會回傳 拷貝后的值;
     * 但如果被拷貝的值存在回圈參考的話,copyMember() 會回傳 undefined;這種情況下,可通過給 copyMember() 傳遞回呼函式(第2個引數),通過回呼函式來獲取拷貝后的值;
     * 無論是否存在回圈參考,通過回呼函式總能拿到拷貝后的值,回呼函式是一種保險的方法
     */
    copyMember(value.mate,function(mateCopy){
        copy.mate = mateCopy;
    });

    // 需要將副本回傳;
    return copy;
};

4.9. 創建帶預設拷貝規則的拷貝函式

如果您在使用 deepCopy() 時,經常需要使用不同的預設,那您可能需要使用 createDeepCopy(presetTypeCopierMap) 來創建一個帶有預設 presetTypeCopierMap 的另一個新的 deepCopy() 函式;

createDeepCopy(presetTypeCopierMap) 方法會創建一個新的深拷貝函式,這個新的深拷貝函式 和 deepCopy() 具有相同的功能,只是 不同的物件,且預設為傳給 createDeepCopy(presetTypeCopierMap) 函式的 presetTypeCopierMap 引數,如下:

const presetTypeCopierMap = new Map();
presetTypeCopierMap.set(Person,function(value,copyMember,options){
    let copy = new Person(value.sex);  // 創建新的 Person 實體,作為 value 的副本

    // 給副本 copy 設定基本型別的成員;
    copy.name = value.name;
    copy.email = value.email;

    // 如果參考型別的成員也需要深拷貝,則需要通過 copyMember() 函式拷貝 參考型別的成員
    copy.idCard = copyMember(value.idCard);
    /**
     * copyMember() 函式會回傳 拷貝后的值;
     * 但如果被拷貝的值存在回圈參考的話,copyMember() 會回傳 undefined;這種情況下,可通過給 copyMember() 傳遞回呼函式(第2個引數),通過回呼函式來獲取拷貝后的值;
     * 無論是否存在回圈參考,通過回呼函式總能拿到拷貝后的值,回呼函式是一種保險的方法
     */
    copyMember(value.mate,function(mateCopy){
        copy.mate = mateCopy;
    });

    // 需要將副本回傳;
    return copy;
});

// deepCopy2 為新的深拷貝函式,功能和 deepCopy 一樣;
const deepCopy2 = createDeepCopy(presetTypeCopierMap);

其中 deepCopy2 為新的深拷貝函式,功能和 deepCopy 一樣;通過 deepCopy2 來進行深拷貝操作,如下:

let copy = deepCopy2(value);

轉載請註明出處,本文鏈接:https://www.uj5u.com/qianduan/263487.html

標籤:其他

上一篇:VUE入門+5個小案例

下一篇:關于vue+node+socket.io的客服聊天專案優化空間請教

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • vue移動端上拉加載

    可能做得過于簡單或者比較low,請各位大佬留情,一起探討技術 ......

    uj5u.com 2020-09-10 04:38:07 more
  • 優美網站首頁,頂部多層導航

    一個個人用的瀏覽器首頁,可以把一下常用的網站放在這里,平常打開會比較方便。 第一步,HTML代碼 <script src=https://www.cnblogs.com/szharf/p/"js/jquery-3.4.1.min.js"></script> <div id="navigate"> <ul> <li class="labels labels_1"> ......

    uj5u.com 2020-09-10 04:38:47 more
  • 頁面為要加<!DOCTYPE html>

    最近因為寫一個js函式,需要用到$(window).height(); 由于手寫demo的時候,過于自信,其實對前端方面的認識也不夠體系,用文本檔案直接敲出來的html代碼,第一行沒有加上<!DOCTYPE html> 導致了$(window).height();的結果直接是整個document的高 ......

    uj5u.com 2020-09-10 04:38:52 more
  • WordPress網站程式手動升級要做好資料備份

    WordPress博客網站程式在進行升級前,必須要做好網站資料的備份,這個問題良家佐言是遇見過的;在剛開始接觸WordPress博客程式的時候,因為升級問題和博客網站的修改的一些嘗試,良家佐言是吃盡了苦頭。因為購買的是西部數碼的空間和域名,每當佐言把自己的WordPress博客網站搞到一塌糊涂的時候 ......

    uj5u.com 2020-09-10 04:39:30 more
  • WordPress程式不能升級為5.4.2版本的原因

    WordPress是一款個人博客系統,受到英文博客愛好者和中文博客愛好者的追捧,并逐步演化成一款內容管理系統軟體;它是使用PHP語言和MySQL資料庫開發的,用戶可以在支持PHP和MySQL資料庫的服務器上使用自己的博客。每一次WordPress程式的更新,就會牽動無數WordPress愛好者的心, ......

    uj5u.com 2020-09-10 04:39:49 more
  • 使用CSS3的偽元素進行首字母下沉和首行改變樣式

    網頁中常見的一種效果,首字改變樣式或者首行改變樣式,效果如下圖。 代碼: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, ......

    uj5u.com 2020-09-10 04:40:09 more
  • 關于a標簽的講解

    什么是a標簽? <a> 標簽定義超鏈接,用于從一個頁面鏈接到另一個頁面。 <a> 元素最重要的屬性是 href 屬性,它指定鏈接的目標。 a標簽的語法格式:<a href=https://www.cnblogs.com/summerxbc/p/"指定要跳轉的目標界面的鏈接">需要展示給用戶看見的內容</a> a標簽 在所有瀏覽器中,鏈接的默認外觀如下: 未被訪問的鏈接帶 ......

    uj5u.com 2020-09-10 04:40:11 more
  • 前端輪播圖

    在需要輪播的頁面是引入swiper.min.js和swiper.min.css swiper.min.js地址: 鏈接:https://pan.baidu.com/s/15Uh516YHa4CV3X-RyjEIWw 提取碼:4aks swiper.min.css地址 鏈接:https://pan.b ......

    uj5u.com 2020-09-10 04:40:13 more
  • 如何設定html中的背景圖片(全屏顯示,且不拉伸)

    1 <style>2 body{background-image:url(https://uploadbeta.com/api/pictures/random/?key=BingEverydayWallpaperPicture); 3 background-size:cover;background ......

    uj5u.com 2020-09-10 04:40:16 more
  • Java學習——HTML詳解(上)

    HTML詳解 初識HTML Hyper Text Markup Language(超文本標記語言) 1 <!--DOCTYPE:告訴瀏覽器我們要使用什么規范--> 2 <!DOCTYPE html> 3 <html lang="en"> 4 <head> 5 <!--meta 描述性的標簽,描述一些 ......

    uj5u.com 2020-09-10 04:40:33 more
最新发布
  • 我的第一個NPM包:panghu-planebattle-esm(胖虎飛機大戰)使用說明

    好家伙,我的包終于開發完啦 歡迎使用胖虎的飛機大戰包!! 為你的主頁添加色彩 這是一個有趣的網頁小游戲包,使用canvas和js開發 使用ES6模塊化開發 效果圖如下: (覺得圖片太sb的可以自己改) 代碼已開源!! Git: https://gitee.com/tang-and-han-dynas ......

    uj5u.com 2023-04-20 07:59:23 more
  • 生產事故-走近科學之消失的JWT

    入職多年,面對生產環境,盡管都是小心翼翼,慎之又慎,還是難免捅出簍子。輕則滿頭大汗,面紅耳赤。重則系統停擺,損失資金。每一個生產事故的背后,都是寶貴的經驗和教訓,都是專案成員的血淚史。為了更好地防范和遏制今后的各類事故,特開此專題,長期更新和記錄大大小小的各類事故。有些是親身經歷,有些是經人耳傳口授 ......

    uj5u.com 2023-04-18 07:55:04 more
  • 記錄--Canvas實作打飛字游戲

    這里給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 打開游戲界面,看到一個畫面簡潔、卻又富有挑戰性的游戲。螢屏上,有一個白色的矩形框,里面不斷下落著各種單詞,而我需要迅速地輸入這些單詞。如果我輸入的單詞與螢屏上的單詞匹配,那么我就可以獲得得分;如果我輸入的單詞錯誤或者時間過長,那么我就會輸 ......

    uj5u.com 2023-04-04 08:35:30 more
  • 了解 HTTP 看這一篇就夠

    在學習網路之前,了解它的歷史能夠幫助我們明白為何它會發展為如今這個樣子,引發探究網路的興趣。下面的這張圖片就展示了“互聯網”誕生至今的發展歷程。 ......

    uj5u.com 2023-03-16 11:00:15 more
  • 藍牙-低功耗中心設備

    //11.開啟藍牙配接器 openBluetoothAdapter //21.開始搜索藍牙設備 startBluetoothDevicesDiscovery //31.開啟監聽搜索藍牙設備 onBluetoothDeviceFound //30.停止監聽搜索藍牙設備 offBluetoothDevi ......

    uj5u.com 2023-03-15 09:06:45 more
  • canvas畫板(滑鼠和觸摸)

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>canves</title> <style> #canvas { cursor:url(../images/pen.png),crosshair; } #canvasdiv{ bo ......

    uj5u.com 2023-02-15 08:56:31 more
  • 手機端H5 實作自定義拍照界面

    手機端 H5 實作自定義拍照界面也可以使用 MediaDevices API 和 <video> 標簽來實作,和在桌面端做法基本一致。 首先,使用 MediaDevices.getUserMedia() 方法獲取攝像頭媒體流,并將其傳遞給 <video> 標簽進行渲染。 接著,使用 HTML 的 < ......

    uj5u.com 2023-01-12 07:58:22 more
  • 記錄--短視頻滑動播放在 H5 下的實作

    這里給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 短視頻已經無數不在了,但是主體還是使用 app 來承載的。本文講述 H5 如何實作 app 的視頻滑動體驗。 無聲勝有聲,一圖頂百辯,且看下圖: 網址鏈接(需在微信或者手Q中瀏覽) 從上圖可以看到,我們主要實作的功能也是本文要講解的有: ......

    uj5u.com 2023-01-04 07:29:05 more
  • 一文讀懂 HTTP/1 HTTP/2 HTTP/3

    從 1989 年萬維網(www)誕生,HTTP(HyperText Transfer Protocol)經歷了眾多版本迭代,WebSocket 也在期間萌芽。1991 年 HTTP0.9 被發明。1996 年出現了 HTTP1.0。2015 年 HTTP2 正式發布。2020 年 HTTP3 或能正... ......

    uj5u.com 2022-12-24 06:56:02 more
  • 【HTML基礎篇002】HTML之form表單超詳解

    ??一、form表單是什么

    ??二、form表單的屬性

    ??三、input中的各種Type屬性值

    ??四、標簽 ......

    uj5u.com 2022-12-18 07:17:06 more