Các phương pháp hay nhất về phông chữ

Tối ưu hoá phông chữ web cho Các chỉ số quan trọng về trang web.

Katie Hempenius
Katie Hempenius

Bài viết này thảo luận về các phương pháp hay nhất về hiệu suất cho phông chữ. Phông chữ web có nhiều cách tác động đến hiệu suất:

Bài viết này được chia thành ba phần: tải phông chữ, phân phối phông chữhiển thị phông chữ. Mỗi phần giải thích cách hoạt động của một khía cạnh cụ thể trong vòng đời phông chữ và đưa ra các phương pháp hay nhất tương ứng.

Đang tải phông chữ

Phông chữ thường là tài nguyên quan trọng vì nếu không có phông chữ, người dùng có thể không xem được nội dung trang. Do đó, các phương pháp hay nhất để tải phông chữ thường tập trung vào việc đảm bảo phông chữ được tải càng sớm càng tốt. Bạn cần đặc biệt chú ý đến phông chữ được tải từ các trang web của bên thứ ba vì việc tải các tệp phông chữ này xuống đòi hỏi phải thiết lập kết nối riêng.

Nếu bạn không chắc phông chữ của trang có được yêu cầu kịp thời hay không, hãy kiểm tra thẻ Timing (Thời gian) trong bảng điều khiển Network (Mạng) trong Chrome DevTools để biết thêm thông tin.

Ảnh chụp màn hình thẻ Timing (Thời gian) trong DevTools

Tìm hiểu về @font-face

Trước khi tìm hiểu các phương pháp hay nhất để tải phông chữ, bạn cần hiểu rõ cách hoạt động của @font-face và tác động của việc này đến việc tải phông chữ.

Nội dung khai báo @font-face là một phần thiết yếu khi làm việc với bất kỳ phông chữ web nào. Ở mức tối thiểu, tệp này sẽ khai báo tên sẽ được dùng để tham chiếu đến phông chữ và cho biết vị trí của tệp phông chữ tương ứng.

@font-face {
  font-family: "Open Sans";
  src: url("/fonts/OpenSans-Regular-webfont.woff2") format("woff2");
}

Có một quan niệm sai lầm phổ biến là phông chữ được yêu cầu khi gặp nội dung khai báo @font-face. Điều này không đúng. Bản thân nội dung khai báo @font-face không kích hoạt quá trình tải phông chữ xuống. Thay vào đó, phông chữ chỉ được tải xuống nếu được tham chiếu theo kiểu được sử dụng trên trang. Ví dụ: như sau:

@font-face {
  font-family: "Open Sans";
  src: url("/fonts/OpenSans-Regular-webfont.woff2") format("woff2");
}

h1 {
  font-family: "Open Sans"
}

Nói cách khác, trong ví dụ trên, Open Sans sẽ chỉ được tải xuống nếu trang có chứa phần tử <h1>.

Do đó, khi nghĩ đến việc tối ưu hoá phông chữ, bạn cần cân nhắc các tệp kiểu như chính các tệp phông chữ. Việc thay đổi nội dung hoặc cách phân phối của các tệp kiểu có thể ảnh hưởng đáng kể đến thời điểm phông chữ đến. Tương tự, việc xoá CSS không dùng đến và tách các tệp biểu định kiểu có thể làm giảm số lượng phông chữ mà một trang tải.

Khai báo phông chữ cùng dòng

Hầu hết các trang web sẽ được hưởng lợi rất nhiều từ việc đưa các nội dung khai báo phông chữ và các kiểu quan trọng khác vào cùng dòng trong <head> của tài liệu chính thay vì đưa các nội dung đó vào một tệp định kiểu bên ngoài. Điều này cho phép trình duyệt khám phá các nội dung khai báo phông chữ sớm hơn vì trình duyệt không cần phải đợi tải tệp định kiểu bên ngoài xuống.

<head>
  <style>
    @font-face {
        font-family: "Open Sans";
        src: url("/fonts/OpenSans-Regular-webfont.woff2") format("woff2");
    }

    body {
        font-family: "Open Sans";
    }

    ...etc.

  </style>
</head>

Việc nội tuyến CSS quan trọng có thể là một kỹ thuật nâng cao hơn mà không phải trang web nào cũng có thể thực hiện được. Lợi ích về hiệu suất là rõ ràng, nhưng bạn cần thêm các quy trình và công cụ xây dựng để đảm bảo CSS cần thiết (và lý tưởng là chỉ CSS quan trọng) được nội tuyến chính xác và mọi CSS bổ sung được phân phối theo cách không chặn kết xuất.

