使用JavaScript预加载图像

2020/10/07 18:41 · javascript ·  · 0评论

我在下面编写的函数是否足以在当今大多数(如果不是全部)浏览器中预加载图像?

function preloadImage(url)
{
    var img=new Image();
    img.src=url;
}

我有一个图像URL数组,可以循环访问这些URL,并preloadImage为每个URL调用该函数。

是。这应该适用于所有主要的浏览器。

试试这个,我认为这样更好。

var images = [];
function preload() {
    for (var i = 0; i < arguments.length; i++) {
        images[i] = new Image();
        images[i].src = preload.arguments[i];
    }
}

//-- usage --//
preload(
    "http://domain.tld/gallery/image-001.jpg",
    "http://domain.tld/gallery/image-002.jpg",
    "http://domain.tld/gallery/image-003.jpg"
)

资料来源:http : //perishablepress.com/3-ways-preload-images-css-javascript-ajax/

就我而言,将回调添加到onload事件的函数中非常有用

function preloadImage(url, callback)
{
    var img=new Image();
    img.src=url;
    img.onload = callback;
}

然后包装它,以完成URL数组的情况,以完成对所有图像的回调预加载:https :
//jsfiddle.net/4r0Luoy7/

function preloadImages(urls, allImagesLoadedCallback){
    var loadedCounter = 0;
  var toBeLoadedNumber = urls.length;
  urls.forEach(function(url){
    preloadImage(url, function(){
        loadedCounter++;
            console.log('Number of loaded images: ' + loadedCounter);
      if(loadedCounter == toBeLoadedNumber){
        allImagesLoadedCallback();
      }
    });
  });
  function preloadImage(url, anImageLoadedCallback){
      var img = new Image();
      img.onload = anImageLoadedCallback;
      img.src = url;
  }
}

// Let's call it:
preloadImages([
    '//upload.wikimedia.org/wikipedia/commons/d/da/Internet2.jpg',
  '//www.csee.umbc.edu/wp-content/uploads/2011/08/www.jpg'
], function(){
    console.log('All images were loaded');
});

CSS2替代:http//www.thecssninja.com/css/even-better-image-preloading-with-css2

body:after {
  content: url(img01.jpg) url(img02.jpg) url(img03.jpg);
  display: none; 
}

CSS3替代:https//perishablepress.com/preload-images-css3/
(H / T Linh大坝)

.preload-images {
  display: none; 
  width: 0;
  height: 0;
  background: url(img01.jpg),
              url(img02.jpg),
              url(img03.jpg);
}

注意:容器中的图像display:none可能不会预加载。也许能见度:隐藏会更好,但我尚未对此进行测试。感谢Marco Del Valle指出了这一点

这种方法更加精细。在这里,您将所有预加载的图像存储在容器中,可能是div。然后,您可以显示图像或将其在DOM中移动到正确的位置。

function preloadImg(containerId, imgUrl, imageId) {
    var i = document.createElement('img'); // or new Image()
    i.id = imageId;
    i.onload = function() {
         var container = document.getElementById(containerId);
         container.appendChild(this);
    };
    i.src = imgUrl;
}

在这里尝试,我也添加了一些评论

我建议您使用try / catch来防止某些可能的问题:

面向对象:

    var preloadImage = function (url) {
        try {
            var _img = new Image();
            _img.src = url;
        } catch (e) { }
    }

标准:

    function preloadImage (url) {
        try {
            var _img = new Image();
            _img.src = url;
        } catch (e) { }
    }

另外,虽然我喜欢DOM,但是旧的愚蠢的浏览器在使用DOM时可能会遇到问题,因此请完全避免使用IMHO,这与freedev的贡献相反。Image()在旧的垃圾桶浏览器中有更好的支持。

const preloadImage = src => 
  new Promise(r => {
    const image = new Image()
    image.onload = r
    image.onerror = r
    image.src = src
  })


// Preload an image
await preloadImage('https://picsum.photos/100/100')

// Preload a bunch of images in parallel 
await Promise.all(images.map(x => preloadImage(x.src)))

是的,这将起作用,但是浏览器将限制(在4-8之间)实际调用,因此不会缓存/预加载所有所需的图像。

更好的方法是在使用图像之前调用onload,如下所示:

