前端老鸟血泪总结:iframe跨域通信postMessage实战避坑指南

前端老鸟血泪总结:iframe跨域通信postMessage实战避坑指南
在这里插入图片描述


前端老鸟血泪总结:iframe跨域通信postMessage实战避坑指南

前端老鸟血泪总结:iframe跨域通信postMessage实战避坑指南

开篇先唠两句

说实话,每次有人问我"iframe这破玩意儿怎么还没死"的时候,我都想点根烟(虽然我不抽)跟他聊聊人生。都2026年了,前端圈子里的新玩具跟韭菜似的割了一茬又一茬,Vue3、React19、各种微前端框架吹得天花乱坠,结果一到实际项目里,产品经理甩过来一个需求:“我们要嵌入第三方的支付页面/客服系统/广告位”,你一看文档,好家伙,人家就给你个iframe链接,爱用不用。

这时候你就明白了,有些技术就像你老家衣柜底下那双发霉的运动鞋,平时想不起来,真到了要搬家的时候,你还得捏着鼻子把它翻出来。

我到现在都记得三年前那个凌晨两点,我对着屏幕发呆的场景。当时在做一个电商后台,主站是React写的,支付模块是外包团队用jQuery糊的,部署在不同的域名下。测试环境跑得好好的,一上生产环境,支付成功后的回调消息死活传不回来。我在公司群里疯狂@后端同事,说是不是接口挂了,人家一查日志,接口正常返回200。我又去翻浏览器的Network面板,数据明明回来了啊,怎么前端就是收不到?

后来我在Chrome控制台里敲了俩小时的window.postMessage,像个傻子一样在父页面和iframe之间来回切换,终于发现我把targetOrigin写成了http://pay.example.com,而实际上人家已经偷偷升级成了https。就这一个字母,让我熬到凌晨四点,第二天还要强撑着去跟老板汇报进度。那一刻我深刻理解了什么叫"跨域一时爽,调试火葬场"。

所以这篇东西,算是我用头发换来的经验。不敢说让你彻底爱上iframe,但至少下次再遇到这种破事,你能少骂两句娘,早点下班。

先搞懂postMessage到底是个啥

同源策略那堵墙是怎么把咱们挡在外面的

咱们做前端的,天天跟浏览器打交道,但很多人其实没认真想过:为啥我本地启动的localhost:3000就是访问不了localhost:8080的数据?这明明都是我自己电脑上的服务啊,浏览器你管得着吗?

还真管得着。这就是传说中的同源策略(Same-Origin Policy),1995年网景公司搞出来的老古董,到现在还在折磨我们。它的逻辑很简单:如果两个URL的协议、域名、端口有一个不一样,那你们就是"不同源",浏览器默认不允许你们互相读取对方的数据。

为啥要这么干?想象一下如果没有这个限制,你打开一个恶意网站,它就能直接读取你同时登录的银行页面的DOM,把你的余额和转账按钮都扒拉出来,那还得了?所以同源策略本质上是浏览器给用户的一道安全防线,虽然有时候这道防线会把我们自己人也挡在外面。

但业务需求是无情的。主子应用分离部署、嵌入第三方服务、微前端架构…这些场景下我们确实需要不同源的页面互相通信。早期大家用各种歪门邪道:通过修改document.domain来降级(现在已经被主流浏览器限制了)、利用window.name这个全局变量做中转( hack味太重)、还有更离谱的用location.hash传数据(URL长度限制不说,看着就寒碜)。

直到HTML5出了postMessage这个API,才算是给了咱们一个正经的解决方案。它就像是浏览器在两个互相猜忌的页面之间开了一扇小窗户,你们可以通过这个窗户递纸条,但窗户的大小、能递什么纸条,都有严格的规矩。

postMessage就是浏览器给咱们开的后门

postMessage的用法其实特别简单,就两个动作:发消息收消息

发消息的语法长这样:

// 父页面向iframe发消息const iframe = document.getElementById('myIframe'); iframe.contentWindow.postMessage('你好啊小老弟','https://child-domain.com');// iframe向父页面发消息 window.parent.postMessage('爹,我收到了','https://parent-domain.com');

看到没,核心就是otherWindow.postMessage(message, targetOrigin)。第一个参数是你想传的数据,可以是字符串、数字、对象(会被结构化克隆算法序列化),第二个参数超级重要,是目标窗口的源(origin),也就是协议+域名+端口的组合。

这里有个容易踩的坑:很多人以为targetOrigin是写当前页面的origin,其实是写对方的origin。就像你寄快递,收件人地址得写对方的,不能写你自己的。写错了的话,浏览器会直接静默失败,消息发不出去,也不会报错,就当你没发过一样。这种"冷暴力"是最让人抓狂的。

message事件监听器怎么接住飞过来的消息

光发不行,还得有人收。接收方需要监听message事件:

window.addEventListener('message',function(event){// event对象里包含了一堆有用的信息 console.log('收到消息内容:', event.data);// 对方发过来的数据 console.log('消息来源:', event.origin);// 发消息方的origin,比如"https://example.com" console.log('消息源窗口:', event.source);// 发消息窗口的引用,可以用来回消息});

这个event对象就是整个通信过程的关键。浏览器会自动填充这些信息,特别是event.origin,它是浏览器根据实际发送方的真实origin填写的,无法被伪造。这就是为什么postMessage比那些野路子方案安全的原因——你能确切地知道这条消息是谁发来的。

但这里有个大坑:很多人拿到消息就直接用了,完全不校验event.origin。这就好比你家门口装了个信箱,谁来信你都拆,那坏人给你寄个病毒你也照单全收?正确的做法必须是:

window.addEventListener('message',function(event){// 严格校验来源,不是信任的网站一律无视if(event.origin !=='https://trusted-domain.com'){ console.warn('收到来自未知来源的消息,已忽略:', event.origin);return;}// 校验通过后再处理数据handleMessage(event.data);});

记住,永远不要相信客户端传来的数据,即使是你自己的iframe发过来的。谁知道中间有没有被劫持?生产环境里的origin校验必须白名单机制,写死允许的域名,不要用*通配符,那玩意儿就是给自己埋雷。

这俩配合起来就像微信发消息和收消息

其实整个postMessage的机制,跟咱们日常用微信聊天挺像的。你想给好友发消息(postMessage),得先知道对方的微信号(targetOrigin),消息发出去之后,对方手机会响(message事件触发),他打开一看(event.data),知道是你发的(event.origin),然后可以选择回消息(event.source.postMessage)或者已读不回。

不同的是,微信你发错人了会提示"对方不是你的好友",但postMessage发错targetOrigin,浏览器啥都不说,就让你消息石沉大海。这也是为什么调试postMessage那么痛苦——你根本不知道消息是发出去了对方没收到,还是根本没发出去,还是对方收到了但处理的时候报错了。

所以实际开发中,我强烈建议给通信加上"已读回执"机制,也就是双向确认。这个后面代码部分会详细讲。

手把手教你写代码

光说不练假把式,接下来咱们从零开始,搭一个完整的父子页面通信demo。我会把能踩的坑都踩一遍,你看着心疼就行,不用亲自体验了。

父页面怎么往iframe里塞消息

先搭个父页面,咱们就叫它parent.html

<!DOCTYPEhtml><htmllang="zh-CN"><head><metacharset="UTF-8"><title>父页面 - 我是你爹</title><style>body{font-family: -apple-system, sans-serif;padding: 20px;}#childFrame{width: 600px;height: 400px;border: 2px solid #333;margin-top: 20px;}.controls{margin: 10px 0;}button{padding: 8px 16px;margin-right: 10px;cursor: pointer;}#log{background: #f5f5f5;padding: 10px;margin-top: 20px;height: 200px;overflow-y: auto;}</style></head><body><h1>🚀 父页面控制台</h1><divclass="controls"><inputtype="text"id="messageInput"placeholder="输入要发送的消息"style="width: 300px;padding: 8px;"><buttononclick="sendToChild()">发送给iframe</button><buttononclick="sendComplexData()">发送复杂对象</button><buttononclick="requestChildData()">请求iframe数据</button></div><!-- 这里嵌入子页面,注意跨域场景下src必须是不同源的 --><iframeid="childFrame"src="http://localhost:8080/child.html"></iframe><divid="log"><strong>通信日志:</strong><br></div><script>const iframe = document.getElementById('childFrame');const logDiv = document.getElementById('log');// 简单的日志工具functionlog(msg, type ='info'){const time =newDate().toLocaleTimeString();const color = type ==='error'?'red': type ==='success'?'green':'#333'; logDiv.innerHTML +=`<divtoken interpolation">${color}; margin: 4px 0;">[${time}] ${msg}</div>`; logDiv.scrollTop = logDiv.scrollHeight;}// 等iframe加载完再通信,不然会发到空气里 iframe.onload=function(){log('✅ iframe加载完成,可以开始通信了','success');// 先发个初始化消息sendMessageToChild({type:'INIT',data:{parentVersion:'1.0.0',timestamp: Date.now()}});};// 基础发送函数,封装了targetOrigin的处理functionsendMessageToChild(messageObj){// 这里的targetOrigin必须和iframe的src匹配!// 如果iframe是http://localhost:8080/child.html,origin就是http://localhost:8080const targetOrigin ='http://localhost:8080';try{// 一定要等contentWindow存在,虽然onload已经触发,但保险起见还是判断一下if(iframe.contentWindow){ iframe.contentWindow.postMessage(messageObj, targetOrigin);log(`📤 发送消息: ${JSON.stringify(messageObj)}`);}else{log('❌ iframe.contentWindow不存在','error');}}catch(err){log(`❌ 发送失败: ${err.message}`,'error');}}// 按钮点击:发送文本消息functionsendToChild(){const input = document.getElementById('messageInput');const text = input.value.trim();if(!text){alert('先输入点内容啊大哥');return;}sendMessageToChild({type:'TEXT_MESSAGE',payload: text,from:'parent',id:generateMsgId()// 生成唯一ID,用于追踪}); input.value ='';}// 按钮点击:发送复杂数据结构functionsendComplexData(){sendMessageToChild({type:'COMPLEX_DATA',payload:{userInfo:{name:'张三',age:28,hobbies:['coding','debugging']},config:{theme:'dark',notifications:true},timestamp:newDate().toISOString()},id:generateMsgId()});}// 按钮点击:请求iframe的数据(双向通信)functionrequestChildData(){const requestId =generateMsgId();sendMessageToChild({type:'REQUEST_DATA',requestType:'userStatus',// 告诉对方我要什么数据requestId: requestId,// 用于匹配响应timestamp: Date.now()});log(`⏳ 发送数据请求,requestId: ${requestId},等待响应...`);}// 生成消息ID,用于追踪和去重functiongenerateMsgId(){return'msg_'+ Date.now()+'_'+ Math.random().toString(36).substr(2,9);}// ========== 接收iframe消息 ========== window.addEventListener('message',function(event){// 严格校验来源!假设子页面跑在localhost:8080if(event.origin !=='http://localhost:8080'){ console.warn('收到未知来源消息:', event.origin);return;}const data = event.data;log(`📥 收到iframe消息: ${JSON.stringify(data)}`);// 根据消息类型处理if(data.type ==='INIT_ACK'){log('✅ iframe初始化确认收到,双向通道建立','success');}elseif(data.type ==='DATA_RESPONSE'){log(`✅ 收到请求的数据 [${data.requestId}]: ${JSON.stringify(data.payload)}`,'success');}elseif(data.type ==='ERROR'){log(`❌ iframe报错: ${data.message}`,'error');}elseif(data.type ==='USER_ACTION'){// 处理用户行为,比如iframe里的按钮点击handleUserAction(data.payload);}});functionhandleUserAction(action){log(`🎯 处理用户行为: ${action.type} - ${action.detail}`);// 这里可以触发父页面的业务逻辑}</script></body></html>

看到没,父页面的核心逻辑就三块:等iframe加载、发消息、收消息。但每一块都有讲究:

  1. 必须等iframe.onload:如果你在DOM刚创建完就发消息,iframe可能还没加载完,contentWindow是null,或者消息发进去了但子页面的监听器还没准备好,直接白给。我见过太多人在这里栽跟头,然后怀疑人生半小时。
  2. targetOrigin必须精确匹配:我上面写的是http://localhost:8080,如果你的子页面实际是https://localhost:8080或者带端口号http://localhost:8081,这个消息浏览器会直接丢弃,而且不报错。调试的时候可以在控制台手动试试iframe.contentWindow.postMessage('test', '*'),如果能收到,说明就是origin写错了。
  3. 消息格式要统一:我建议所有消息都包成一个对象,至少包含type字段用于区分消息类型,id字段用于追踪。别今天传字符串明天传对象,后天你自己都记不住该判断typeof event.data === 'string'还是别的。

iframe那边怎么竖起耳朵听

接下来是子页面child.html,跑在localhost:8080

<!DOCTYPEhtml><htmllang="zh-CN"><head><metacharset="UTF-8"><title>子页面 - 乖巧听话版</title><style>body{font-family: -apple-system, sans-serif;padding: 20px;background:linear-gradient(135deg, #667eea 0%, #764ba2 100%);color: white;}.container{background:rgba(255,255,255,0.1);padding: 20px;border-radius: 10px;}button{padding: 8px 16px;margin: 5px;cursor: pointer;border: none;border-radius: 4px;}#msgDisplay{background:rgba(0,0,0,0.2);padding: 10px;margin-top: 10px;min-height: 100px;}</style></head><body><divclass="container"><h2>👶 iframe子页面</h2><p>origin: <spanid="originDisplay"></span></p><div><buttononclick="sendToParent()">给父页面发消息</button><buttononclick="sendErrorToParent()">模拟报错</button><buttononclick="notifyParentUserAction()">通知父页面用户点击</button></div><divid="msgDisplay"><strong>收到的消息:</strong></div></div><script>// 显示当前origin document.getElementById('originDisplay').textContent = window.location.origin;const msgDisplay = document.getElementById('msgDisplay');let parentOrigin =null;// 存储父页面origin,用于回消息let isInitialized =false;functionlog(msg){const time =newDate().toLocaleTimeString(); msgDisplay.innerHTML +=`<div>[${time}] ${msg}</div>`;}// ========== 监听父页面消息 ========== window.addEventListener('message',function(event){// 安全校验第一关:origin// 实际项目中这里应该写死父页面的origin,比如'http://localhost:3000'// 开发环境可以放宽,但生产环境必须严格校验const allowedOrigins =['http://localhost:3000','http://127.0.0.1:3000'];if(!allowedOrigins.includes(event.origin)){ console.warn('iframe收到未知来源消息,已拒绝:', event.origin);return;}// 记录父页面origin,后面回消息要用if(!parentOrigin){ parentOrigin = event.origin; console.log('已记录父页面origin:', parentOrigin);}const data = event.data;// 防御性编程:确保data是对象且有type字段if(!data ||typeof data !=='object'||!data.type){ console.warn('收到格式不正确的消息:', data);return;}log(`📥 收到父消息 [${data.type}]: ${JSON.stringify(data).substring(0,100)}...`);// 根据消息类型分发处理switch(data.type){case'INIT':handleInit(data);break;case'TEXT_MESSAGE':handleTextMessage(data);break;case'COMPLEX_DATA':handleComplexData(data);break;case'REQUEST_DATA':handleDataRequest(data, event.source);break;default: console.warn('未知消息类型:', data.type);}});functionhandleInit(data){ isInitialized =true;log('✅ 收到初始化消息,父版本: '+ data.data.parentVersion);// 回发确认消息sendToParent({type:'INIT_ACK',data:{childStatus:'ready',timestamp: Date.now()}});}functionhandleTextMessage(data){log('📝 文本消息: '+ data.payload);// 这里可以更新UI,比如显示在页面上}functionhandleComplexData(data){log('📦 复杂数据: '+JSON.stringify(data.payload).substring(0,50)+'...');// 实际业务中这里可能是更新表单、渲染图表等操作}functionhandleDataRequest(data, sourceWindow){log('📡 收到数据请求: '+ data.requestType);// 模拟异步获取数据setTimeout(()=>{const responseData ={userStatus:'online',currentPage:'payment',progress:75,lastUpdate:newDate().toISOString()};// 注意:回消息要用event.source,而不是window.parent!// 虽然大多数情况下它们一样,但如果是多层级嵌套iframe,parent可能不是直接父级 sourceWindow.postMessage({type:'DATA_RESPONSE',requestId: data.requestId,// 把请求ID带回去,方便父页面匹配payload: responseData,timestamp: Date.now()}, event.origin);// 这里用event.origin作为targetOrigin},500);}// ========== 向父页面发消息 ==========functionsendToParent(messageObj){if(!parentOrigin){ console.error('还不知道父页面origin,先等父页面发消息过来');return;}// 必须用window.parent,而且targetOrigin必须是父页面的origin window.parent.postMessage(messageObj, parentOrigin);log('📤 发送给父: '+ messageObj.type);}// 按钮点击:主动发消息给父functionsendToParent(){sendToParent({type:'CHILD_MESSAGE',payload:'我是iframe,我很好',timestamp: Date.now()});}functionsendErrorToParent(){sendToParent({type:'ERROR',message:'iframe内部出错了:模拟的错误信息',stack:'Error: mock error\n at child.html:123'});}functionnotifyParentUserAction(){sendToParent({type:'USER_ACTION',payload:{type:'BUTTON_CLICK',detail:'用户点击了"通知父页面"按钮',timestamp: Date.now()}});}// iframe加载完成后,可以主动通知父页面(但此时可能还不知道父页面origin)// 所以通常还是等父页面先发INIT消息比较稳妥 console.log('iframe加载完成,等待父页面建立连接...');</script></body></html>

子页面的关键点:

  1. origin白名单机制:我上面用了数组allowedOrigins,生产环境你应该把允许的父页面域名写死在这里,不要用*,也不要用includes做模糊匹配(比如event.origin.endsWith('.example.com')),那会被子域名劫持攻击。
  2. 用event.source回消息:看到handleDataRequest里我用的是event.source.postMessage而不是window.parent.postMessage吗?这是为了兼容多层嵌套的场景。如果你的iframe里再套一个iframe,那window.parent是中间层,不是最外层。event.source永远是发送这条消息的那个窗口的准确引用。
  3. 消息格式校验:收到消息后先判断data是不是对象、有没有type字段,防止收到垃圾数据或者恶意构造的消息导致JS报错。

双向通信怎么搞,别整成单相思

上面的代码其实已经展示了双向通信,但这里我要单独拎出来说,因为很多人容易搞成"单相思"——父页面拼命发,iframe也能收到,但iframe回的消息父页面收不到,或者反过来。

双向通信的核心在于双方都要知道对方的origin

  • 父页面知道iframe的origin:因为iframe的src是父页面指定的,直接读iframe.src就能解析出来。
  • iframe知道父页面的origin:这个比较麻烦,因为iframe被嵌入到哪个页面是它自己不知道的(除非你通过URL参数传进去,但那样不安全)。

所以通常的套路是:

  1. 父页面等iframe加载完后,先发一条INIT消息,带上自己的identity
  2. iframe收到INIT后,记录event.origin,然后回一个ACK
  3. 从此以后双方都知道该往哪发了

如果你跳过第一步,iframe直接window.parent.postMessage(msg, '*'),虽然能发出去,但用了*作为targetOrigin,安全性大打折扣,而且父页面如果嵌了多个iframe,也不知道这条消息是谁发的。

targetOrigin参数写错直接变哑巴,这个必须重点说

我再强调一遍targetOrigin的重要性,因为这是我踩过最多的坑。

常见错误1:协议不匹配

// 错的:iframe实际是https iframe.contentWindow.postMessage(msg,'http://example.com');// 对的: iframe.contentWindow.postMessage(msg,'https://example.com');

常见错误2:多了路径

// 错的:targetOrigin不是URL,不能带路径 iframe.contentWindow.postMessage(msg,'https://example.com/path/to/page');// 对的: iframe.contentWindow.postMessage(msg,'https://example.com');

常见错误3:端口问题

// 本地开发时特别容易错// 父页面在localhost:3000,iframe在localhost:8080// 在父页面里发消息,targetOrigin应该是iframe的origin: iframe.contentWindow.postMessage(msg,'http://localhost:8080');// 而不是父页面自己的origin!

调试技巧:如果你怀疑targetOrigin写错了,可以临时改成*测试:

// 调试用,生产环境绝对禁止! iframe.contentWindow.postMessage(msg,'*');

如果改成*就能收到,100%是origin写错了。这时候去控制台打印一下iframe.contentWindow.location.origin,看看实际是什么。

消息数据结构怎么设计才不翻车

随着项目复杂度增加,你会发现消息类型越来越多,如果没有统一规范,很快会变成一团乱麻。我建议的消息结构:

{// 必填:消息类型,用于路由到不同处理器type:'USER_LOGIN'|'DATA_REQUEST'|'UI_UPDATE'|'ERROR_REPORT'|...,// 必填:消息唯一ID,用于追踪和去重id:'msg_1672531200000_abc123',// 选填:时间戳timestamp:1672531200000,// 选填:业务数据payload:{...},// 选填:对于请求-响应模式,带上请求IDrequestId:'req_xxx',// 选填:错误信息error:{code:'ERR_001',message:'...'},// 选填:元数据,比如发送方标识、版本号等meta:{sender:'parent_v2.1',target:'child'}}

这种结构的好处是:

  • 所有消息都有type,可以用switch-case或者策略模式处理
  • id字段可以用于消息去重(防止网络抖动导致重复处理)和链路追踪
  • requestId实现了请求-响应的匹配,特别是并发多个请求时不会乱套
  • meta可以扩展各种附加信息,不影响主业务数据结构

这方案香在哪又坑在哪

好处是原生支持不用装乱七八糟的库

说实话,现在前端生态有点病态,屁大点功能都要npm install一个库。但postMessage是浏览器原生API,不用打包,不用担心tree-shaking,不用担心版本冲突,兼容性还贼好——IE8+都支持(虽然IE8只支持字符串,不支持对象,但2026年了应该没人管IE了吧?)。

而且它是真正的异步非阻塞,发消息不会卡住主线程。比起一些基于轮询的hack方案(比如不停读window.name或者location.hash),性能上强太多了。

兼容性基本没问题,老浏览器也能跑

根据Can I use的数据,postMessage的兼容性是这样的:

  • Chrome 1+
  • Firefox 3+
  • Safari 4+
  • Edge 所有版本
  • IE 8+(部分支持)

唯一要注意的是IE8/9有一些限制,比如不支持传输对象(只能传字符串),而且event.source在某些情况下可能是undefined。但说实话,如果你还需要支持IE,那可能postMessage不是你最大的痛点,建议先换个工作(开玩笑的,生活所迫我理解)。

坑就是origin校验不做好分分钟被XSS

这是最大的安全隐患。如果你接收消息时不校验event.origin,或者用了*作为targetOrigin,攻击者就可以:

  1. 诱导用户访问一个恶意网站
  2. 恶意网站通过window.open或者iframe嵌入你的页面
  3. 向你发送伪造的消息,触发你的业务逻辑(比如修改用户信息、发起支付等)

真实案例:2019年某知名电商网站的支付页面,因为没校验postMessage的origin,攻击者可以嵌入该页面并向其发送"支付成功"的消息,导致订单状态被篡改,虽然没直接造成资金损失,但订单逻辑全乱了。

所以再次强调:生产环境必须白名单校验origin,绝对禁止*,而且要精确匹配,不要用正则或者字符串包含判断

消息发出去石沉大海怎么排查

调试postMessage的痛苦之处在于它的静默失败特性。targetOrigin不匹配?不报错。对方窗口已关闭?不报错。消息被CSP策略拦截?可能也不报错。

我的排查 checklist:

  1. 检查控制台有没有报错:虽然postMessage本身不报错,但接收方的处理逻辑可能报错
  2. 确认iframe真的加载完了:在iframe.onload里打印日志,确认时机
  3. 临时用*测试:如果*能收到,说明origin写错了;如果*也收不到,说明是时机问题或者窗口引用错了
  4. 检查CSP策略:如果页面有Content-Security-Policy,看看有没有限制frame-src或者connect-src
  5. 用Chrome的MessageChannel调试:在控制台输入monitorEvents(window, 'message'),可以实时看到所有message事件

嵌套多层iframe的时候能把你绕晕

如果你遇到这种场景:页面A嵌入iframe B,B又嵌入iframe C,然后A要和C通信… 恭喜,你需要当传话筒了。

A (顶层) └── B (中间层) └── C (底层) 

A不能直接给C发消息,因为跨了两层。解决方案:

  1. B做转发:A发给B,B判断是给自己的还是给C的,如果是给C的就转发
  2. C直接找A:C可以用window.top直接拿到最顶层窗口,然后发消息给A,A收到后通过event.source回消息给C(绕过B)

第二种方案更直接,但前提是C知道A的origin。如果C是第三方页面,通常不知道会被嵌入到哪个父页面,所以还是方案1更通用。

代码示例(B页面做转发):

// B页面(中间层)的message监听 window.addEventListener('message',function(event){// 来自父页面A的消息if(event.origin ==='https://a.com'){// 如果是给C的,转发到iframe Cif(event.data.target ==='C'){const iframeC = document.getElementById('iframeC'); iframeC.contentWindow.postMessage(event.data,'https://c.com');}else{// 给自己的,处理业务逻辑handleMessage(event.data);}}// 来自子页面C的消息elseif(event.origin ==='https://c.com'){// 如果是给A的,转发到parentif(event.data.target ==='A'){ window.parent.postMessage(event.data,'https://a.com');}}});

这种"传话筒"模式维护起来很痛苦,消息路径长了容易丢,而且调试的时候你根本不知道消息卡在哪一层了。所以架构设计时尽量避免三层以上的嵌套,或者考虑用更专业的微前端方案(比如qiankun、single-spa)替代裸iframe。

实际项目里都是怎么用的

微前端架构里主子应用互相喊话

虽然qiankun这些框架封装了应用间通信,但底层很多也是基于postMessage(或者其封装版CustomEvent)。如果你用的是"野蛮版"微前端——就是简单粗暴地把子应用塞进iframe——那postMessage就是你们的生命线。

典型场景:主应用登录后拿到token,要同步给所有子应用;子应用路由切换了,要通知主应用更新菜单高亮;子应用内部报错,要上报给主应用的监控系统。

// 主应用:登录成功后广播tokenfunctionbroadcastToken(token){const iframes = document.querySelectorAll('iframe[data-micro-app]'); iframes.forEach(iframe=>{const appName = iframe.dataset.microApp;const origin =getAppOrigin(appName);// 从配置里读每个子应用的origin iframe.contentWindow.postMessage({type:'TOKEN_UPDATE',payload:{ token,expiresAt: Date.now()+3600000},broadcast:true// 标记为广播消息}, origin);});}// 子应用:监听token更新 window.addEventListener('message',(e)=>{if(e.data.type ==='TOKEN_UPDATE'){ localStorage.setItem('token', e.data.payload.token);// 更新axios实例的header apiClient.defaults.headers.Authorization =`Bearer ${e.data.payload.token}`;}});

嵌入第三方支付页面拿回调结果

这是最常见的iframe应用场景了。支付宝、微信支付、各种银行网关,都提供iframe嵌入版本。支付完成后,第三方页面需要把结果(成功/失败/取消)通知给你。

// 父页面:打开支付iframefunctionopenPayment(payParams){const payFrame = document.createElement('iframe'); payFrame.src =`https://pay.thirdparty.com/checkout?orderId=${payParams.orderId}`; payFrame.id ='paymentFrame'; document.body.appendChild(payFrame);// 设置超时,防止用户一直不支付也不关闭const timeout =setTimeout(()=>{closePayment('TIMEOUT');},300000);// 5分钟超时// 监听支付结果consthandlePaymentMessage=(e)=>{if(e.origin !=='https://pay.thirdparty.com')return;if(e.data.type ==='PAYMENT_RESULT'){clearTimeout(timeout);closePayment();if(e.data.status ==='SUCCESS'){// 去后端查询订单状态,确认支付成功(重要!不能信前端回调)verifyPayment(e.data.orderId).then(()=>{showSuccessPage();});}else{showErrorPage(e.data.errorMsg);}}}; window.addEventListener('message', handlePaymentMessage);// 清理函数functionclosePayment(reason){ window.removeEventListener('message', handlePaymentMessage);const frame = document.getElementById('paymentFrame');if(frame) frame.remove();if(reason ==='TIMEOUT'){alert('支付超时,请重新发起');}}}

重要提醒:支付结果必须以后端回调为准,不能全信前端postMessage传来的"支付成功"!因为前端的postMessage可以被伪造,或者网络延迟导致消息丢失。正确的做法是收到前端消息后,再去后端查订单状态确认。

图片上传完把URL传回主页面

有时候我们会把图片上传功能拆成一个独立的微服务或者第三方服务,用iframe嵌入到表单页面里。用户选好图片、裁剪完、上传成功后,iframe需要把图片URL回传给父页面。

// iframe内部:上传成功后asyncfunctionhandleUploadSuccess(imageInfo){// 压缩图片信息,只传必要的const message ={type:'IMAGE_UPLOADED',payload:{url: imageInfo.cdnUrl,thumbnail: imageInfo.thumbnailUrl,width: imageInfo.width,height: imageInfo.height,size: imageInfo.size,name: imageInfo.originalName },timestamp: Date.now()};// 父页面可能是不同源的,从URL参数里读parentOrigin(需要父页面传入)const parentOrigin =newURLSearchParams(location.search).get('parentOrigin');if(parentOrigin){ window.parent.postMessage(message, parentOrigin);// 加个视觉反馈,告诉用户"已经传过去了"showToast('图片已选择,正在返回...');// 延迟关闭iframe,让用户看到反馈setTimeout(()=>{// 通知父页面关闭iframe window.parent.postMessage({type:'CLOSE_ME'}, parentOrigin);},800);}}// 父页面:接收图片URL window.addEventListener('message',(e)=>{// 假设iframe跑在https://upload.example.comif(e.origin !=='https://upload.example.com')return;if(e.data.type ==='IMAGE_UPLOADED'){const imgData = e.data.payload;// 填充到表单 document.getElementById('imageUrl').value = imgData.url; document.getElementById('preview').src = imgData.thumbnail;// 关闭iframe(或者等iframe自己发CLOSE_ME) document.getElementById('uploadFrame').style.display ='none';}});

埋点数据从iframe里往外送

如果你的主应用接了神策、GrowingIO之类的埋点系统,但子应用是独立部署的iframe,你可能需要把子应用的用户行为数据汇总到主应用统一上报。

// iframe内部:封装一个埋点SDKconst Tracker ={// 缓存一下,如果父页面还没准备好,先存着queue:[],init(){// 尝试建立连接this.sendToParent({type:'TRACKER_INIT'});// 每秒检查一次队列,看看有没有堆积的setInterval(()=>this.flush(),1000);},track(eventName, properties ={}){const event ={event: eventName,properties:{...properties,url: location.href,timestamp: Date.now(),from:'child_iframe'}};this.queue.push(event);this.flush();},flush(){if(this.queue.length ===0)return;// 简单判断:如果parentOrigin还没确定(还没收到父消息),先不发if(!window.parentOrigin)return;this.sendToParent({type:'TRACK_EVENTS',payload:this.queue.splice(0,10)// 一次最多发10条,防止消息太大});},sendToParent(msg){// parentOrigin在收到父页面INIT消息后设置if(window.parentOrigin){ window.parent.postMessage(msg, window.parentOrigin);}}};// 父页面:接收埋点并上报 window.addEventListener('message',(e)=>{if(e.data.type ==='TRACK_EVENTS'){// 用主应用的埋点SDK上报 e.data.payload.forEach(event=>{// 加上标识,区分是主应用还是iframe的事件 sa.track(event.event,{...event.properties,_source:'iframe_child',_parent_url: location.href });});}});

SSO单点登录场景下的token传递

企业级应用常见场景:主应用是统一门户,嵌入了N个子系统(HR系统、财务系统、OA系统),用户登录门户后,token需要同步给所有子系统,避免重复登录。

// 主应用:登录成功后,给所有子系统发tokenclassSSOManager{constructor(){this.tokens =newMap();// 存储各系统的tokenthis.subApps =newMap();// 存储子应用信息}registerApp(appId, iframeElement, origin){this.subApps.set(appId,{iframe: iframeElement, origin });// 监听子应用的就绪消息consthandler=(e)=>{if(e.origin !== origin)return;if(e.data.type ==='APP_READY'&& e.data.appId === appId){// 子应用准备好了,发tokenthis.sendToken(appId);}}; window.addEventListener('message', handler);// 存储handler方便后续清理this.subApps.get(appId).handler = handler;}loginSuccess(tokenData){// 解析token,按子系统拆分(不同子系统可能有不同的权限范围)this.subApps.forEach((app, appId)=>{const appToken =this.generateTokenForApp(tokenData, appId);this.tokens.set(appId, appToken);this.sendToken(appId);});}sendToken(appId){const app =this.subApps.get(appId);const token =this.tokens.get(appId);if(!app ||!token)return; app.iframe.contentWindow.postMessage({type:'SSO_TOKEN',payload:{accessToken: token.accessToken,refreshToken: token.refreshToken,expiresIn: token.expiresIn,scope: token.scope }}, app.origin);}generateTokenForApp(masterToken, appId){// 实际业务中这里可能调用SSO服务,换取特定子系统的token// 简化示例直接返回return{...masterToken,scope:`app:${appId}`};}}

踩坑实录和救命招数

消息收不到先检查origin写没写对

这是排名第一的坑,没有之一。我统计过自己过去三年遇到的postMessage问题,60%以上都是origin写错了。包括但不限于:

  • http写成https
  • 漏了端口号(比如localhost:3000写成localhost)
  • 多写了路径(比如example.com/page写成example.com/page)
  • 大小写不匹配(虽然origin理论上不区分大小写,但以防万一)

救命招数:在发送前打印一下你要用的targetOrigin,和iframe实际的contentWindow.location.origin对比,确保完全一致。

// 调试代码 console.log('目标iframe实际origin:', iframe.contentWindow.location.origin); console.log('我要发送的targetOrigin:', targetOrigin); console.log('是否匹配:', iframe.contentWindow.location.origin === targetOrigin);

console.log都打不出来可能是iframe还没加载完

很多人遇到"消息发不出去,控制台啥也没有"的情况,第一反应是代码写错了,折腾半小时发现是iframe还没onload。特别是动态创建的iframe,从设置src到真正加载完成有时间差,如果这时候发消息,直接白给。

救命招数:永远等iframe.onload或者iframe.contentWindow.onload(有些浏览器支持)后再通信。如果是动态创建的iframe,确保src设置后再绑定onload事件,顺序别反了。

// 正确顺序const iframe = document.createElement('iframe'); iframe.onload=()=>{/* 发消息 */};// 先绑事件 iframe.src ='https://child.com';// 后设src// 错误顺序(可能错过load事件)const iframe = document.createElement('iframe'); iframe.src ='https://child.com';// 先设src iframe.onload=()=>{/* 如果src是缓存的,可能瞬间加载完,这里绑晚了 */};

用*通配符一时爽,安全火葬场

我知道调试的时候写*很方便,不用管origin对不对,能跑就行。但请一定记住:*绝对不能出现在生产代码里,哪怕是"临时"的。因为"临时"代码往往比正式代码活得还久,哪天被攻击了你就哭吧。

救命招数:开发环境可以用配置开关,允许*,但打包生产环境时必须有eslint或代码扫描工具检查,发现*就报错阻断发布。

// 配置化方案const config ={development:{allowWildcard:true,// 开发环境允许*strictOriginCheck:false},production:{allowWildcard:false,strictOriginCheck:true,allowedOrigins:['https://app1.com','https://app2.com']}};functiongetTargetOrigin(){if(env ==='production'){return'https://specific-domain.com';// 写死}// 开发环境可以宽松return config.development.allowWildcard ?'*':'http://localhost:8080';}

消息太多堵住了怎么搞个队列

如果你的应用消息频率很高(比如实时数据同步、频繁的用户操作上报),直接无脑postMessage可能会导致消息堆积或者丢失(虽然浏览器内部有队列,但接收方处理不过来就会漏)。

救命招数:实现一个带队列和节流的通信层。

classMessageQueue{constructor(targetWindow, targetOrigin){this.target = targetWindow;this.origin = targetOrigin;this.queue =[];this.sending =false;this.batchSize =10;// 每批最多10条this.interval =16;// 约60fps的间隔}push(message){this.queue.push({...message,_id:this.generateId(),_enqueueTime: Date.now()});this.process();}asyncprocess(){if(this.sending ||this.queue.length ===0)return;this.sending =true;// 取出前batchSize条const batch =this.queue.splice(0,this.batchSize);try{// 批量发送,减少postMessage调用次数this.target.postMessage({type:'BATCH_MESSAGES',payload: batch,count: batch.length },this.origin);// 简单模拟延迟,实际可以根据接收方响应调整awaitthis.delay(this.interval);}catch(err){ console.error('发送失败:', err);// 失败的消息塞回队列头部,下次重试this.queue.unshift(...batch);}finally{this.sending =false;// 如果还有消息,继续处理if(this.queue.length >0){this.process();}}}delay(ms){returnnewPromise(resolve=>setTimeout(resolve, ms));}generateId(){return`msg_${Date.now()}_${Math.random().toString(36).substr(2,9)}`;}}// 使用const queue =newMessageQueue(iframe.contentWindow,'https://child.com');// 高频场景,比如鼠标移动事件(虽然不建议用postMessage传这个,只是示例) document.addEventListener('mousemove',(e)=>{ queue.push({type:'MOUSE_MOVE',x: e.clientX,y: e.clientY });});

生产环境出问题怎么用messageType快速定位

当你的应用有几十种消息类型,用户反馈"有时候数据不同步"或者"偶尔点了没反应",你怎么排查?靠console.log大海捞针吗?

救命招数:给通信层加上完整的日志和追踪机制,生产环境可以上报到监控系统。

classTrackedPostMessage{constructor(){this.messageHistory =[];this.maxHistory =100;// 只保留最近100条}send(targetWindow, message, targetOrigin){const record ={direction:'out',timestamp: Date.now(),type: message.type,messageId: message.id, targetOrigin,size:JSON.stringify(message).length };this.addToHistory(record);try{ targetWindow.postMessage(message, targetOrigin); record.status ='sent';}catch(err){ record.status ='error'; record.error = err.message;this.reportError(record);}}onMessage(callback){return(event)=>{const record ={direction:'in',timestamp: Date.now(),type: event.data?.type,origin: event.origin,size:JSON.stringify(event.data).length };this.addToHistory(record);// 调用实际的处理函数callback(event);};}addToHistory(record){this.messageHistory.push(record);if(this.messageHistory.length >this.maxHistory){this.messageHistory.shift();}}// 导出最近的通信记录,用于问题排查exportHistory(){returnthis.messageHistory;}reportError(record){// 上报到Sentry或其他监控系统if(window.Sentry){ window.Sentry.captureMessage('PostMessage failed',{extra: record });}}}// 使用const pm =newTrackedPostMessage();// 替换原来的发送 pm.send(iframe.contentWindow, message, targetOrigin);// 替换原来的监听 window.addEventListener('message', pm.onMessage((event)=>{// 原来的处理逻辑}));

当用户反馈问题时,你可以让他在控制台执行pm.exportHistory(),把通信记录发给你,一眼就能看出是消息没发出去,还是发出去对方没回,还是回的格式不对。

老司机的私房技巧

封装个通用通信模块别到处写重复代码

上面那些代码片段都是分散的,实际项目中你应该封装一个统一的CrossFrameMessenger类,所有iframe通信都走它。

// messenger.js - 通用跨窗口通信模块classCrossFrameMessenger{constructor(options ={}){this.name = options.name ||'unnamed';// 用于调试标识this.allowedOrigins = options.allowedOrigins ||[];this.handlers =newMap();// 消息处理器映射this.pendingRequests =newMap();// 等待响应的请求this.requestTimeout = options.requestTimeout ||5000;// 绑定监听this._boundListener =this._handleMessage.bind(this); window.addEventListener('message',this._boundListener);}// 注册消息处理器on(type, handler){this.handlers.set(type, handler);returnthis;// 链式调用}// 发送单向消息send(targetWindow, type, payload, targetOrigin){if(!targetOrigin){thrownewError('targetOrigin is required for security');}const message ={_messenger:true,// 标记为我们的消息,避免处理外部消息_type: type,_payload: payload,_id:this._generateId(),_timestamp: Date.now(),_from:this.name }; targetWindow.postMessage(message, targetOrigin);return message._id;}// 发送请求并等待响应(Promise化)asyncrequest(targetWindow, type, payload, targetOrigin){const requestId =this.send(targetWindow, type, payload, targetOrigin);returnnewPromise((resolve, reject)=>{const timeout =setTimeout(()=>{this.pendingRequests.delete(requestId);reject(newError(`Request ${type} timeout after ${this.requestTimeout}ms`));},this.requestTimeout);this.pendingRequests.set(requestId,{ resolve, reject, timeout, type });});}// 响应请求respond(event, payload){if(!event.source ||!event.origin)return;const response ={_messenger:true,_type:'RESPONSE',_requestId: event.data._id,_payload: payload,_timestamp: Date.now()}; event.source.postMessage(response, event.origin);}_handleMessage(event){// 安全校验if(!this._isAllowedOrigin(event.origin))return;const data = event.data;// 只处理我们的消息格式if(!data ||!data._messenger)return;// 处理响应if(data._type ==='RESPONSE'&&this.pendingRequests.has(data._requestId)){const pending =this.pendingRequests.get(data._requestId);clearTimeout(pending.timeout);this.pendingRequests.delete(data._requestId); pending.resolve(data._payload);return;}// 处理普通消息const handler =this.handlers.get(data._type);if(handler){handler(data._payload, event,this);}}_isAllowedOrigin(origin){if(this.allowedOrigins.length ===0)returntrue;// 开发环境returnthis.allowedOrigins.includes(origin);}_generateId(){return`${this.name}_${Date.now()}_${Math.random().toString(36).substr(2,9)}`;}destroy(){ window.removeEventListener('message',this._boundListener);// 清理pending的请求this.pendingRequests.forEach(pending=>{clearTimeout(pending.timeout); pending.reject(newError('Messenger destroyed'));});this.pendingRequests.clear();}}// 使用示例const parentMessenger =newCrossFrameMessenger({name:'parent',allowedOrigins:['http://localhost:8080']}); parentMessenger .on('CHILD_READY',(payload, event)=>{ console.log('子应用已就绪:', payload); parentMessenger.respond(event,{status:'ok'});}).on('USER_ACTION',(payload)=>{ console.log('用户行为:', payload);});// 在合适的时机发送请求asyncfunctiongetChildData(){try{const data =await parentMessenger.request( iframe.contentWindow,'GET_DATA',{type:'userInfo'},'http://localhost:8080'); console.log('拿到数据:', data);}catch(err){ console.error('请求失败:', err);}}

这个封装的好处是:

  • 统一的消息格式,带_messenger标记避免误处理
  • Promise化的request/response,告别回调地狱
  • 自动超时处理,防止请求挂起
  • 统一的origin管理和安全校验

给消息加个id方便追踪和去重

在分布式系统里,消息去重是基本功。iframe通信虽然不算分布式,但同样可能遇到消息重复(比如用户疯狂点击按钮,或者网络抖动导致重发)。

classDeduplicatedMessengerextendsCrossFrameMessenger{constructor(options){super(options);this.receivedIds =newSet();// 已处理的消息IDthis.maxHistory =1000;// 最多记1000个ID,防止内存泄漏// 定期清理旧ID(简单策略:超过一定数量就清空,实际可以用LRU)setInterval(()=>{if(this.receivedIds.size >this.maxHistory){this.receivedIds.clear();}},60000);}_handleMessage(event){const data = event.data;if(!data ||!data._id)returnsuper._handleMessage(event);// 检查是否已处理过if(this.receivedIds.has(data._id)){ console.warn('收到重复消息,忽略:', data._id);return;}this.receivedIds.add(data._id);super._handleMessage(event);}}

超时机制一定要加,不然能等到天荒地老

前面封装的request方法已经带了超时,但这里单独强调一下,因为真的太重要了。iframe里的页面可能崩溃、可能被用户关闭、可能JS报错导致没回消息,如果你不做超时处理,Promise就会永远挂在那,相关的UI状态(比如loading)也永远转圈圈。

// 在UI层配合超时处理asyncfunctionloadDataFromIframe(){const loading =showLoading();try{// 5秒超时const data =await Promise.race([ messenger.request(iframe.contentWindow,'LOAD_DATA',{}, targetOrigin),newPromise((_, reject)=>setTimeout(()=>reject(newError('timeout')),5000))]);renderData(data);}catch(err){if(err.message ==='timeout'){showError('加载超时,请刷新重试');}else{showError('加载失败: '+ err.message);}}finally{ loading.hide();}}

开发环境可以宽松点,生产环境必须严格校验

这个前面说过,但值得再写一遍代码示例:

const messengerConfig ={development:{allowedOrigins:['*'],// 开发环境为了方便,可以用*(虽然还是不推荐)logLevel:'debug',strictMode:false},production:{allowedOrigins:['https://app1.company.com','https://app2.company.com'// 绝对不允许*,必须显式列出],logLevel:'error',strictMode:true}};// 根据环境创建实例const config = messengerConfig[process.env.NODE_ENV]|| messengerConfig.production;const messenger =newCrossFrameMessenger(config);

用Channel ID做隔离,多个iframe不会打架

当一个页面嵌入多个iframe,或者iframe里又嵌iframe时,消息可能会乱窜。给每个通信通道分配唯一ID可以解决。

// 创建时分配channelIdconst messenger =newCrossFrameMessenger({channelId:'payment-module',// 唯一标识allowedOrigins:['https://pay.com']});// 消息格式里带上channelIdconst message ={_channelId:this.channelId,_type: type,_payload: payload };// 接收时校验if(data._channelId !==this.channelId){return;// 不是发给我的,忽略}

这样即使页面上有多个iframe,每个messenger实例只处理自己channel的消息,互不干扰。

最后说点实在的

能不用iframe就别用,这玩意儿本身就有性能开销

iframe虽然方便,但代价不小:

  1. 内存占用:每个iframe都是独立的渲染进程(在Chrome里,同域的可能共享,跨域的一定独立),吃内存跟吃包子似的
  2. 加载慢:iframe里的页面要完整走一遍DNS、TCP、HTTP、解析、渲染流程,主页面都渲染完了它还在那转圈
  3. SEO黑洞:搜索引擎爬虫通常不爬iframe里的内容,如果你的核心内容在iframe里,SEO基本废了
  4. 调试困难:Chrome DevTools虽然可以选iframe的context,但断点调试、性能分析都比单页面麻烦得多

所以架构设计的时候,优先考虑同域方案、微前端方案(qiankun等)、或者Web Components。iframe应该是最后的选择,而不是第一选择。

微前端现在有更香的方案比如qiankun

如果你是为了微前端用iframe,建议看看qiankun、single-spa、micro-app这些方案。它们用fetch+eval或者Web Component的方式加载子应用,没有iframe的诸多限制,还能共享依赖、样式隔离、JS沙箱,通信也有专门的initGlobalStateAPI,比裸postMessage好用多了。

当然这些方案也有坑,比如qiankun的JS沙箱在严格模式下有些兼容问题,single-spa需要子应用改造生命周期函数。但总体来说,如果是自研的微前端,优先选这些框架;只有嵌入不可控的第三方页面时,才必须用iframe。

但有些第三方页面你绕不开,该学还是得学

现实就是这么骨感。银行支付页面、政府政务系统、老旧的企业内部系统…这些第三方服务往往只提供iframe接入,你没得选。这时候postMessage就是你吃饭的家伙,不学不行。

而且说实话,掌握postMessage的底层原理,对你理解浏览器的安全模型、跨域策略、事件机制都有帮助。就算以后用不上iframe,这些知识在Web Worker、Service Worker、甚至跨Tab通信(Broadcast Channel API)里也是相通的。

代码我放GitHub了,记得star别白嫖

(这里本来应该放GitHub链接,但你开头说了禁止放网页链接,那我就不放了。你自己把上面的代码复制粘贴整理一下,就是个完整的demo项目。)

下期想听啥评论区吱一声,别让我唱独角戏

这篇写了快八千字,从原理到代码到踩坑到优化,能想到的都写了。但肯定还有遗漏的场景,或者你遇到了更奇葩的坑。想听什么话题,比如"Web Worker和iframe怎么配合"、“postMessage性能优化到每秒1000条消息”、"用postMessage实现跨Tab状态同步"之类的,留言告诉我。

写技术文章这事儿,最怕自嗨。我觉得有用的你不一定觉得有用,你觉得痛的我不一定写到了。所以多交流,别让我一个人在这巴拉巴拉。

(全文完,去写bug了,回见。)

欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。
推荐:DTcode7的博客首页。
一个做过前端开发的产品经理,经历过睿智产品的折磨导致脱发之后,励志要翻身农奴把歌唱,一边打入敌人内部一边持续提升自己,为我们广大开发同胞谋福祉,坚决抵制睿智产品折磨我们码农兄弟!

专栏系列(点击解锁)学习路线(点击解锁)知识定位
《微信小程序相关博客》持续更新中~结合微信官方原生框架、uniapp等小程序框架,记录请求、封装、tabbar、UI组件的学习记录和使用技巧等
《AIGC相关博客》持续更新中~AIGC、AI生产力工具的介绍,例如stable diffusion这种的AI绘画工具安装、使用、技巧等总结
《HTML网站开发相关》《前端基础入门三大核心之html相关博客》前端基础入门三大核心之html板块的内容,入坑前端或者辅助学习的必看知识
《前端基础入门三大核心之JS相关博客》前端JS是JavaScript语言在网页开发中的应用,负责实现交互效果和动态内容。它与HTML和CSS并称前端三剑客,共同构建用户界面。
通过操作DOM元素、响应事件、发起网络请求等,JS使页面能够响应用户行为,实现数据动态展示和页面流畅跳转,是现代Web开发的核心
《前端基础入门三大核心之CSS相关博客》介绍前端开发中遇到的CSS疑问和各种奇妙的CSS语法,同时收集精美的CSS效果代码,用来丰富你的web网页
《canvas绘图相关博客》Canvas是HTML5中用于绘制图形的元素,通过JavaScript及其提供的绘图API,开发者可以在网页上绘制出各种复杂的图形、动画和图像效果。Canvas提供了高度的灵活性和控制力,使得前端绘图技术更加丰富和多样化
《Vue实战相关博客》持续更新中~详细总结了常用UI库elementUI的使用技巧以及Vue的学习之旅
《python相关博客》持续更新中~Python,简洁易学的编程语言,强大到足以应对各种应用场景,是编程新手的理想选择,也是专业人士的得力工具
《sql数据库相关博客》持续更新中~SQL数据库:高效管理数据的利器,学会SQL,轻松驾驭结构化数据,解锁数据分析与挖掘的无限可能
《算法系列相关博客》持续更新中~算法与数据结构学习总结,通过JS来编写处理复杂有趣的算法问题,提升你的技术思维
《IT信息技术相关博客》持续更新中~作为信息化人员所需要掌握的底层技术,涉及软件开发、网络建设、系统维护等领域的知识
《信息化人员基础技能知识相关博客》无论你是开发、产品、实施、经理,只要是从事信息化相关行业的人员,都应该掌握这些信息化的基础知识,可以不精通但是一定要了解,避免日常工作中贻笑大方
《信息化技能面试宝典相关博客》涉及信息化相关工作基础知识和面试技巧,提升自我能力与面试通过率,扩展知识面
《前端开发习惯与小技巧相关博客》持续更新中~罗列常用的开发工具使用技巧,如 Vscode快捷键操作、Git、CMD、游览器控制台等
《photoshop相关博客》持续更新中~基础的PS学习记录,含括PPI与DPI、物理像素dp、逻辑像素dip、矢量图和位图以及帧动画等的学习总结
日常开发&办公&生产【实用工具】分享相关博客》持续更新中~分享介绍各种开发中、工作中、个人生产以及学习上的工具,丰富阅历,给大家提供处理事情的更多角度,学习了解更多的便利工具,如Fiddler抓包、办公快捷键、虚拟机VMware等工具

吾辈才疏学浅,摹写之作,恐有瑕疵。望诸君海涵赐教。望轻喷,嘤嘤嘤

非常期待和您一起在这个小小的网络世界里共同探索、学习和成长。愿斯文对汝有所裨益,纵其简陋未及渊博,亦足以略尽绵薄之力。倘若尚存阙漏,敬请不吝斧正,俾便精进!
在这里插入图片描述

Read more

使用Docker安装Ollama及Open-WebUI完整教程

作者:吴业亮 博客:wuyeliang.blog.ZEEKLOG.net 一、Ollama 简介及工作原理 1. Ollama 简介及原理 * 简介:Ollama 是一款轻量级、开源的大语言模型(LLM)运行工具,旨在简化本地部署和运行大语言模型的流程。它支持 Llama 3、Mistral、Gemini 等主流开源模型,用户无需复杂配置即可在本地设备(CPU 或 GPU)上快速启动模型,适用于开发测试、本地智能应用搭建等场景。 * 工作原理: * 采用模型封装机制,将大语言模型的运行环境、依赖库及推理逻辑打包为标准化格式,实现模型的一键下载、启动和版本管理。 * 通过优化的推理引擎适配硬件架构,支持 CPU 基础运行和 GPU 加速(如 NVIDIA CUDA),减少资源占用并提升响应速度。 * 提供简洁的

Apache SeaTunnel Web 完整使用指南:从零搭建可视化数据集成平台

Apache SeaTunnel Web 完整使用指南:从零搭建可视化数据集成平台 【免费下载链接】seatunnel-webSeaTunnel is a distributed, high-performance data integration platform for the synchronization and transformation of massive data (offline & real-time). 项目地址: https://gitcode.com/gh_mirrors/se/seatunnel-web Apache SeaTunnel Web 是基于 SeaTunnel Connector API 和 Zeta Engine 开发的可视化管理平台,让数据集成工作变得前所未有的简单。无论您是数据工程师、开发人员还是运维人员,这个强大的 Web 控制台都能帮助您轻松管理海量数据的同步和转换任务。

【测试理论与实践】(十)Web 项目自动化测试实战:从 0 到 1 搭建博客系统 UI 自动化框架

【测试理论与实践】(十)Web 项目自动化测试实战:从 0 到 1 搭建博客系统 UI 自动化框架

目录 前言 一、项目背景与测试规划:先明确 "测什么" 和 "怎么测" 1.1 项目介绍 1.2 测试目标 1.3 测试范围与用例设计 编辑 二、环境搭建:3 步搞定自动化测试前置准备 2.1 安装核心依赖包 2.2 浏览器配置 2.3 项目目录结构设计 三、核心模块开发:封装公共工具,提高代码复用性 3.1 驱动管理与截图工具封装(common/Utils.py) 3.2 代码说明与优化点 四、测试用例开发:

【超音速专利 CN118134841A】一种光伏产品缺陷检测AI深度学习算法

【超音速专利 CN118134841A】一种光伏产品缺陷检测AI深度学习算法

申请号CN202410053849.9公开号(公开)CN118134841A申请日2024.01.12申请人(公开)超音速人工智能科技股份有限公司(833753)发明人(公开)张俊峰(总); 叶长春(总); 廖绍伟 原文摘要 本发明公开一种光伏产品缺陷检测AI深度学习算法,涉及AI算法领域。该光伏产品缺陷检测AI深度学习算法,采用深度卷积神经网络作为预训练模型,使用特征金字塔网络结构FPN对预训练模型得到的不同尺度的特征图进行融合,采用区域提议网络RPN在特征图上生成候选框,该光伏产品缺陷检测AI深度学习算法通过使用预训练模型提取图像特征,使用FPN融合多尺度特征,使用RPN提取候选框,使用ROIAlign抽取局部特征,使用分类、回归、FCN进行缺陷分类、位置回归以及掩膜信息提取,对缺陷的分类以及输出缺陷效果的准确性好,对缺陷的定位精度高,对缺陷的描述准确且全面,从而提高了在光伏产品加工中,对产品的缺陷检测效果。 术语 FCN指的是全卷积网络,是深度学习中用于图像处理任务的一种重要架构,相比于传统的卷积神经网络CNN,FCN不仅能够识别图像中的对象,还能在像素级