🎉Ultra är live · 50% rabatt — begränsad tid
Prestandaguide för 3D-visning på webben — Implementeringsstrategier Three.js vs Babylon.js
2025/08/09

Prestandaguide för 3D-visning på webben — Implementeringsstrategier Three.js vs Babylon.js

Bemästra prestandan hos 3D-viewers på webben med vår omfattande guide. Jämför implementeringar av Three.js och Babylon.js, optimera laddningstider och lär dig professionella tekniker för transparenta bakgrunder, automatisk rotation och belysning.

Välj ditt 3D-webbramverk: prestanda först

När du implementerar 3D-viewers på webben handlar valet mellan Three.js och Babylon.js inte bara om funktioner — det handlar om prestanda, bundle-storlek och användarupplevelse. Den här guiden tillhandahåller beprövade strategier för att optimera prestandan hos 3D-visning, komplett med kodexempel och verkliga benchmarks.

Ramverksjämförelse: ur ett prestandaperspektiv

Snabb beslutsmatris

AspektThree.jsBabylon.js
Bundle-storlek~130KB (core)~2,5MB (630KB gzipped)
InlärningskurvaBrantareMjukare
Inbyggda funktionerMinimalaOmfattande
PrestandakontrollMaximalAutomatiserad
Bäst förSkräddarsydda lösningarSnabb utveckling

Filosofiska skillnader

  • Three.js: lätt rendering-motor med granulär kontroll
  • Babylon.js: komplett 3D-motor med allt inkluderat

Minimala implementeringsexempel

Three.js: transparent bakgrund med automatisk rotation

import * as THREE from 'three';
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader';

// Minimal Three.js-viewer setup
function initThreeViewer(canvas, modelUrl) {
  // Scenuppsättning med transparent bakgrund
  const scene = new THREE.Scene();
  scene.background = null; // Transparent

  // Kamera
  const camera = new THREE.PerspectiveCamera(
    75,
    canvas.width / canvas.height,
    0.1,
    1000
  );
  camera.position.z = 5;

  // Renderer med alphakanal
  const renderer = new THREE.WebGLRenderer({
    canvas,
    alpha: true,
    antialias: true,
    powerPreference: "high-performance"
  });
  renderer.setPixelRatio(Math.min(window.devicePixelRatio, 2));

  // Optimerad belysning
  const ambientLight = new THREE.AmbientLight(0xffffff, 0.6);
  const directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);
  directionalLight.position.set(10, 10, 5);
  scene.add(ambientLight, directionalLight);

  // Ladda GLB med automatisk rotation
  const loader = new GLTFLoader();
  let model;

  loader.load(modelUrl, (gltf) => {
    model = gltf.scene;
    scene.add(model);

    // Centrera och skala modellen
    const box = new THREE.Box3().setFromObject(model);
    const center = box.getCenter(new THREE.Vector3());
    model.position.sub(center);

    const size = box.getSize(new THREE.Vector3());
    const maxDim = Math.max(size.x, size.y, size.z);
    model.scale.multiplyScalar(2 / maxDim);
  });

  // Animationsslinga med automatisk rotation
  function animate() {
    requestAnimationFrame(animate);

    if (model) {
      model.rotation.y += 0.01;
    }

    renderer.render(scene, camera);
  }
  animate();

  // Hantera resize
  window.addEventListener('resize', () => {
    camera.aspect = canvas.width / canvas.height;
    camera.updateProjectionMatrix();
    renderer.setSize(canvas.width, canvas.height);
  });
}

Babylon.js: komplett viewer med skuggor

import * as BABYLON from '@babylonjs/core';
import '@babylonjs/loaders/glTF';

// Babylon.js-viewer med fulla funktioner
function initBabylonViewer(canvas, modelUrl) {
  // Motoruppsättning
  const engine = new BABYLON.Engine(canvas, true, {
    preserveDrawingBuffer: true,
    stencil: true,
    powerPreference: "high-performance"
  });

  // Scen med transparent bakgrund
  const scene = new BABYLON.Scene(engine);
  scene.clearColor = new BABYLON.Color4(0, 0, 0, 0);

  // Kamera med automatisk rotation
  const camera = new BABYLON.ArcRotateCamera(
    "camera",
    BABYLON.Tools.ToRadians(45),
    BABYLON.Tools.ToRadians(60),
    10,
    BABYLON.Vector3.Zero(),
    scene
  );
  camera.attachControl(canvas, true);
  camera.wheelDeltaPercentage = 0.01;

  // Optimerad belysning med skuggor
  const light = new BABYLON.DirectionalLight(
    "light",
    new BABYLON.Vector3(-1, -2, -1),
    scene
  );
  light.position = new BABYLON.Vector3(20, 40, 20);
  light.intensity = 0.7;

  const ambientLight = new BABYLON.HemisphericLight(
    "ambient",
    new BABYLON.Vector3(0, 1, 0),
    scene
  );
  ambientLight.intensity = 0.3;

  // Skuggenerator
  const shadowGenerator = new BABYLON.ShadowGenerator(1024, light);
  shadowGenerator.useExponentialShadowMap = true;

  // Ladda modell med optimering
  BABYLON.SceneLoader.LoadAssetContainer(
    "",
    modelUrl,
    scene,
    (container) => {
      container.addAllToScene();

      // Tillämpa skuggor på alla meshes
      container.meshes.forEach(mesh => {
        mesh.receiveShadows = true;
        shadowGenerator.addShadowCaster(mesh);
      });

      // Automatisk rotation
      scene.registerBeforeRender(() => {
        container.meshes[0].rotation.y += 0.01;
      });
    }
  );

  // Renderingsslinga
  engine.runRenderLoop(() => {
    scene.render();
  });

  // Hantera resize
  window.addEventListener('resize', () => {
    engine.resize();
  });
}