Kết nối trước với các nguồn gốc quan trọng của bên thứ ba

Nếu trang web của bạn tải phông chữ từ một trang web của bên thứ ba, thì bạn nên sử dụng preconnect gợi ý về tài nguyên để thiết lập(các) kết nối sớm với nguồn gốc của bên thứ ba. Bạn nên đặt gợi ý tài nguyên trong <head> của tài liệu. Gợi ý tài nguyên bên dưới thiết lập một kết nối để tải tệp kiểu phông chữ.

<head>
  <link rel="preconnect" href="https://fonts.com">
</head>

Để kết nối trước kết nối dùng để tải tệp phông chữ xuống, hãy thêm một gợi ý về tài nguyên preconnect riêng biệt sử dụng thuộc tính crossorigin. Không giống như biểu định kiểu, tệp phông chữ phải được gửi qua kết nối CORS.

<head>
  <link rel="preconnect" href="https://fonts.com">
  <link rel="preconnect" href="https://fonts.com" crossorigin>
</head>

Khi sử dụng gợi ý tài nguyên preconnect, hãy lưu ý rằng trình cung cấp phông chữ có thể phân phát các kiểu phông chữ và phông chữ từ các nguồn gốc riêng biệt. Ví dụ: đây là cách sử dụng gợi ý tài nguyên preconnect cho Google Fonts.

<head>
  <link rel="preconnect" href="https://fonts.googleapis.com">
  <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
</head>

Hãy thận trọng khi sử dụng preload để tải phông chữ

Mặc dù preload rất hiệu quả trong việc giúp người dùng dễ dàng tìm thấy phông chữ ở giai đoạn đầu quá trình tải trang, nhưng điều này sẽ khiến người dùng phải lấy đi tài nguyên trình duyệt khỏi việc tải các tài nguyên khác.

Nội tuyến khai báo phông chữ và điều chỉnh biểu định kiểu có thể là một phương pháp hiệu quả hơn. Những điều chỉnh này giúp giải quyết nguyên nhân gốc rễ của phông chữ được phát hiện muộn hơn thay vì chỉ cung cấp giải pháp.

Ngoài ra, bạn cũng nên sử dụng preload làm chiến lược tải phông chữ một cách thận trọng vì chiến lược này bỏ qua một số chiến lược đàm phán nội dung tích hợp sẵn của trình duyệt. Ví dụ: preload bỏ qua các nội dung khai báo unicode-range và nếu được sử dụng một cách thận trọng, bạn chỉ nên dùng preload để tải một định dạng phông chữ duy nhất.

Tuy nhiên, khi sử dụng các tệp định kiểu bên ngoài, việc tải trước các phông chữ quan trọng nhất có thể rất hiệu quả vì trình duyệt sẽ không phát hiện được liệu phông chữ có cần thiết hay không cho đến sau này.

Phân phối phông chữ

Việc phân phối phông chữ nhanh hơn sẽ giúp hiển thị văn bản nhanh hơn. Ngoài ra, nếu phông chữ được phân phối đủ sớm, điều này có thể giúp loại bỏ sự thay đổi bố cục do hoán đổi phông chữ.

Sử dụng phông chữ tự lưu trữ

Trên lý thuyết, việc sử dụng phông chữ tự lưu trữ sẽ mang lại hiệu suất tốt hơn vì loại bỏ việc thiết lập kết nối của bên thứ ba. Tuy nhiên, trong thực tế, sự khác biệt về hiệu suất giữa hai tuỳ chọn này không rõ ràng: ví dụ: Web Almanac nhận thấy rằng các trang web sử dụng phông chữ của bên thứ ba có tốc độ kết xuất nhanh hơn so với các trang web sử dụng phông chữ của bên thứ nhất.

Nếu bạn đang cân nhắc sử dụng phông chữ tự lưu trữ, hãy xác nhận rằng trang web của bạn đang sử dụng Mạng phân phối nội dung (CDN)HTTP/2. Nếu không sử dụng những công nghệ này, thì phông chữ tự lưu trữ sẽ ít có khả năng mang lại hiệu suất tốt hơn. Để biết thêm thông tin, hãy xem phần Mạng phân phối nội dung.