function (imageUrls, index) {  
    var img = new Image();

    img.onload = function () {
        console.log('isCached: ' + isCached(imageUrls[index]));
        *DoSomething..*

    img.src = imageUrls[index]
}

function isCached(imgUrl) {
    var img = new Image();
    img.src = imgUrl;
    return img.complete || (img .width + img .height) > 0;
}

这是我的方法:

var preloadImages = function (srcs, imgs, callback) {
    var img;
    var remaining = srcs.length;
    for (var i = 0; i < srcs.length; i++) {
        img = new Image;
        img.onload = function () {
            --remaining;
            if (remaining <= 0) {
                callback();
            }
        };
        img.src = srcs[i];
        imgs.push(img);
    }
};

符合ECMAScript 2017的浏览器解决方案

注意:如果您使用像Babel这样的编译器,这也将起作用。

'use strict';

function imageLoaded(src, alt = '') {
    return new Promise(function(resolve) {
        const image = document.createElement('img');

        image.setAttribute('alt', alt);
        image.setAttribute('src', src);

        image.addEventListener('load', function() {
            resolve(image);
        });
    });
}

async function runExample() {
    console.log("Fetching my cat's image...");

    const myCat = await imageLoaded('https://placekitten.com/500');

    console.log("My cat's image is ready! Now is the time to load my dog's image...");

    const myDog = await imageLoaded('https://placedog.net/500');

    console.log('Whoa! This is now the time to enable my galery.');

    document.body.appendChild(myCat);
    document.body.appendChild(myDog);
}

runExample();

您可能还已经等待所有图像加载。

async function runExample() {
    const [myCat, myDog] = [
        await imageLoaded('https://placekitten.com/500'),
        await imageLoaded('https://placedog.net/500')
    ];

    document.body.appendChild(myCat);
    document.body.appendChild(myDog);
}

或用于Promise.all并行加载它们。

async function runExample() {
    const [myCat, myDog] = await Promise.all([
        imageLoaded('https://placekitten.com/500'),
        imageLoaded('https://placedog.net/500')
    ]);

    document.body.appendChild(myCat);
    document.body.appendChild(myDog);
}

有关承诺的更多信息

有关“异步”功能的更多信息

有关销毁任务的更多信息

有关ECMAScript 2015的更多信息

有关ECMAScript 2017的更多信息

您可以将此代码移动到index.html,以从任何url预加载图像

<link rel="preload" href="https://via.placeholder.com/160" as="image">

我可以确认问题中的方法足以触发图像的下载和缓存(除非您已通过响应头禁止浏览器这样做),至少应在以下条件中进行:

  • 镀铬74
  • Safari 12
  • Firefox 66
  • 边缘17

为了测试这一点,我制作了一个小型Web应用程序,其中包含多个端点,每个端点睡眠10秒钟,然后再提供小猫的照片。然后,我添加了两个网页,其中一个包含一个<script>标签,其中使用问题中的preloadImage函数预加载了每只小猫,而另一个包含了使用<img>标签的页面上的所有小猫

在上面的所有浏览器中,我发现如果我先访问预加载器页面,等待一会儿,然后转到带有<img>标签的页面,则我的小猫会立即呈现。这表明预加载器已在所有测试的浏览器中成功将小猫加载到缓存中。

您可以在https://github.com/ExplodingCabbage/preloadImage-test上查看或试用我用来测试它的应用程序

特别要注意的是,即使循环的图像数量超过了浏览器一次愿意做出的并行请求的数量,该技术也可以在上面的浏览器中使用,这与罗宾的回答相反图像预加载的速率当然会受到浏览器愿意发送的并行请求数量的限制,但最终它将请求您调用的每个图像URL preloadImage()

截至2020年的工作解决方案

这篇文章的大多数答案不再起作用-(至少在Firefox上)

这是我的解决方案:

var cache = document.createElement("CACHE");
cache.style = "position:absolute;z-index:-1000;opacity:0;";
document.body.appendChild(cache);
function preloadImage(url) {
    var img = new Image();
    img.src = url;
    img.style = "position:absolute";
    cache.appendChild(img);
}

用法:

preloadImage("example.com/yourimage.png");

显然,<cache>这不是“已定义”元素,因此您可以根据需要使用a <div>

在CSS中使用它,而不是应用style属性:

cache {
    position: absolute;
    z-index: -1000;
    opacity: 0;
}

cache image {
    position: absolute;
}

如果您对此进行了测试,请发表评论。

笔记:

  • 不适display: none;用于缓存-这不会加载图像。
  • 请勿调整图像元素的大小,因为这也会影响使用图像时所加载图像的质量。
  • position: absolute必须对图像进行设置,因为图像元素最终将使其脱离视口-导致它们无法加载并影响性能。

更新

尽管上述解决方案有效,但我还是做了一些小的更新,以使其结构良好:

(这现在也可以在一个功能中接受多个图像)

var cache = document.createElement("CACHE");
document.body.appendChild(cache);
function preloadImage() {
    for (var i=0; i<arguments.length; i++) {
        var img = new Image();
        img.src = arguments[i];
        var parent = arguments[i].split("/")[1]; // Set to index of folder name
        if ($(`cache #${parent}`).length == 0) {
            var ele = document.createElement("DIV");
            ele.id = parent;
            cache.appendChild(ele);
        }
        $(`cache #${parent}`)[0].appendChild(img);
        console.log(parent);
    }
}

preloadImage(
    "assets/office/58.png",
    "assets/leftbutton/124.png",
    "assets/leftbutton/125.png",
    "assets/leftbutton/130.png",
    "assets/leftbutton/122.png",
    "assets/leftbutton/124.png"
);

预习:

在此处输入图片说明

笔记:

  • 尽量不要同时保留太多图像(这可能会导致严重的性能问题)-我通过隐藏图像来解决此问题,我知道在某些事件中这些图像将不可见。然后,当然,在需要时再次向他们展示。

使用头部的link标签,浏览器将运行最佳。

export function preloadImages (imageSources: string[]): void {
  imageSources
    .forEach(i => {
      const linkEl = document.createElement('link');
      linkEl.setAttribute('rel', 'preload');
      linkEl.setAttribute('href', i);
      linkEl.setAttribute('as', 'image');
      document.head.appendChild(linkEl);
    });
}
本文地址:http://javascript.askforanswer.com/shiyongjavascriptyujiazaituxiang.html
文章标签:
版权声明:本文为原创文章,版权归 javascript 所有,欢迎分享本文,转载请保留出处!

文件下载

老薛主机终身7折优惠码boke112

上一篇:
下一篇:

评论已关闭!