Strategier för prestandaoptimering

1. Modelloptimering

Kontroll av filstorlek

// Komprimeringsjämförelse
const modelSizes = {
  uncompressed: "26MB",
  draco: "5MB (-80%)",
  meshopt: "4MB (-85%)",
  quantized: "8MB (-70%)"
};

Texturkomprimering

// Three.js texturoptimering
const textureLoader = new THREE.TextureLoader();
const texture = textureLoader.load('texture.jpg');
texture.minFilter = THREE.LinearMipmapLinearFilter;
texture.magFilter = THREE.LinearFilter;
texture.generateMipmaps = true;

// Basis Universal-komprimering (50-75% mindre)
import { BasisTextureLoader } from 'three/examples/jsm/loaders/BasisTextureLoader';
const basisLoader = new BasisTextureLoader();
basisLoader.setTranscoderPath('basis/');
basisLoader.load('texture.basis', (texture) => {
  material.map = texture;
});

Mesh-förenkling

// LOD (Level of Detail)-implementering
const lod = new THREE.LOD();

// Hög detalj (nära)
const highDetail = await loadModel('model-300k.glb');
lod.addLevel(highDetail, 0);

// Mediumdetalj
const mediumDetail = await loadModel('model-60k.glb');
lod.addLevel(mediumDetail, 50);

// Låg detalj (långt borta)
const lowDetail = await loadModel('model-15k.glb');
lod.addLevel(lowDetail, 100);

scene.add(lod);

2. Implementering av lazy loading

// Progressiv laddning med platshållare
class LazyModel {
  constructor(placeholderUrl, highQualityUrl) {
    this.placeholder = placeholderUrl;
    this.highQuality = highQualityUrl;
    this.loaded = false;
  }

  async load(scene, callback) {
    // Ladda low-res-platshållare omedelbart
    const placeholder = await this.loadGLB(this.placeholder);
    scene.add(placeholder);
    callback(placeholder);

    // Ladda hög upplösning i bakgrunden
    const highQuality = await this.loadGLB(this.highQuality);

    // Mjuk övergång
    highQuality.visible = false;
    scene.add(highQuality);

    // Fade-övergång
    this.fadeTransition(placeholder, highQuality, () => {
      scene.remove(placeholder);
      this.loaded = true;
    });
  }

  fadeTransition(out, in, complete) {
    const duration = 500; // ms
    const start = performance.now();

    function animate() {
      const elapsed = performance.now() - start;
      const progress = Math.min(elapsed / duration, 1);

      out.material.opacity = 1 - progress;
      in.material.opacity = progress;

      if (progress < 1) {
        requestAnimationFrame(animate);
      } else {
        in.visible = true;
        complete();
      }
    }
    animate();
  }
}

3. Prestandaövervakning

// FPS-räknare och prestandastatistik
class PerformanceMonitor {
  constructor(renderer) {
    this.renderer = renderer;
    this.fps = 0;
    this.frame = 0;
    this.lastTime = performance.now();

    // GPU-minneövervakning
    this.memory = {
      geometries: 0,
      textures: 0,
      programs: 0
    };
  }

  update() {
    this.frame++;
    const currentTime = performance.now();

    if (currentTime >= this.lastTime + 1000) {
      this.fps = (this.frame * 1000) / (currentTime - this.lastTime);
      this.frame = 0;
      this.lastTime = currentTime;

      // Uppdatera minnesstatistik
      const info = this.renderer.info;
      this.memory = {
        geometries: info.memory.geometries,
        textures: info.memory.textures,
        programs: info.programs.length
      };

      console.log(`FPS: ${this.fps.toFixed(1)} | ` +
                  `Geometries: ${this.memory.geometries} | ` +
                  `Textures: ${this.memory.textures}`);
    }
  }
}

SEO- & tillgänglighetsoptimering

Platshållarbilder för SEO

<div class="model-viewer-container">
  <!-- SEO-vänlig platshållare -->
  <img
    src="model-preview.jpg"
    alt="3D-modell av produktnamn"
    loading="lazy"
    style="position: absolute; width: 100%; height: 100%;"
    id="placeholder"
  />

  <!-- 3D-canvas (dold initialt) -->
  <canvas
    id="viewer-canvas"
    style="display: none;"
    aria-label="Interaktiv 3D-modellviewer"
  />

  <!-- Laddningsindikator -->
  <div class="loading-spinner" style="display: none;">
    Laddar 3D-modell...
  </div>
