Оптимизация больших приложений

Игорь Шеко

Что такое большое приложение?

Что обычно тормозит?

Серверные запросы

Загрузка бандла

Рендер компонентов

Что-то закостылили на фронте

Пcихология

Главные советы

Взять и переписать

Прикрутить SSR

Давайте оптимизировать под

самые

слабые

телефоны

Только для новых лидов:

"Давайте заменим команду"

Всем спасибо

Игорь Шеко

Lead Front end developer @ Voximplant

</ВРЕДНЫЕ СОВЕТЫ>

Пcихология

Технологический долг

Рендер компонентов

Загрузка бандла

Серверные запросы

С

З

Р

Д

П

С

З

Р

Д

П

С

З

Р

Д

П

Команда

Клиенты

Оптимизация запросов

Поговорите с вашим бекендом, если:

- Запросы выполняются медленно (>100мс)

- У вас все еще http(s) v1.1

- Ответ не сжат или сжат неверно

- Кеширование запросов не применяется

Почему такое возможно

- паралельная разработка

- разные команды

- оптимизация клиента

Как это решать

- исправить код

- кеширование на стороне браузера

- кеширование на фронте

- что угодно еще

export default class mAaxios{
  post(url,params){
    const cache = this.checkCache(url,params);
    if(cache) return cache;
    //...
    return this.writeCache(url,params,data);
  }
  
  put(){
    //...
  }
  get(){
    //...
  }
  
  private cache:Map<string,any>;
  private blackList:string;
  
  constructor(){
    this.axios = //...
    
    setInterval(()=>this.heartbeat,1000);
  }
  
}

PWA

- Очень просто, решает все проблемы с кешированием (почти)

- Но нужен специалист

- И нужно как-то интегрировать...

Оптимизация архитектуры

- Продукт создавался стихийно

- Вы новый лид на проекте

- Не понятно с чего начать

Вынесите в Vuex все что можно

- Эта запись есть в Vuex?

- Почему ее туда не занесли?

Уберите из Vuex все что не нужно

const store = {
  user:{
    id: 0,
    name: "Ivan",
    //....
    avatar: ""
  }
}
/**
 * Reduce the code which written in Vue.js for getting the getters
 * @param {String} [namespace] - Module's namespace
 * @param {Object|Array} getters
 * @return {Object}
 */
export const mapGetters = normalizeNamespace((namespace, getters) => {
  const res = {}
  normalizeMap(getters).forEach(({ key, val }) => {
    // The namespace has been mutated by normalizeNamespace
    val = namespace + val
    res[key] = function mappedGetter () {
      if (namespace && !getModuleByNamespace(this.$store, 'mapGetters', namespace)) {
        return
      }
      if (process.env.NODE_ENV !== 'production' && !(val in this.$store.getters)) {
        console.error(`[vuex] unknown getter: ${val}`)
        return
      }
      return this.$store.getters[val]
    }
    // mark vuex getter for devtools
    res[key].vuex = true
  })
  return res
})

То что у вас Vue - не значит что все должно быть в компонентах

const state = Vue.observable({ count: 0 })
const Demo = {
  render(h) {
    return h('button', {
      on: { click: () => { state.count++ }}
    }, `счётчик — ${state.count}`)
  }
}

Сервисы

app.js

POST img.png

GET userData

El1.vue

POST img.png

GET userData

El3.vue

El40.vue

El2.vue

El1.vue

El3.vue

El40.vue

El2.vue

Vuex

El1.vue

getUserMedia

mediaRecorder

El1.vue

El3.vue

El40.vue

El2.vue

Сервис

request

response

Единственная ответственность

Отчуждаем

Легко паралелится в WebWorker

Оптимизация бандла

Webpack Bundle Analyzer

Импортозамещение

export default class PubSub{
  private listeners:Map<string,EventListener[]> = new Map();
  
  addEventListener(event:string,listener:EventListener){
    const listeners = this.listenerts.get(event)||[];
    this.listenerts.set([...listeners, listener]);
  }
  
  removeEventListener(event:string,listener?:EventListener){
    if(!listener)
      this.listenerts.remove(event);
    this.listenerts.set(this.listeners.filter(item=>item!==listener]);
  }
  
  dispatch(event:string, data?:any){
    const listeners = this.listenerts.get(event)||[];
    listeners.forEach(item=>item.handleEvent(data));
  }
}
public static debounce(callback: (...args: any[]) => void, 
  wait: number, 
  immediate: boolean = false) {
    let timeout = null;
    return (...args: any[]) => {
      const next = () => callback(...args);
      clearTimeout(timeout);
      timeout = setTimeout(next, wait);
      if (immediate) {
        next();
      }
    };
  };

Оптимизация рендера

Реактивность - ваш враг *

* на момент оптимизации

Как работает реактивность.

Сейчас.
В коде.

 let childOb = !shallow && observe(val)
  Object.defineProperty(obj, key, {
    enumerable: true,
    configurable: true,
    get: function reactiveGetter () {
      const value = getter ? getter.call(obj) : val
      if (Dep.target) {
        dep.depend()
        if (childOb) {
          childOb.dep.depend()
          if (Array.isArray(value)) {
            dependArray(value)
          }
        }
      }
      return value
    },
    set: function reactiveSetter (newVal) {
      const value = getter ? getter.call(obj) : val
      if (newVal === value || (newVal !== newVal && value !== value)) {
        return
      }
      if (process.env.NODE_ENV !== 'production' && customSetter) {
        customSetter()
      }
      if (getter && !setter) return
      if (setter) {
        setter.call(obj, newVal)
      } else {
        val = newVal
      }
      childOb = !shallow && observe(newVal)
      dep.notify()
    }
  })

Тоже самое, но с Proxy.

data = new Proxy(data_without_proxy, {  // Override data to have a proxy in the middle
  get(obj, key) {
    deps.get(key).depend(); // <-- Remember the target we're running
    return obj[key]; // call original data
  },

  set(obj, key, newVal) {
    obj[key] = newVal; // Set original data to new value
    deps.get(key).notify(); // <-- Re-run stored functions
    return true;
  }
});

Evan You on Proxies

2.5

3.x

Рендеринг 3000 state-full компонентов

Асинхронные компоненты

Vue.component('MyComponent', function (resolve, reject) {
  setTimeout(function () {
    // Pass the component definition to the resolve callback
    resolve({
      template: '<div>I am async!</div>'
    })
  }, 1000)
})


const AsyncComponent = () => ({
  // The component to load (should be a Promise)
  component: import('./MyComponent.vue'),
  // A component to use while the async component is loading
  loading: LoadingComponent,
  // A component to use if the load fails
  error: ErrorComponent,
  // Delay before showing the loading component. Default: 200ms.
  delay: 200,
  // The error component will be displayed if a timeout is
  // provided and exceeded. Default: Infinity.
  timeout: 3000
})

const Foo = () => import('./AsyncComponent.vue')


const router = new VueRouter({
  routes: [
    { path: '/foo', component: Foo }
  ]
})

const Bar = () => import(/* webpackChunkName: "group-foo" */ './Bar.vue')
const Baz = () => import(/* webpackChunkName: "group-foo" */ './Baz.vue')

Заключение.

Оптимизация - комплексный процесс

Никогда не поздно оптимизировать

Неверная оптимизация - лучший способ потерять команду

Но иногда рано

И когда вы наконец отимизировались, напишите, наконец, документацию.

+ Таск трекер

+ Историю оптимизаций

Всем спасибо

Игорь Шеко

Front-end team lead @ Voximplant

Made with Slides.com