高級前端進階(六)

来源:https://www.cnblogs.com/ywjbokeyuan/archive/2022/09/07/16599152.html
-Advertisement-
Play Games

最近有個需求,就是上傳圖片的時候,圖片過大,需要壓縮一下圖片再上傳。 需求雖然很容易理解,但要做到,不是那麼容易的。 這裡涉及到的知識有點多,不多說,本篇博客有點重要呀! 一、圖片URL轉Blob(圖片大小不變) 註意點:圖片不能跨域!!! 方式一:通過XHR請求獲取 function urlToB ...


最近有個需求,就是上傳圖片的時候,圖片過大,需要壓縮一下圖片再上傳。
需求雖然很容易理解,但要做到,不是那麼容易的。
這裡涉及到的知識有點多,不多說,本篇博客有點重要呀!

一、圖片URL轉Blob(圖片大小不變)

註意點:圖片不能跨域!!!

方式一:通過XHR請求獲取

function urlToBlobByXHR(url) {
    const xhr = new XMLHttpRequest();
    xhr.open("get", url);
    xhr.responseType = "blob"; // 設置響應請求格式
    xhr.onload = (e) => {
        if (e.target.status == 200) {
            console.log(e.target.response); // e.target.response返回的就是Blob。
            return e.target.response;// 這樣是不行的
        }
        else {
            console.log("異常");
        }
    };
    xhr.send();
}
urlToBlobByXHR("圖片URL"); // 調用

我們知道,XHR操作是非同步的,只有在onload方法裡面才能獲取到Blob,相應的業務代碼也要寫到裡面。怎麼能夠做到調用這個方法,直接得到Blob結果呢?
Promise便解決了諸如此類的痛點。

function urlToBlobByXHR(url) {
    return new Promise((resolve, reject) => {
        const xhr = new XMLHttpRequest();
        xhr.open("get", url);
        xhr.responseType = "blob";
        xhr.onload = (e) => {
            if (e.target.status == 200) {
                resolve(e.target.response); // resolve
            }
            else {
                reject("異常"); // reject
            }
        };
        xhr.send();
    })
}
async f() {
    try {
    console.log(await urlToBlobByXHR(this.imgUrl)); // 直接返回Blob
  } catch (e) {
    console.log(e);
  }
}
f(); // 調用

方式二:通過canvas轉化(圖片大小會變大很多)

基本原理:就是新建一個canvas元素,然後在裡面將圖片畫上去,接著利用canvas轉為Blob。

function canvasToBlob(imgUrl) {
    return new Promise((resolve, reject) => {
        const imgObj = new Image();
        imgObj.src = imgUrl;
        imgObj.onload = () => {
            const canvasObj = document.createElement("canvas");
            const ctx = canvasObj.getContext("2d");
            canvasObj.width = imgObj.naturalWidth;
            canvasObj.height = imgObj.naturalHeight;
            ctx.drawImage(imgObj, 0, 0, canvasObj.width, canvasObj.height);
            canvasObj.toBlob((blob) => {
                resolve(blob);
            });
        }
    })
}

const blobCanvas = await canvasToBlob(imgUrl); // 調用,直接獲取到blob

不過呢,利用canvas轉化,圖片會變大很多,在canvas上面畫圖片,期間圖片解析度會改變,加上可能還有圖片解析的原因,會導致圖片變大很多。
而且canvas是可以截圖的,不過這一點是人為可以控制的。

二、圖片壓縮

原理:我們知道在canvas裡面畫圖,canvas相當於圖片的容器,既然是容器,那便可以控制容器的寬高,相應的改變圖片的寬高,通過這一點,不就可以縮小圖片了嗎?
不過要註意的是,縮小圖片要等比例的縮小,雖然提供的介面裡面支持更改圖片清晰度,但個人並不建議這麼做,至於原因自己想吧。

版本一:

// imageUrl:圖片URL,圖片不能跨域
// maxSize:圖片最大多少M
// scale:圖片放大比例
function compressImg1(imageUrl, maxSize = 1, scale = 0.8, imgWidth, imgHeight) {
    let maxSizeTemp = maxSize * 1024 * 1024;
    return new Promise((resolve, reject) => {
        const imageObj = new Image();
        imageObj.src = imageUrl;
        imageObj.onload = () => {
            const canvasObj = document.createElement("canvas");
            const ctx = canvasObj.getContext("2d");
            if (imgWidth && imgHeight) { // 等比例縮小
                canvasObj.width = scale * imgWidth;
                canvasObj.height = scale * imgHeight;
            }
            else {
                canvasObj.width = imageObj.naturalWidth;
                canvasObj.height = imageObj.naturalHeight;
            }
            ctx.drawImage(imageObj, 0, 0, canvasObj.width, canvasObj.height);
            canvasObj.toBlob((blob) => {
                resolve({ blob, canvasObj });
            });
        }
    }).then(({ blob, canvasObj }) => {
        if (blob.size / maxSizeTemp < maxSize) {
            let file = new File([blob], `test${imageUrl.substring(imageUrl.lastIndexOf("."))}`);
            return Promise.resolve({ blob, file });
        }
        else {
            return compressImg1(imageUrl, maxSize, scale, canvasObj.width, canvasObj.height); // 遞歸調用
        }
    })
}
const { blob } = await compressImg1("圖片地址"); // 調用

需求是實現了,但用到了遞歸,性能完全由縮小比例跟圖片大小決定。
圖片過大的話或者縮小比例大了點,會導致不斷遞歸,性能低下,這是肯定的。
以上還有兩個耗時的操作:
1、不斷請求圖片
2、不斷操作DOM

版本二:

有個潛規則,能不用遞歸就不用遞歸。
試想,怎樣一步到位可以把圖片縮小到需要的大小呢?再深入直接一點,如何得到有效的scale,等比例縮小後就能使圖片縮小到想要的程度呢?
然後再把以上兩個耗時操作再優化一下,只需載入一次圖片。便得到了版本二。

function compressImg2(imageUrl, maxSize = 1, scale = 1) {
    let maxSizeTemp = maxSize * 1024 * 1024;
    return new Promise((resolve, reject) => {
        const imageObj = new Image(); // 只需載入一次圖片
        imageObj.src = imageUrl;
        imageObj.onload = () => {
            const canvasObj = document.createElement("canvas"); // 只需創建一次畫布
            const ctx = canvasObj.getContext("2d");
            canvasObj.width = imageObj.naturalWidth;
            canvasObj.height = imageObj.naturalHeight;
            ctx.drawImage(imageObj, 0, 0, canvasObj.width, canvasObj.height);
            canvasObj.toBlob((blob1) => {
                resolve({ imageObj, blob1, canvasObj, ctx });
            });
        }
    }).then(({ imageObj, blob1, canvasObj, ctx }) => {
        if (blob1.size / maxSizeTemp < maxSize) {
            let file = new File([blob1], `test${imageUrl.substring(imageUrl.lastIndexOf("."))}`);
            return Promise.resolve({ blob: blob1, file });
        }
        else {
            const ratio = Math.round(blob1.size / maxSizeTemp); // 比例
            canvasObj.width = (imageObj.naturalWidth / ratio) * scale; // 比例調整
            canvasObj.height = (imageObj.naturalHeight / ratio) * scale;
            ctx.drawImage(imageObj, 0, 0, canvasObj.width, canvasObj.height);
            return new Promise((resolve) => {
                canvasObj.toBlob((blob2) => {
                    let file = new File([blob2], `test${imageUrl.substring(imageUrl.lastIndexOf("."))}`);
                    resolve({ blob: blob2, file });
                });
            })
        }
    })
}

版本三(Promise轉為async await)

我們知道Promise跟asnc await是等價的。

