Web应用的类型以及它们各自的理想实现/交付方式

本文翻译自Jason Miller的博客文章Application Holotypes

分析现实世界中的应用程序的特征很困难。我们经常看到有人对应用程序做出概括,既有漫不经心的,也有经过数据统计的:“单页应用程序比多页应用程序慢”或“TTI 低的应用程序加载速度更快”。然而,这些概括对于我们关心的应用程序的性能和架构特征都不太准确。我认为主要决定因素之一是产品的功能和设计约束,根据应用程序的功能和约束对其进行分类可以为每个应用程序面临的问题提供更有针对性和更有效的解决方案。

构建一组类别,以便有效地将应用程序分组是一种挑战:很难预测所有可能的组,每个组设定的边界都是主观的,可能会随着时间的推移而改变。此外,像这样的抽象分组可能很难推理或可视化。例如,我们可以向“胖客户端、以页面为中心的富媒体应用程序,具有离线浏览和用户生成内容的功能”的开发人员推荐哪些性能优化技术?如果转而问的是我们可以向“类似Instagram的应用程序”的开发人员推荐什么,这种讨论就更具体化、更容易进行了。

为了建立这种讨论框架,我们可以构建一个典型应用程序列表。这些应用程序既可以代表当今的网络,也可以基于我们预见到的开发人员为响应未来趋势和平台计划而做出的改变。为了方便起见,代表Web的长尾历史和遗留内容的部分典型应用程序更加通用,而代表当前和即将推出的部分典型应用程序可以更狭窄地限定讨论范围,以便提供更具体的建议。

每个典型应用程序都附有粗略的类别名称、额外的真实案例以及定义其架构的特征和约束。我还根据其架构提供了理想的部署实施和交付技术。

社交网络应用

  • 典型:Facebook
  • 其他案例:LinkedIn、Reddit、Google+
  • 特点:多面性、子应用程序、无限滚动内容、用户生成内容、实时更新、通知
  • 限制:过深的会话深度、大规模、实时更新、嵌入式内容的资源争用、嵌套应用程序、SEO
  • 理想实现方式:具有外壳和登录页面预渲染的单页应用程序。
  • 理想交付方式:独立显示模式下的 PWA。或TWA。

社交媒体应用

  • 典型:Instagram
  • 其他案例:Youtube、Twitter
  • 特点:富媒体、无限滚动内容、用户生成内容、实时更新、通知、可嵌入性、嵌入式内容
  • 限制:扩展会话深度、实时更新、嵌入式内容的资源争用、不间断媒体播放、SEO
  • 理想实现方式:具有应用程序外壳预渲染和缓存的单页应用程序。
  • 理想交付方式:独立显示模式下的 PWA。

店面应用

  • 典型:亚马逊
  • 其他案例:百思买、新蛋、Shopify(基于商店)
  • 特点:搜索、支付、可发现性、过滤和排序 限制:浅到中等会话深度、小交互、高购物车/结帐流失率、SEO
  • 理想实现方式:具有 CSR/SPA 接管或 turbolinks 样式转换的服务器呈现站点。
  • 理想交付方式:默认显示模式下的 PWA。

内容网站

  • 典型:CNN
  • 其他案例:FT、BBC、BuzzFeed、Engadget、Salon、Smashing Magazine、The Onion
  • 特点:可发现性、富媒体、嵌入式内容 限制:浅会话深度(~1)、广告和多变量测试的资源争用、SEO
  • 理想实现方式:具有轮播图风格过渡效果的服务器端渲染网站。
  • 理想交付方式:默认显示模式下的 PWA。

PIM应用

  • 典型:Gmail
  • 其他案例:Google 日历、Outlook.com、Fastmail
  • 特点:胖客户端、无限列表、嵌入式内容、富文本编辑、清理、MDI、存储、离线和同步、通知
  • 限制:延长会话长度、敏感且大量不可缓存的数据、高安全风险、经常离线
  • 理想实现方式:具有应用程序外壳缓存的单页应用程序。
  • 理想交付方式:独立显示模式下的 PWA。

生产力工具

  • 典型:Google Docs
  • 其他案例:Office.com、Zoho、Dropbox、Box
  • 特点:胖客户端、富文本编辑、离线和同步、文件系统、剪贴板、存储、图像处理、嵌入式内容
  • 限制:延长会话长度和多个并发会话有利于客户端实现。
  • 理想实现方式:单页应用程序。考虑应用程序前端缓存。
  • 理想交付方式:独立显示模式下的 PWA。

媒体播放器

  • 典型:Spotify
  • 其他案例:Youtube Music、Google Play Music、Tidal、Soundcloud、Pandora、Deezer
  • 特点:富媒体、胖客户端、无限滚动内容、过滤和排序、通知、操作系统集成、离线、可嵌入性
  • 限制:延长会话长度,用户浏览其他页面时必须继续播放。
  • 理想实现方式:具有应用程序前端预渲染和缓存的单页应用程序。服务器渲染 <head> 以SEO友好。
  • 理想交付方式:独立显示模式下的 PWA。

