Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
Высокопроизводительные визуализации данных в браузере (CodeFest)
Search
Vladimir Agafonkin
March 29, 2013
Programming
2
450
Высокопроизводительные визуализации данных в браузере (CodeFest)
Vladimir Agafonkin
March 29, 2013
Tweet
Share
More Decks by Vladimir Agafonkin
See All by Vladimir Agafonkin
Fast by Default: Algorithmic Optimization in Practice (dotJS 2019)
mourner
0
1.9k
Fast by Default: Algorithmic Optimization in Practice (JSCamp 2019)
mourner
0
330
How Vector Maps Work (2019)
mourner
1
310
Fast by default: everyday algorithmic thinking for developers [RU]
mourner
1
250
Fast by default: everyday algorithmic thinking for developers
mourner
1
410
Mapbox GL: How vector maps work
mourner
0
300
Random cool stuff I've been hacking on (mostly spatial indices)
mourner
0
430
Как работают современные интерактивные карты на WebGL
mourner
1
440
Leaflet, WebGL & the Future of Web Mapping
mourner
5
3.1k
Other Decks in Programming
See All in Programming
[技育CAMPアカデミア]アイディアを形に!【超入門】スマホアプリ開発〜リリースまでの流れをご紹介
teamlab
PRO
0
360
スキーマ駆動開発による品質とスピードの両立 - 私達は何故、スキーマを書くのか
kentaroutakeda
0
170
OpenAPIを中心に考えるAPI開発入門 / Introduction to API Development with a Focus on OpenAPI
seike460
PRO
2
170
Build Apps for iOS, Android & Desktop in 100% Kotlin With Compose Multiplatform (mDevCamp 2024)
zsmb
0
280
"config" ってなんだ? / What is "config"?
okashoi
0
240
DMMプラットフォームがTiDB Cloudを採用した背景
pospome
8
4k
0→1と1→10の狭間で Javaという技術選定を振り返る/Reflecting on the Decision to Choose Java Between Scaling from 0 to 1 and 1 to 10
jaguar_imo
2
380
はてなにおける CSS Modules、及び CSS Modules に足りないもの / CSS Modules in Hatena, and CSS Modules missing parts
mizdra
7
910
Goのエラースタックトレースの歴史と今後
sonatard
7
1.1k
⼤規模⾔語モデルの拡張(RAG)が 終わったかも知れない件について
nearme_tech
23
15k
Changed Rules: Architectures with Lightweight Stores
manfredsteyer
PRO
0
240
1BRC--Nerd Sniping the Java Community
gunnarmorling
0
340
Featured
See All Featured
The Success of Rails: Ensuring Growth for the Next 100 Years
eileencodes
30
6k
Designing with Data
zakiwarfel
96
4.8k
Design and Strategy: How to Deal with People Who Don’t "Get" Design
morganepeng
116
18k
The Power of CSS Pseudo Elements
geoffreycrofte
60
5k
Designing for Performance
lara
601
67k
How STYLIGHT went responsive
nonsquared
92
4.8k
Creating an realtime collaboration tool: Agile Flush - .NET Oxford
marcduiker
14
1.5k
What's new in Ruby 2.0
geeforr
337
31k
GraphQLとの向き合い方2022年版
quramy
32
12k
Fight the Zombie Pattern Library - RWD Summit 2016
marcelosomers
227
16k
Building Effective Engineering Teams - LeadDev
addyosmani
28
1.8k
Building Adaptive Systems
keathley
31
1.9k
Transcript
Владимир Агафонкин высокопроизводительные визуализации данных в браузере март 2013
agafonkin.com /mourner
rain.in.ua
визуализации данных
None
None
None
None
None
None
интерактивность
•реагирование на действия пользователя интерактивность
•реагирование на действия пользователя •навигация по данным интерактивность
•реагирование на действия пользователя •навигация по данным •фильтрация данных интерактивность
None
None
None
None
None
потребность в интерактивности увеличивается
ответственность за обработку данных всё больше смещается от сервера к
клиенту
чистый JS — быстро рендеринг, DOM — медленно
правило №1 чем меньше всего отображаем, тем быстрее
обработка данных рендеринг
обработка данных рендеринг
•фильтрация объектов уменьшение данных
•фильтрация объектов •геометрическое отсечение уменьшение данных
•фильтрация объектов •геометрическое отсечение •геометрическое упрощение уменьшение данных
•фильтрация объектов •геометрическое отсечение •геометрическое упрощение •кластеризация (группировка) уменьшение данных
фильтрация
поиск 2D-объектов •поиск объектов в текущей прямоугольной видимой области
поиск 2D-объектов •поиск объектов в текущей прямоугольной видимой области •поиск
объектов в точке (под курсором)
загрузка данных — 1 раз поиск/фильтрация — много раз
сетка
None
древовидные структуры данных •binary heap •binary search tree •range tree
•k-d tree •quadtree •R-tree
точки в quadtree
прямоугольники в R-tree
var tree = new RTree(); tree.insert( {x: 5, y: 10,
w: 10, h: 15}, obj); ... tree.search( {x: 7, y: 7, w: 5, h: 5}); github.com/imbcmdth/RTree
kothic.org/js
определение коллизий
Crossfilter (для многих измерений)
отсечение ломаных линий алгоритм Коэна-Сазерленда
отсечение полигонов алгоритм Сазерленда-Ходжмана
упрощение ломаных линий
упрощение по расстоянию
алгоритм Дугласа-Пекера
mourner.github.com/simplify-js
кластеризация группировка похожих объектов
None
None
иерархическая кластеризация 1 раз для всех масштабов
загрузка и обработка данных
UI JS UI
UI JS браузер залипает на объемных вычислениях UI
Web Workers
<script src='data.js'></script> <script> ... worker.postMessage(HUGE_DATA_ARRAY); ... </script> загрузка и пересылка
в Worker
UI Worker JS загрузка данных UI
UI Worker JS браузер залипает на загрузке и пересылке данных
загрузка данных UI
importScripts('data.js'); ... onmessage = function (e) { var result =
searchData(e.data.query); postMessage(result); } загрузка в Worker
UI Worker JS загрузка данных UI UI
UI Worker JS браузер залипает на получении данных загрузка данных
UI UI
var array = new Float16Array(len); ... var buffer = array.buffer;
postMessage(buffer, [buffer]); // с этого момента buffer недоступен transferable objects (Chrome, Firefox)
UI Worker JS загрузка данных UI UI
UI Worker JS браузер не залипает, данные пересылаются как ArrayBuffer
загрузка данных UI UI
function addNumbers(a, b) { 'use asm'; a = a |
0; // int b = +b; // double return +(a + b); // double } светлое будущее — asm.js
светлое будущее — asm.js •оптимизация узких мест вычислений
светлое будущее — asm.js •оптимизация узких мест вычислений •пока только
в FF Nightly
светлое будущее — asm.js •оптимизация узких мест вычислений •пока только
в FF Nightly •обратная совместимость!
технологии рендеринга SVG, Canvas, HTML, WebGL
SVG •быстрые нативные события для интерактивности
SVG •быстрые нативные события для интерактивности •легко обновлять отдельные объекты
SVG •быстрые нативные события для интерактивности •легко обновлять отдельные объекты
•тормозит страницу (при большом кол-ве объектов)
Canvas •после отрисовки не влияет на отзывчивость страницы
Canvas •после отрисовки не влияет на отзывчивость страницы •повторяющиеся части
можно отрисовать один раз и раскопировать
Canvas •после отрисовки не влияет на отзывчивость страницы •повторяющиеся части
можно отрисовать один раз и раскопировать •можно попиксельно рисовать/ обрабатывать в Worker
Canvas •дорого перерисовывать при каждом обновлении
Canvas •дорого перерисовывать при каждом обновлении •очень сложно с реализацией
интерактивности
WebGL •основной способ для 3D- визуализаций
WebGL •основной способ для 3D- визуализаций •очень быстро в 2D,
если нужно отрисовать много спрайтов
WebGL •основной способ для 3D- визуализаций •очень быстро в 2D,
если нужно отрисовать много спрайтов •в остальных случаях преимущество в скорости перед Canvas-2D спорно
WebGL •основной способ для 3D- визуализаций •очень быстро в 2D,
если нужно отрисовать много спрайтов •в остальных случаях преимущество в скорости перед Canvas-2D спорно •API намного сложнее и неудобнее
WebGL •основной способ для 3D- визуализаций •очень быстро в 2D,
если нужно отрисовать много спрайтов •в остальных случаях преимущество в скорости перед Canvas-2D спорно •API намного сложнее и неудобнее •поддержки в IE и iOS не ожидается
HTML •удобно использовать для текста и элементов интерфейса
HTML •удобно использовать для текста и элементов интерфейса •очень удобно
анимировать с помощью CSS Transitions
HTML •удобно использовать для текста и элементов интерфейса •очень удобно
анимировать с помощью CSS Transitions •тормозит страницу при большом кол-ве объектов
производительность Canvas
частичная перерисовка
частичная перерисовка
частичная перерисовка
function drawStar(x, y) { ... // нарисовать звезду в x,
y } drawStar(10, 20); drawStar(50, 70); ...
function drawStar() { var canvas = document.createElement('canvas'); ... // нарисовать
звезду return canvas; } var star = drawStar(); ctx.drawImage(star, 10, 20); ctx.drawImage(star, 50, 70); ... копирование
группируйте отрисовку по стилям, минимизируйте stroke/fill
function drawLine(x1, x2, y1, y2) { ctx.strokeStyle = 'red'; ctx.beginPath();
ctx.moveTo(x1, y1); ctx.lineTo(x2, y2); ctx.stroke(); } drawLine(10, 20, 30, 40); drawLine(200, 10, 0, 50); drawLine(30, 40, 70, 0);
function drawLine(x1, x2, y1, y2) { ctx.beginPath(); ctx.moveTo(x1, y1); ctx.lineTo(x2,
y2); } ctx.strokeStyle = 'red'; drawLine(10, 20, 30, 40); drawLine(200, 10, 0, 50); drawLine(30, 40, 70, 0); ctx.stroke();
var data = ctx.getImageData(0, 0, width, height).data; worker.postMessage(data, [data]); ...
worker.onmessage = function (e) { var imageData = ctx.createImageData(width, height); imageData.data.set(e.data); ctx.putImageData(imageData, 0, 0); } Canvas + Worker
var pixels = new Uint8ClampedArray( width * height); function drawPixel(x,
y, r, g, b, a) { var i = 4 * (256 * y + x); pixels[i] = r; pixels[i + 1] = g; pixels[i + 2] = b; pixels[i + 3] = a; } ... postMessage(pixels.buffer, [pixels.buffer]); рисование пикселей в Worker
var pixels = new Uint8ClampedArray(data); for (var x = 0;
x < width; x++) { for (var y = 0; y < height; y++) { var i = 4 * (256 * y + x); pixels[i] = 2 * pixels[i]; pixels[i + 1] = 2 * pixels[i + 1]; pixels[i + 2] = 2 * pixels[i + 2]; } } ... postMessage(pixels.buffer, [pixels.buffer]); обработка пикселей в Worker
UTFGrid
UTFGrid •65535 разных символов
UTFGrid •65535 разных символов •каждый символ — 4х4 пикселя
UTFGrid •65535 разных символов •каждый символ — 4х4 пикселя •сетка
+ маппинг
UTFGrid •65535 разных символов •каждый символ — 4х4 пикселя •сетка
+ маппинг •в среднем 1-3 КБ на тайл 256х256
Спасибо! Вопросы? Владимир Агафонкин
[email protected]