async function compressImg(imageUrl, maxSize = 1, scale = 1) {
    let maxSizeTemp = maxSize * 1024 * 1024;
    const { imageObj, blob1, canvasObj, ctx } = await new Promise((resolve, reject) => {
        const imageObj = new Image();
        imageObj.src = imageUrl;
        imageObj.onload = () => {
            const canvasObj = document.createElement("canvas");
            const ctx = canvasObj.getContext("2d");
            canvasObj.width = imageObj.naturalWidth;
            canvasObj.height = imageObj.naturalHeight;
            // console.log(canvasObj);
            ctx.drawImage(imageObj, 0, 0, canvasObj.width, canvasObj.height);
            canvasObj.toBlob((blob1) => {
                // console.log('blob1', blob1);
                resolve({ imageObj, blob1, canvasObj, ctx });
            });
        };
    });
    if (blob1.size / maxSizeTemp < maxSize) {
        let file = new File([blob1], `test${imageUrl.substring(imageUrl.lastIndexOf("."))}`);
        return Promise.resolve({ blob: blob1, file });
    }
    else {
        // const ratio = Math.round(Math.sqrt(blob1.size / maxSizeTemp));
        const ratio = Math.round(blob1.size / maxSizeTemp);
        // console.log('ratio', ratio);
        canvasObj.width = (imageObj.naturalWidth / ratio) * scale;
        canvasObj.height = (imageObj.naturalHeight / ratio) * scale;
        // console.log(canvasObj);
        ctx.drawImage(imageObj, 0, 0, canvasObj.width, canvasObj.height);
        const { blob: blob2, file } = await new Promise((resolve) => {
            canvasObj.toBlob((blob2) => {
                // console.log('blob2', blob2);
                let file = new File([blob2], `test${imageUrl.substring(imageUrl.lastIndexOf("."))}`);
                resolve({ blob: blob2, file });
            });
        })
        return { blob: blob2, file };
    }
}

三、詳細講解下Promise

簡單的一個例子

let p = new Promise((resolve) => {
  setTimeout(() => {
    resolve(123456); // 5秒後輸出123456
  }, 5000);
});
p.then((s) => {
  console.log(s); // 通過then的參數就可以獲取到結果
});

let s = await p; // async await轉換,簡化then寫法
console.log(s);

其實呢,Promise本質上就是回調函數的使用,而Promise主要是為瞭解決回調地獄(回調函數嵌套)而出現的,async await寫法主要是為了簡化方便。

咱來模擬一下最簡單的Promise,手寫一個簡單一點的。

// 首先定義一下Promise狀態
const status = {
  pending: "pending",
  fulfilled: "fulfilled",
  rejected: "rejected",
};

不支持非同步(先來個簡單的)

function MyPromise(executor) {
  const self = this;// this指向
  self.promiseStatus = status.pending;
  self.promiseValue = undefined;
  self.reason = undefined;
  function resolve(value) {
    if (self.promiseStatus == status.pending) {
      self.promiseStatus = status.fulfilled;
      self.promiseValue = value;
    }
  }
  function reject(reason) {
    if (self.promiseStatus == status.pending) {
      self.promiseStatus = status.rejected;
      self.reason = reason;
    }
  }
  try {
    executor(resolve, reject); // 在這裡比較難以理解,函數resolve作為函數executor的參數,new MyPromise調用的時候,傳的也是個函數。
  } catch (e) {
    reject(e);
  }
}
MyPromise.prototype.then = function (onResolve, onReject) { // 利用原型添加方法
  const self = this;
  if (self.promiseStatus == status.fulfilled) {
    onResolve(self.promiseValue);
  }
  if (self.promiseStatus == status.rejected) {
    onReject(self.reason);
  }
};
// 調用
const myPromise = new MyPromise((resolve, reject) => { // MyPromise的參數也是個函數
  resolve(123456); // 暫時不支持非同步
});
myPromise.then((data) => {
  console.log("data", data); // 輸出123456
});

支持非同步的