Nếu sử dụng phông chữ tự lưu trữ, bạn cũng nên áp dụng một số tính năng tối ưu hoá tệp phông chữ mà nhà cung cấp phông chữ của bên thứ ba thường cung cấp tự động, ví dụ: chế độ cài đặt phụ phông chữ và nén WOFF2. Mức độ nỗ lực cần thiết để áp dụng các tối ưu hoá này sẽ tuỳ thuộc một chút vào các ngôn ngữ mà trang web của bạn hỗ trợ. Cụ thể, hãy lưu ý rằng việc tối ưu hoá phông chữ cho ngôn ngữ CJK có thể đặc biệt khó khăn.

Sử dụng WOFF2

Trong số các phông chữ hiện đại, WOFF2 là phông chữ mới nhất, có hỗ trợ trình duyệt rộng nhất và mang lại khả năng nén tốt nhất. Vì sử dụng Brotli, WOFF2 nén tốt hơn 30% so với WOFF, dẫn đến việc tải ít dữ liệu hơn và do đó hiệu suất nhanh hơn.

Do tính năng hỗ trợ trình duyệt, các chuyên gia hiện chỉ đề xuất sử dụng WOFF2:

Thực tế, chúng tôi cho rằng đã đến lúc tuyên bố: Chỉ sử dụng WOFF2 và quên mọi thứ khác.

Việc này sẽ giúp đơn giản hoá CSS và quy trình làm việc của bạn một cách đáng kể, đồng thời ngăn chặn mọi trường hợp tải phông chữ không chính xác hoặc tải hai lần do nhầm lẫn. WOFF2 hiện được hỗ trợ ở mọi nơi. Vì vậy, trừ phi bạn cần hỗ trợ các trình duyệt thực sự cổ, hãy chỉ sử dụng WOFF2. Nếu không thể, hãy cân nhắc việc không phân phát phông chữ web nào cho những trình duyệt cũ đó. Điều này sẽ không thành vấn đề nếu bạn có một chiến lược dự phòng mạnh mẽ. Khách truy cập trên các trình duyệt cũ sẽ chỉ thấy phông chữ dự phòng của bạn.

Bram Stein, trong cuốn Web Almanac 2022

Phông chữ con

Tệp phông chữ thường bao gồm một lượng lớn ký tự cho tất cả các ký tự mà phông chữ đó hỗ trợ. Tuy nhiên, bạn có thể không cần tất cả các ký tự trên trang và có thể giảm kích thước tệp phông chữ bằng cách tạo tập hợp con phông chữ.

Phần mô tả unicode-range trong nội dung khai báo @font-face thông báo cho trình duyệt biết những ký tự nào có thể được dùng cho phông chữ đó.

@font-face {
    font-family: "Open Sans";
    src: url("/fonts/OpenSans-Regular-webfont.woff2") format("woff2");
    unicode-range: U+0025-00FF;
}

Một tệp phông chữ sẽ được tải xuống nếu trang chứa một hoặc nhiều ký tự khớp với phạm vi Unicode. unicode-range thường được dùng để phân phát các tệp phông chữ khác nhau tuỳ thuộc vào ngôn ngữ mà nội dung trang sử dụng.

unicode-range thường được dùng kết hợp với kỹ thuật tạo tập hợp con. Phông chữ con bao gồm một phần nhỏ hơn các ký tự có trong tệp phông chữ gốc. Ví dụ: thay vì phân phát tất cả ký tự cho tất cả người dùng, một trang web có thể tạo các phông chữ con riêng biệt cho ký tự La-tinh và ký tự Cyrillic. Số lượng ký tự trên mỗi phông chữ rất khác nhau: phông chữ Latinh thường có khoảng 100 đến 1.000 ký tự trên mỗi phông chữ; phông chữ CJK có thể có hơn 10.000 ký tự. Việc xoá các ký tự không dùng đến có thể làm giảm đáng kể kích thước tệp của phông chữ.

Một số nhà cung cấp phông chữ có thể tự động cung cấp các phiên bản tệp phông chữ với các tập hợp con khác nhau. Ví dụ: Google Fonts thực hiện việc này theo mặc định:

/* devanagari */
@font-face {
  font-family: 'Poppins';
  font-style: normal;
  font-weight: 400;
  font-display: swap;
  src: url(https://fonts.gstatic.com/s/poppins/v20/pxiEyp8kv8JHgFVrJJbecnFHGPezSQ.woff2) format('woff2');
  unicode-range: U+0900-097F, U+1CD0-1CF6, U+1CF8-1CF9, U+200C-200D, U+20A8, U+20B9, U+25CC, U+A830-A839, U+A8E0-A8FB;
}
/* latin-ext */
@font-face {
  font-family: 'Poppins';
  font-style: normal;
  font-weight: 400;
  font-display: swap;
  src: url(https://fonts.gstatic.com/s/poppins/v20/pxiEyp8kv8JHgFVrJJnecnFHGPezSQ.woff2) format('woff2');
  unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF;
}
/* latin */
@font-face {
  font-family: 'Poppins';
  font-style: normal;
  font-weight: 400;
  font-display: swap;
  src: url(https://fonts.gstatic.com/s/poppins/v20/pxiEyp8kv8JHgFVrJJfecnFHGPc.woff2) format('woff2');
  unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
}

Khi chuyển sang tự lưu trữ, đây là một phương pháp tối ưu hoá dễ bị bỏ qua và dẫn đến các tệp phông chữ lớn hơn trên máy.

Bạn cũng có thể tạo nhóm con phông chữ theo cách thủ công nếu nhà cung cấp phông chữ cho phép việc này, thông qua API (Google Fonts hỗ trợ việc này bằng cách cung cấp tham số text) hoặc bằng cách chỉnh sửa tệp phông chữ theo cách thủ công rồi tự lưu trữ. Các công cụ tạo tập hợp con phông chữ bao gồm phông chữ conglyphanger. Tuy nhiên, hãy kiểm tra giấy phép của phông chữ bạn sử dụng để biết liệu có cho phép tạo tập hợp con và tự lưu trữ hay không.

Sử dụng ít phông chữ web hơn

Phông chữ nhanh nhất để phân phối là phông chữ không được yêu cầu ngay từ đầu. Phông chữ hệ thống và phông chữ biến thể là hai cách có thể giúp giảm số lượng phông chữ web được sử dụng trên trang web của bạn.

Phông chữ hệ thống là phông chữ mặc định mà giao diện người dùng của thiết bị của người dùng sử dụng. Phông chữ hệ thống thường khác nhau tuỳ theo hệ điều hành và phiên bản. Vì phông chữ đã được cài đặt nên bạn không cần tải phông chữ xuống. Phông chữ hệ thống có thể đặc biệt phù hợp với văn bản nội dung.

Để sử dụng phông chữ của hệ thống trong CSS, hãy liệt kê system-ui làm font-family:

font-family: system-ui

Ý tưởng đằng sau phông chữ biến đổi là một phông chữ biến duy nhất có thể được dùng để thay thế cho nhiều tệp phông chữ. Phông chữ biến đổi hoạt động bằng cách xác định kiểu phông chữ "mặc định" và cung cấp "trục" để thao tác với phông chữ. Ví dụ: bạn có thể sử dụng phông chữ biến thiên có trục Weight để triển khai chữ cái mà trước đây cần có phông chữ riêng biệt cho chữ cái sáng, thường, đậm và rất đậm.

Không phải ai cũng được hưởng lợi khi chuyển sang phông chữ biến thể. Phông chữ thay đổi chứa nhiều kiểu, vì vậy, thường có kích thước tệp lớn hơn so với các phông chữ riêng lẻ không biến đổi chỉ chứa một kiểu. Những trang web sẽ thấy sự cải thiện lớn nhất khi sử dụng phông chữ biến thể là những trang web sử dụng (và cần sử dụng) nhiều kiểu và độ đậm phông chữ.

Kết xuất phông chữ

Khi gặp phông chữ web chưa tải, trình duyệt sẽ gặp phải một vấn đề nan giải: có nên tạm hoãn việc hiển thị văn bản cho đến khi phông chữ web đã tải xong hay không? Hay có nên hiển thị văn bản bằng phông chữ dự phòng cho đến khi phông chữ web đến không?

Các trình duyệt xử lý trường hợp này theo cách khác nhau. Theo mặc định, trình duyệt dựa trên Chromium và Firefox sẽ chặn việc hiển thị văn bản trong tối đa 3 giây nếu phông chữ web liên kết chưa tải; Safari sẽ chặn việc hiển thị văn bản vô thời hạn.

Bạn có thể định cấu hình hành vi này bằng cách sử dụng thuộc tính font-display. Lựa chọn này có thể có ý nghĩa đáng kể: font-display có thể ảnh hưởng đến LCP, FCP và độ ổn định của bố cục.

Chọn chiến lược font-display phù hợp

font-display cho trình duyệt biết cách tiếp tục hiển thị văn bản khi phông chữ web được liên kết chưa tải. Giá trị này được xác định cho từng phông chữ.

@font-face {
  font-family: Roboto, Sans-Serif
  src: url(/fonts/roboto.woff) format('woff'),
  font-display: swap;
}

Có 5 giá trị có thể có cho font-display:

Giá trị Khoảng thời gian chặn Khoảng thời gian hoán đổi
Tự động Tuỳ theo trình duyệt Tuỳ theo trình duyệt
Chặn 2-3 giây Vô hạn
Hoán đổi 0 mili giây Vô hạn
Dự phòng 100 mili giây 3 giây
Không bắt buộc 100 mili giây Không có
  • Khoảng thời gian chặn: Khoảng thời gian chặn bắt đầu khi trình duyệt yêu cầu phông chữ web. Trong khoảng thời gian chặn, nếu không có phông chữ web, phông chữ sẽ được hiển thị bằng phông chữ dự phòng vô hình, do đó người dùng sẽ không nhìn thấy văn bản. Nếu không có phông chữ vào cuối khoảng thời gian chặn, phông chữ sẽ được hiển thị bằng phông chữ dự phòng.
  • Thời gian hoán đổi: Giai đoạn hoán đổi đến sau giai đoạn chặn. Nếu phông chữ web có sẵn trong khoảng thời gian hoán đổi, thì phông chữ đó sẽ được "hoán đổi".

Các chiến lược font-display phản ánh các quan điểm khác nhau về sự đánh đổi giữa hiệu suất và tính thẩm mỹ. Do đó, rất khó để đưa ra một phương pháp đề xuất vì nó phụ thuộc vào lựa chọn ưu tiên của từng cá nhân, tầm quan trọng của phông chữ web đối với trang và thương hiệu, cũng như mức độ khó chịu của phông chữ đến muộn khi được hoán đổi.

Đối với hầu hết các trang web, đây là 3 chiến lược phù hợp nhất:

  • Nếu hiệu suất là ưu tiên hàng đầu: Hãy sử dụng font-display: optional. Đây là phương pháp "hiệu quả" nhất: quá trình kết xuất văn bản bị trì hoãn không quá 100 mili giây và đảm bảo rằng sẽ không có sự thay đổi bố cục liên quan đến việc hoán đổi phông chữ. Tuy nhiên, nhược điểm ở đây là phông chữ web sẽ không được sử dụng nếu đến muộn.

  • Nếu việc hiển thị văn bản nhanh chóng là ưu tiên hàng đầu, nhưng bạn vẫn muốn đảm bảo phông chữ web đã được sử dụng: Hãy dùng font-display: swap nhưng hãy nhớ phân phối phông chữ đủ sớm để không làm thay đổi bố cục. Nhược điểm của tuỳ chọn này là sự thay đổi đột ngột khi phông chữ đến muộn.

  • Nếu việc đảm bảo văn bản hiển thị trong phông chữ web là ưu tiên hàng đầu: Hãy sử dụng font-display: block nhưng nhớ phân phối phông chữ đủ sớm để giảm thiểu độ trễ của văn bản. Nhược điểm của phương pháp này là việc hiển thị văn bản ban đầu sẽ bị trễ. Xin lưu ý rằng mặc dù có sự hiển thị này, nhưng bố cục vẫn có thể thay đổi vì văn bản thực sự được vẽ ở chế độ ẩn và do đó, không gian phông chữ dự phòng được dùng để đặt trước không gian. Khi phông chữ trên web tải, điều này có thể yêu cầu không gian khác biệt và do đó phải chuyển đổi. Tuy nhiên, đây có thể là sự thay đổi ít khó chịu hơn so với font-display: swap vì bản thân văn bản sẽ không bị chuyển đổi.

Ngoài ra, hãy lưu ý rằng bạn có thể kết hợp hai phương pháp này: ví dụ: sử dụng font-display: swap cho thương hiệu và các phần tử trang khác có hình thức khác biệt; sử dụng font-display: optional cho phông chữ dùng trong văn bản nội dung.

Giảm sự thay đổi giữa phông chữ dự phòng và phông chữ web

Để giảm tác động của CLS, bạn có thể sử dụng các thuộc tính size-adjust mới. Để biết thêm thông tin, hãy xem bài viết về CSS size-adjust. Đây là một tính năng mới trong bộ công cụ của chúng tôi, vì vậy, tính năng này hiện vẫn còn khá phức tạp và cần thực hiện theo cách thủ công. Nhưng chắc chắn đây là một công cụ để thử nghiệm và theo dõi các điểm cải tiến về công cụ trong tương lai!

Kết luận

Phông chữ web vẫn là điểm tắc nghẽn hiệu suất, nhưng chúng tôi có nhiều lựa chọn ngày càng tăng để tối ưu hoá phông chữ nhằm giảm thiểu điểm tắc nghẽn này nhiều nhất có thể.