图形编辑器

  • 典型:Figma
  • 其他案例:AutoCAD、Tinkercad、Photopea、Polarr
  • 特点:3D 渲染和 GPU、图像处理、全屏和指针捕获、MDI、存储、离线、文件系统、线程、wasm
  • 限制:会话长度长、对输入和渲染延迟的敏感性、大对象/文件
  • 理想实现方式:单页应用。将更轻量的浏览 UI 与编辑器分开。
  • 理想交付方式:独立显示模式下的 PWA。

媒体编辑器

  • 典型:Soundtrap
  • 其他案例:Looplabs
  • 特点:音频处理、设备集成(midi、usb)、存储、离线、文件系统、线程、wasm
  • 限制:长会话长度、低延迟 DSP、低延迟媒体录制和播放、大文件大小/IO
  • 理想实现方式:单页应用。将更轻量的浏览 UI 与编辑器分开。
  • 理想交付方式:独立显示模式下的 PWA。

工程工具

  • 典型:Codesandbox
  • 其他案例:Codepen、Jupyter Notebook、RStudio、StackBlitz
  • 特点:胖客户端、MDI、存储、离线、文件系统、线程、嵌入式内容
  • 限制:极长的会话长度、低延迟文本输入、大内存占用、自定义输入处理和文本渲染、预览内容的安全性
  • 理想实现方式:单页应用。考虑将浏览 UI 与编辑器分开。
  • 理想交付方式:独立显示模式下的 PWA。

沉浸式/AAA游戏

  • 典型:Stadia
  • 其他案例:Heraclos、Duelyst、OUIGO 特点:3D 渲染和 GPU、P2P、音频处理、全屏和指针捕获、存储、离线、文件系统、线程、设备集成(游戏手柄)、wasm
  • 限制:会话长度长(高度交互)、沉浸感、对输入和渲染延迟极其敏感、需要一致或分步的 FPS、极端的资产大小
  • 理想实现方式:单页应用
  • 理想交付:全屏显示模式下的 PWA。

休闲游戏

  • 典型:Robostorm
  • 其他案例:Tank Off、War Brokers、GoreScript、Air Wars、”.io games”
  • 特点:2D 和 3D 渲染和 GPU、P2P、音频处理、存储、离线、可嵌入性
  • 限制:会话长度长、对输入和渲染延迟敏感、需要一致/分步 FPS
  • 理想实现方式:单页应用
  • 理想交付方式:嵌入另一个站点,或全屏显示模式下的 PWA。

托管钱包和非托管钱包

钱包按是否被托管可以分为:

  • 非托管钱包(non-custodial wallets)(也称为自托管钱包),意味着用户始终可以控制自己的私钥和资金。非托管钱包有trust、MetaMask、Coinbase等。
  • 托管钱包(custodial wallets),意味着用户的私钥和资金都放在钱包软件公司的服务器里。托管钱包有Binance Wallet等。

托管钱包就像将你的贵重物品存放在仓库中,而非托管钱包就像将它们存放在家中的保险箱中。因此,托管钱包需要较少的个人责任,但受第三方支配,而非托管钱包由你自己完全控制,但也意味着你承担全部责任,确保你的物品安全并保管你的密码和密钥。

SSG就是页面静态化技术

静态站点生成 (Static-Site Generation,缩写为 SSG),也被称为预渲染,是另一种流行的构建快速网站的技术。如果用服务端渲染一个页面所需的数据对每个用户来说都是相同的,那么我们可以只渲染一次,提前在构建过程中完成,而不是每次请求进来都重新渲染页面。预渲染的页面生成后作为静态 HTML 文件被服务器托管。

SSG就是页面静态化技术,是门户网站、WordPress博客常用的Web优化手段之一。

SSR类似于PHP程序,每有一个HTTP请求到来就会被运行一次,渲染输出HTML字符串给前端。如果网站的内容没有实时性,我们可以把渲染输出的HTML字符串缓存一段时间,当有HTTP请求到来时,直接返回缓存给前端,而不是实时渲染输出。这就是页面静态化技术的核心思想。

参考

服务端渲染 (SSR)

Does Stopping a Docker Container Equate to Shutting Down a VirtualBox VM?

To clarify, stopping a running Docker container is not the same as pausing it. Stopping a container terminates the processes inside the container and releases associated resources (like CPU and memory), but the container’s filesystem and state remain intact, similar to how VirtualBox retains hard disk data after shutting down a VM. The container’s data stays unless explicitly deleted (using rm).

