前端图片加载失败、 img 出现裂图的原因全解析

在前端开发过程中,我们几乎都遇到过这种情况:
页面中某张图片加载不出来,显示成一个小小的“裂图”图标。

这看似简单的问题,实际上可能由多种原因造成,尤其是在 HTTPS 环境下,混合内容机制(Mixed Content) 是最常见、也最容易被误解的根源之一。

本文将带你系统梳理裂图的各种原因、排查思路,并重点讲清楚混合内容的原理与浏览器行为。

一、什么是“裂图”?

“裂图”(broken image)是指浏览器尝试加载 <img> 标签的图片资源失败时的表现形式。
常见表现:

  • 图片区域显示为灰底、叉号、占位符;
  • 控制台出现 Failed to load resourceMixed Content 警告;
  • Network 面板中图片请求状态码为 404 / 403 / blocked。

二、常见的裂图原因汇总

2.1 图片资源不存在

最基础的情况。可能是:

  • 路径错误(相对/绝对路径混乱);
  • 资源被删除或未上传;
  • OSS/CDN 缓存未刷新;
  • 拼接 URL 时丢失了参数(如签名 URL 过期)。

排查建议:

  • 打开 Network 面板;
  • 直接在浏览器地址栏访问图片 URL;
  • 看返回码是否为 404 或 403。

2.2 图片服务器防盗链(Referer 校验)

很多图片或 CDN 都会校验 Referer,限制图片只能在指定域名下访问。
如果请求来源不在白名单内,服务器会拒绝访问(403)。

典型症状:

  • Network 状态码 403;
  • 响应头中带有自定义防盗链提示。

解决办法:

  • 通过后端代理请求;
  • 或联系服务端将当前域加入 Referer 白名单。