</div>

<script>
// Progressiv förbättring
if (WebGL2RenderingContext) {
  // Ladda 3D-viewer
  loadViewer().then(() => {
    document.getElementById('placeholder').style.display = 'none';
    document.getElementById('viewer-canvas').style.display = 'block';
  });
} else {
  // Reserv till statisk bild
  console.log('WebGL stöds inte');
}
</script>

Strukturerade data för 3D-innehåll

<script type="application/ld+json">
{
  "@context": "https://schema.org/",
  "@type": "3DModel",
  "name": "Produktnamn 3D-modell",
  "description": "Interaktiv 3D-vy av produktnamn",
  "image": "https://example.com/model-preview.jpg",
  "encoding": {
    "@type": "3DModelEncoding",
    "encodingFormat": "model/gltf-binary",
    "contentUrl": "https://example.com/model.glb"
  }
}
</script>

Plattformsspecifika optimeringar

Mobilprestanda

// Adaptiv kvalitet baserat på enhet
function getQualitySettings() {
  const isMobile = /iPhone|iPad|Android/i.test(navigator.userAgent);
  const gpu = detectGPUTier();

  if (isMobile) {
    return {
      pixelRatio: Math.min(window.devicePixelRatio, 2),
      shadowMapSize: 512,
      textureSize: 1024,
      antialias: false,
      modelQuality: 'standard' // 30k polygoner
    };
  }

  // Desktopinställningar
  return {
    pixelRatio: window.devicePixelRatio,
    shadowMapSize: 2048,
    textureSize: 2048,
    antialias: true,
    modelQuality: gpu.tier > 2 ? 'ultra' : 'pro'
  };
}

Generator för embed-kod

// Embed-kod med ett klick för Modelfy 3D-modeller
function generateEmbedCode(modelId, options = {}) {
  const defaults = {
    width: '100%',
    height: '500px',
    autoRotate: true,
    background: 'transparent',
    controls: true,
    quality: 'auto'
  };

  const settings = { ...defaults, ...options };

  return `
<!-- Modelfy 3D-viewer -->
<iframe
  src="https://modelfy3d.com/embed/${modelId}"
  width="${settings.width}"
  height="${settings.height}"
  frameborder="0"
  allow="autoplay; fullscreen; xr-spatial-tracking"
  data-auto-rotate="${settings.autoRotate}"
  data-background="${settings.background}"
  data-controls="${settings.controls}"
  data-quality="${settings.quality}"
  loading="lazy"
></iframe>
  `.trim();
}

// Funktion för att kopiera till urklipp
function copyEmbedCode() {
  const code = generateEmbedCode('your-model-id', {
    width: '800px',
    height: '600px'
  });

  navigator.clipboard.writeText(code).then(() => {
    alert('Embed-koden kopierades till urklipp!');
  });
}

Prestandabenchmarks

Verkliga laddningstider

ModellkvalitetFilstorlekThree.js-laddningBabylon.js-laddningFPS (mobil)FPS (desktop)
Fast (15K)0,5MB0,8s1,2s6060
Standard (30K)1,2MB1,5s2,0s5560
Pro (60K)2,5MB2,8s3,5s4560
Ultra (300K)5MB5,2s6,8s2555

Optimeringspåverkan

// Före optimering
const unoptimized = {
  fileSize: "26MB",
  loadTime: "18s",
  fps: "15 (mobil)",
  memory: "450MB"
};

// Efter optimering
const optimized = {
  fileSize: "2.5MB (-90%)",
  loadTime: "2.8s (-84%)",
  fps: "45 (mobil)",
  memory: "95MB (-79%)"
};

Snabb implementeringschecklista

Innan du distribuerar din 3D-viewer:

  • Modellen komprimerad med Draco eller Meshopt
  • Texturer optimerade (WebP/Basis)
  • LOD implementerat för stora modeller
  • Pixel ratio begränsad på mobil
  • Lazy loading för bättre UX
  • SEO-platshållarbild tillagd
  • Tillgänglighetsetiketter inkluderade
  • Prestandaövervakning aktiv
  • Embed-kod testad

Börja optimera idag

Redo att implementera högpresterande 3D-viewers? Modelfy 3D tillhandahåller:

  • Förhandsoptimerade GLB-exporter
  • Flera kvalitetsnivåer för LOD
  • Generator för embed-kod
  • CDN-hostade viewer-bibliotek

Kom igång →

Resurser för utvecklare

Omfattande integrationsguider och dokumentation för bästa praxis inom prestanda kommer snart. Håll utkik efter detaljerade tutorials om Three.js- och Babylon.js-integration samt exempelimplementeringar.

Bemästra dessa tekniker och ditt 3D-innehåll laddas snabbare, körs smidigare och levererar exceptionella upplevelser på alla enheter. Framtiden för webb-3D är performance-first — se till att din implementering håller takten.

Nyhetsbrev

Bli en del av gemenskapen

Prenumerera på vårt nyhetsbrev för de senaste nyheterna och uppdateringarna