The Difference Between Stop and Pause:

  • Stop: Completely terminates all processes inside the container, similar to turning off the power. When the container is restarted, it begins from its configured entry process.
  • Pause: Freezes all processes inside the container, keeping their state intact, but without releasing resources. A paused container can be resumed, much like the Suspend feature in VirtualBox.

停止(注意不是暂停)运行docker容器是不是相当于VirtualBox的虚拟机关机?

具体来说,停止运行中的 Docker 容器会终止容器内部的进程,释放相关的资源(如 CPU 和内存),但容器的文件系统和状态会被保留,就像 VirtualBox 虚拟机关机后硬盘数据依然保存一样,除非进行删除(rm)操作。

停止和暂停的区别在于:

  • 停止(stop):完全终止容器内的进程,相当于电源关闭,容器在下次启动后会重新开始其配置的入口进程。
  • 暂停(pause):冻结容器内的所有进程,进程状态保持,但并不释放资源,暂停的容器可以恢复继续运行,类似于 VirtualBox 的暂停(Suspend)功能。

Methods for Website Internationalization (Providing Multi-language Content)

For websites that use server-side templating engines (e.g., Laravel’s Blade) to render HTML documents, the approach to localization (internationalization) is as follows: For common pages like login and registration, you can use the __() function to translate text based on the lang= query parameter passed from the frontend. For other pages, the language-specific views should be placed in directories like resources/views/{language_code} (for Laravel) or public/{language_code} (where “public” refers to the web root directory).

For fully decoupled applications, such as those where the frontend is developed using MVVM frameworks like Vue.js or React.js for SPAs, and the backend is built with Laravel to provide APIs, localization methods differ. In this case, you should add language-specific subdomains at the domain level, such as cn.vuejs.org for the Chinese Vue.js site, ja.vuejs.org for the Japanese site, with the default vuejs.org serving the English version. Additionally, the database should support language-specific partitioning, where each language code has its own set of tables or databases.

网站国际化(提供多国语言内容)的方法

对于前端使用服务器端模板引擎(例如Laravel的blade)渲染输出HTML文档的开发方式,本地化(国际化)的方法是,通用的页面,例如登陆注册,可以使用__()函数根据前端传过来的lang=语言代码查询参数翻译文本。其他不同用的页面应该放在resources/views/语言代码目录(对于Laravel)或public/语言代码目录(public代表web根目录)下。

对于前后端完全分离的应用程序,例如前端使用Vue.js、React.js等MVVM框架开发SPA,服务器端使用Laravel开发并提供API给前端调用,本地化(国际化)的方法是,在域名层面添加语言代码子域,例如cn.vuejs.org、ja.vuejs.org分别显示中文Vue.js官网和日文Vue.js官网,主站vuejs.org默认英文Vue.js官网。 数据库也应该根据语言代码分库分表。

Laravel’s DB Facade Doesn’t Trigger Eloquent ORM Model Events

When using the DB facade to perform database operations in Laravel, Eloquent ORM model events like saving, saved, etc., are not triggered.

If you want to avoid triggering model events in event listeners or queued tasks, aside from using Eloquent’s saveQuietly, deleteQuietly, and similar methods, you can directly use the DB facade to execute database operations.

This approach allows you to bypass Eloquent’s event handling when necessary.

Laravel使用DB façade执行数据库操作不会触发Eloquent ORM模型事件

使用DB façade执行数据库操作不会触发saving、saved等Eloquent ORM模型事件。当我们不想在事件监听器或队列任务中触发Eloquent ORM模型事件时,除了使用Eloquent ORM模型的saveQuietly、deleteQuietly等方法之外,还可以直接使用DB façade执行数据库操作。

在Eloquent ORM模型事件监听器和队列任务中,要避免使用Eloquent模型增删改查方法,例如create、update、save等。否则会陷入调用死循环 —— 模型事件监听器分发队列任务,队列任务触发模型事件,模型事件监听器再次分发队列任务,队列任务再次触发模型事件……死循环了。

Modify and Persist Model Instances in Laravel Using the saved Event, Not the saving Event

In Laravel, don’t call the save method on a model instance inside the saving event listener.

If you need to modify a model’s field and persist it within a model event listener, make sure you’re using the saved event, not the saving event. This is particularly important when your event listeners are queued for asynchronous execution.

The saving event occurs before the model is persisted to the database. If you try to modify a field and call save() within this listener, it won’t actually persist to the database, especially when the listener is queued for async execution. For example, modifying the slug field might not actually update in the database.

Instead, use the saved event listener and call saveQuietly to persist the changes, as shown in the example:

static::saved(queueable(function (Topic $topic) {
    // If the slug is empty, translate the title into a slug
    if (!$topic->slug) {
        $topic->slug = app(SlugTranslateHandler::class)->translate($topic->title);
        $topic->saveQuietly();
    }
}));

By using the saved event and saveQuietly, you ensure that your changes are made after the model is successfully persisted, avoiding any issues with asynchronous queue execution.