2.3 响应头设置错误(Content-Type / Content-Disposition

  • 如果返回头不是图片类型(例如 text/html),浏览器可能无法渲染;
  • 如果设置了 Content-Disposition: attachment,浏览器会触发下载行为,但现代浏览器对图片通常会放行显示。

2.4 权限与签名失效

某些云存储(如 OSS / COS / S3)要求签名 URL 才能访问。
签名过期后图片加载失败

2.5 CSP 限制(Content-Security-Policy)

如果页面设置了严格的 CSP 策略,例如:

Content-Security-Policy: img-src https://static.example.com 

那么任何不在允许列表内的图片都会被阻止。

三、混合内容机制(Mixed Content)

这是前端 HTTPS 场景下导致“裂图”的核心原因之一

这种情况一般控制台会报错:net::ERR_CERT_COMMON_NAME_INVALID

3.1 混合内容是什么?

一个通过 HTTPS 加载的页面,去请求**非 HTTPS(HTTP)**的资源。



User agents must rewrite insecure schemes to secure schemes before fetching the resource.



This does not affect the DOM or the URL reflected to script.
<!-- 页面本身是 https://example.com --> <img src="http://img.example.com/a.png"> 

此时页面是安全的,但图片请求是不安全的,浏览器会认定为“混合内容”。

3.2 为什么要阻止混合内容?

因为 HTTP 请求容易被中间人攻击。
攻击者可以篡改图片、注入恶意脚本、或监听请求,从而破坏整个 HTTPS 页面的安全性。

3.3 混合内容的两种类型

类型说明浏览器行为
主动混合内容(Active Mixed Content)能影响页面逻辑的内容,例如 <script><iframe>、XHR、CSS、WebSocket🚫 直接阻止
被动混合内容(Passive Mixed Content)不影响逻辑的内容,例如图片、音视频、CSS 背景图⚙️ 可能被升级或阻止

3.4 可升级内容(Upgradable Content)

从 Chrome 80+ / Edge 79+ / Firefox 83+ 起,浏览器对某些被动混合内容启用了 “自动升级机制”
当页面是 HTTPS 时,如果 <img> 的地址是 HTTP,浏览器会尝试自动改为 HTTPS 重新加载。

3.4.1 举例说明:

<!-- 页面是 https://page.com --> <img src="http://cdn.page.com/pic.png" /> 

浏览器会自动尝试请求:

https://cdn.page.com/pic.png 

如果资源服务器支持 HTTPS,就会成功显示。
如果不支持或证书无效,加载失败(显示裂图)。

注意浏览器在 network 面板请求的是 https 请求,但是 DOM 中还是现实的 http 地址,浏览器可不会自动修改你的 DOM。


那么问题来了,为啥 DOM 没变但是渲染的是 https?



因为 请求完成后,浏览器渲染的是“响应的二进制内容”,不是 URL

图片 <img> 加载流程其实是:DOM 给出 URL 字符串(http)Resource Loader 把它升级成 https 并发送请求请求返回后产生一个 decoded image bitmap(解码好的位图)<img> 标签拿到的是 “图片数据对象”(Image Bitmap / Image Document)把这个 bitmap 交给渲染引擎绘制在页面上

也就是说,渲染用的不是 URL,而是“图片数据”

3.4.2 浏览器不会把< img > 改成 https

因为 URL 升级只是加载规则(fetching rule)的结果,不是文档结构(DOM) 的改变。

浏览器的设计理念:

  • DOM = 你的代码
  • Loaders = 浏览器的行为

浏览器永远不会篡改你的 DOM 代码,只在加载阶段更换 URL。

3.5 哪些资源属于“可升级内容”

元素类型是否会自动升级
<img>✅ 是
<audio> / <video> / <source>✅ 是
CSS 图片(background-image, border-image✅ 是
<script> / <iframe> / XHR❌ 否,直接阻止

📘 官方文档参考:
MDN: 混合内容(Mixed Content)

3.6 注意:不是“所有场景”都自动升级

  • 升级依赖浏览器支持;
  • 如果资源使用了 IP、非标准端口或证书无效,不会升级;
  • 如果页面设置了 CSP:
  • 则会强制所有 HTTP 请求都升级为 HTTPS
  • 升级失败依然会导致裂图。
Content-Security-Policy: upgrade-insecure-requests 

3.7 为什么以前的说法是“不会自动改为 https”

因为在 Chrome 80 之前(2020 年以前),浏览器并不具备“自动升级”行为,只是警告或阻止。
现在的混合内容机制是新一代浏览器安全策略的演进结果。

3.8 HTTP 页面加载 HTTPS 图片会怎样?

不会触发混合内容,HTTP 页面加载 HTTPS 图片完全合法,只是页面本身不安全。

四、其他容易忽略的裂图原因

4.1 跨域限制(Canvas 绘制)

<img> 跨域加载资源后再画到 <canvas> 上,未设置 crossorigin 会触发安全限制。

4.2  图片过大或加载超时

大图未能在超时时间内返回,可能表现为裂图。

4.3 服务端缓存头异常

过期或 ETag 不匹配,导致 CDN 无法正确命中图片。

五、排查与修复建议

步骤操作检查内容
1Network 面板查看状态码 / 请求协议(HTTP or HTTPS)
2Console 控制台搜索 Mixed ContentFailed to load resource
3直接访问 URL检查是否可用、证书是否有效
4检查 CSP是否限制了 img-src 来源
5检查响应头Content-Type、Referer、防盗链策略等

六、总结

混合内容机制是现代浏览器为保证 HTTPS 安全性而做出的妥协与平衡。
对于前端开发者而言,理解它的升级与阻止逻辑,是解决“裂图”的关键。

类别原因浏览器行为解决方案
资源不存在404裂图修正路径
防盗链403裂图调整 Referer 或代理
HTTP → HTTPS 混合内容被动混合内容自动升级或阻止改为 HTTPS
CSP 限制阻止加载裂图修改策略
Content-Type 错误无法解析裂图服务端修正
签名 URL 失效403裂图重新生成签名

Read more

mysql-9.6.0-winx64 安装踩雷教程

mysql-9.6.0-winx64 安装踩雷教程

今天安装了mysql-9.6.0-winx64,有部分踩雷事项。 下载地址:mysql 1、D盘新建文件夹mysql,把文件压缩到这个文件夹底下 2、在安装包的根目录底下建一个my.ini文件。文件里面写的内容可以直接复制。 * 注意:很多旧教程里面的配置信息是错误和新的mysql不匹配。 会面临错误:MySQL 9.6.0 启动失败。根源是 配置项: default_authentication_plugin=mysql_native_password 在 9.6 版本中已被移除,同时因配置错误导致系统表 mysql.component 缺失。 * basedir具体的地址填写你自己的。 * datadir的data现在是没有的,要等后面初始化的时候才生成。 [mysqld]port=3307basedir=D:\\mysql\\mysql-9.6.0-winx64 datadir=D:

By Ne0inhk
Rust异步缓存系统的设计与实现

Rust异步缓存系统的设计与实现

Rust异步缓存系统的设计与实现 一、引言 💡缓存是现代Web应用架构中的核心组件,能够显著提升系统的性能和响应速度。通过将频繁访问的数据存储在高速缓存中,可以减少对数据库或外部API的请求,从而降低延迟和提高吞吐量。Rust语言的异步特性和内存安全保障使得它非常适合用于构建高性能、可靠的异步缓存系统。 在本章中,我们将深入探讨异步缓存系统的设计与实现,包括缓存策略、数据结构选择、并发安全保障、内存管理、错误处理和过期机制等方面。我们还将通过实战项目集成演示如何在用户同步服务、订单处理服务和监控服务中使用异步缓存系统,以及如何优化缓存系统的性能。 二、异步缓存系统的核心概念 2.1 缓存策略 缓存策略决定了数据在缓存中的存储和淘汰方式,常见的缓存策略包括: * LRU(Least Recently Used):最近最少使用策略,淘汰最近最少使用的数据。 * LFU(Least Frequently Used):最不经常使用策略,淘汰使用频率最低的数据。 * FIFO(First In First Out):先进先出策略,淘汰最早进入缓存的数据。 * TTL(

By Ne0inhk
KWDB 硬核实战:30ms 写入千条轨迹,用 SQL 打造物流车队“天眼”系统

KWDB 硬核实战:30ms 写入千条轨迹,用 SQL 打造物流车队“天眼”系统

前言: 随着 5G 和物联网技术的普及,车联网 (Internet of Vehicles, IoV) 正成为数据爆发的新战场。与传统的静态传感器不同,车辆是移动的计算节点,它们每时每刻都在产生海量的时间序列数据:从 GPS 经纬度到发动机转速,从剩余油量到刹车踏板状态。 对于一家拥有数百辆货车的物流公司而言,这些数据就是金矿。通过实时监控,可以有效降低油耗、杜绝违规驾驶、优化配送路线。然而,传统的关系型数据库在面对车辆高频上报(例如每秒 10 次)的轨迹数据时,往往面临写入瓶颈;而单纯的时序数据库又难以处理复杂的车辆档案关联查询。 KWDB (KaiwuDB) 的“多模”特性恰好解决了这一痛点。今天,我们将实战构建一个物流车队实时监控平台,挑战如何在一个数据库内同时搞定“车辆档案管理”与“海量轨迹分析”。 场景设定:我们要为一个拥有 200 辆货车的物流车队构建监控系统。 核心挑战:高频写入:车辆每 10

By Ne0inhk
Spring Boot 数据访问与数据库集成

Spring Boot 数据访问与数据库集成

Spring Boot 数据访问与数据库集成 18.1 学习目标与重点提示 学习目标:掌握Spring Boot数据访问与数据库集成的核心概念与使用方法,包括Spring Boot数据访问的基本方法、Spring Boot与MySQL的集成、Spring Boot与H2的集成、Spring Boot与MyBatis的集成、Spring Boot与JPA的集成、Spring Boot的事务管理、Spring Boot的实际应用场景,学会在实际开发中处理数据库访问问题。 重点:Spring Boot数据访问的基本方法、Spring Boot与MySQL的集成、Spring Boot与H2的集成、Spring Boot与MyBatis的集成、Spring Boot与JPA的集成、Spring Boot的事务管理、Spring Boot的实际应用场景。 18.2 Spring Boot数据访问概述 Spring Boot数据访问是指使用Spring Boot进行数据库操作的方法。 18.2.1 数据访问的定义

By Ne0inhk