function MyPromise(executor) {
  const self = this;
  self.promiseStatus = status.pending;
  self.promiseValue = undefined;
  self.reason = undefined;
  self.onResolve = [];
  self.onReject = [];
  function resolve(value) {
    if (self.promiseStatus == status.pending) {
      self.promiseStatus = status.fulfilled;
      self.promiseValue = value;
      self.onResolve.forEach((fn) => fn(value)); //支持非同步
    }
  }
  function reject(reason) {
    if (self.promiseStatus == status.pending) {
      self.promiseStatus = status.rejected;
      self.reason = reason;
      self.onReject.forEach((fn) => fn(reason)); // //支持非同步
    }
  }
  try {
    executor(resolve, reject);
  } catch (e) {
    reject(e);
  }
}
MyPromise.prototype.then = function (onResolve, onReject) {
  const self = this;
  if (self.promiseStatus == status.fulfilled) {
    onResolve(self.promiseValue);
  }
  if (self.promiseStatus == status.rejected) {
    onReject(self.reason);
  }
  if (self.promiseStatus == status.pending) {
    self.onResolve.push(onResolve);
    self.onReject.push(onReject);
  }
};
// 調用
const myPromise = new MyPromise((resolve, reject) => {
  setTimeout(() => {
    resolve(123456); // 非同步
  }, 3000);
});
myPromise.then((data) => {
  console.log("data", data); // 輸出123456
});

個人覺得,能明白大致原理,會用就行了,至於能不能手寫一個Promise並不是很重要的,不斷重覆造輪子沒啥意思,
但是呢,理解其大概思路以及實現所用到的思想還是很重要的,對成長的幫助很大。

總結

圖片壓縮還有待優化
Promise,大家應該都很熟悉,用的非常多,可真正會用的人並不是太多的。

最後,祝大家中秋快樂!

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、簡介 簡單記錄一下存儲過程的使用。存儲過程是預編譯SQL語句集合,也可以包含一些邏輯語句,而且當第一次調用存儲過程時,被調用的存儲過程會放在緩存中,當再次執行時,則不需要編譯可以立馬執行,使得其執行速度會非常快。 二、使用 創建格式 create procedure 過程名( 變數名 變數類型 ...
  • 摘要:北京國家金融科技認證中心正式公佈了2022年通過“分散式資料庫金融標準驗證”的資料庫產品名單。華為雲GaussDB金融級分散式資料庫以突出的技術優勢通過驗證,躍然榜上,且測試得分遙居前列。 近日,北京國家金融科技認證中心正式公佈了2022年通過“分散式資料庫金融標準驗證”的資料庫產品名單。華為 ...
  • 在2022世界人工智慧大會(WAIC)上,騰訊雲資料庫技術負責人程彬為大家分享了資料庫與 AI 相結合背後的故事。在專場《當資料庫遇上 AI 》中,程彬基於騰訊雲資料庫在 AI 智能化的探索與實踐,剖析資料庫與 AI 融合背後的技術關鍵點,為產業界提供前沿解決方案。以下為演講實錄: 點擊觀看完整版直 ...
  • 項目管理構建工具——Maven(基礎篇) 在前面的內容中我們學習了JDBC並且接觸到了jar包概念 在後面我們的實際開發中會接觸到很多jar包,jar包的導入需要到互聯網上進行就會導致操作繁瑣 Maven在解決了jar包導入繁雜問題的同時,也提供了一套通用的管理和構建Java項目的一系列操作 Mav ...
  • 近年來隨著物聯網技術以及農業自動化應用水平的不斷發展,基於“互聯網+”項目經驗日漸豐富,通過感測器採集對應的信息再通過一些組態軟體類實現自動運轉、自動控制的智能大棚勢在必得。 ...
  • 上傳IPA到iTunes Connect 上一篇我介紹瞭如何在iTunes Connect里準備應用。最後在這篇文章里我會簡單介紹下如何來上傳IPA到iTunes Connect。 登陸iTunes Connect,進入Manage Your Applications頁面後,點擊你創建的應用圖標,進 ...
  • DanceCC提出了一套專門的方案。方案原理基於LLDB Plugin,利用Fishhook,從LLDB的Script Bridge API層面攔截Xcode對LLDB調用,以此來進行耗時監控統計。 ...
  • HMS Core應用內支付服務(In-App Purchases,IAP)為應用提供便捷的應用內支付體驗和簡便的接入流程。開發者的應用集成IAP SDK後,調用IAP SDK介面,啟動IAP收銀台,即可實現應用內支付。 通過應用內支付服務,用戶可以在應用內購買各種類型的虛擬商品,包括一次性商品(包括 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...