window.postMessage学习(问答式)

2024-03-12 17:20

本文主要是介绍window.postMessage学习(问答式),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

demo在文章末尾。

window.postMessage 


1、有什么用?

答:可以安全地实现跨源通信。通常,对于两个不同页面的脚本,只有当执行它们的页面同源时,这两个脚本才能相互通信。
window.postMessage() 方法提供了一种受控机制来规避此限制,只要正确的使用,这种方法就很安全。 

2、两个页面如何算同源?

答:在JavaScript中,同源(Same-Origin)是指两个页面的协议、域名和端口都相同。这是浏览器安全策略的一部分,用于限制不同源之间的交互,以防止潜在的安全风险。同源策略(Same-Origin Policy)是浏览器的一种安全机制,它要求浏览器只允许脚本访问与其自身源相同的资源。这意味着,如果一个JavaScript脚本在一个域名下运行,那么它只能访问该域名下的资源,不能访问其他域名下的资源,除非这两个域名之间有明确的跨域资源共享(CORS)设置。

具体来说,两个页面同源需要满足以下条件:

协议相同:例如,两个页面都使用http://或https://。
域名相同:例如,两个页面都是example.com或子域名如sub.example.com。
端口相同:如果两个页面使用不同的端口号,则它们不是同源的。默认情况下,http://的端口是80,https://的端口是443,但如果明确指定了其他端口号,则必须相同。
如果两个页面不满足上述条件中的任何一个,它们就被认为是不同源的,浏览器会阻止它们之间的一些交互行为,例如:

使用XMLHttpRequest或fetch发送跨域请求。在一个窗口中通过document.domain属性来访问另一个窗口的文档对象模型(DOM)。使用window.open、window.parent、window.top等窗口对象属性来访问或操作不同源的窗口。但是,为了支持一些合理的跨域交互需求,浏览器提供了跨域资源共享(CORS)等机制,允许服务器明确指定哪些源可以访问其资源。此外,postMessage方法是一个例外,它允许不同源的窗口之间安全地传递消息,只要正确地设置消息的来源和目标。

3、targetWindow.postMessage(message, targetOrigin, [transfer]); 语法说明:

(1)targetWindow指的是什么?
在JavaScript中,targetWindow 是一个对另一个浏览器窗口或者iframe的引用。当你想要从一个窗口向另一个窗口发送消息时,你需要有对这个目标窗口的引用。
有几种方式可以获取这样的引用: 


(1.1)iframe的contentWindow属性:如果你有一个<iframe>元素在你的页面上,你可以通过它的contentWindow属性获取对该iframe内部文档的引用。例如:

<iframe id="myIframe" src="http://example.com/iframe-content.html"></iframe>
<script>var iframe = document.getElementById('myIframe');var targetWindow = iframe.contentWindow;// 现在你可以使用targetWindow来向iframe发送消息targetWindow.postMessage('Hello from parent!', '*');
</script>


(1.2)window.open返回的窗口对象:当你使用window.open方法打开一个新的窗口或标签页时,这个方法会返回一个对新打开窗口的引用。你可以使用这个引用来与新窗口通信。例如:

var targetWindow = window.open('http://example.com/new-page.html', '_blank');
// 使用targetWindow向新窗口发送消息
targetWindow.postMessage('Welcome to the new page!', 'http://example.com');


(1.3)window.frames数组或命名窗口:如果你的页面包含多个iframe或frames,你可以通过window.frames数组来访问它们。每个元素都是对应iframe或frame的窗口对象。此外,如果iframe或frame有name属性,你还可以通过window.frames['name']来访问它。例如:

<iframe name="myFrame" src="http://example.com/frame-content.html"></iframe>
<script>var targetWindow = window.frames['myFrame'];// 或者使用数组索引,如果iframe是第一个的话// var targetWindow = window.frames[0];// 使用targetWindow向iframe发送消息targetWindow.postMessage('Message from main page!', '*');
</script>


(2)message,将要发送到目标 window 的数据。字符串,对象等都可以。
(3)targetOrigin
通过窗口的 origin 属性来指定哪些窗口能接收到消息事件,其值可以是字符串"*"(表示无限制)或者一个 URI。在发送消息的时候,如果目标窗口的协议、主机地址或端口这三者的任意一项不匹配 targetOrigin 提供的值,那么消息就不会被发送;只有三者完全匹配,消息才会被发送。这个机制用来控制消息可以发送到哪些窗口;例如,当用 postMessage 传送密码时,这个参数就显得尤为重要,必须保证它的值与这条包含密码的信息的预期接受者的 origin 属性完全一致,来防止密码被恶意的第三方截获。如果你明确的知道消息应该发送到哪个窗口,那么请始终提供一个有确切值的 targetOrigin,而不是 *。不提供确切的目标将导致数据泄露到任何对数据感兴趣的恶意站点。
(4)[transfer]      (常规场景下用不到该属性)
[transfer]是一个可选参数,它是一个包含可传输对象(transferable objects)的数组。这个参数允许你在发送消息时,将某些特定类型的对象的所有权从发送方直接转移给接收方,而不是进行深拷贝。可传输对象通常指的是那些包含大量数据,并且深拷贝成本较高的对象,比如ArrayBuffer、Map、Set、TypedArray(如Int8Array、Uint8Array等)以及某些特定类型的ImageBitmap。
通过将这些对象的所有权直接转移给接收方,可以大大提高数据传输的效率,减少内存使用,并提升性能。当你传递一个对象到[transfer]数组时,该对象的所有权将被转移,发送方将不再拥有该对象,而接收方将获得对该对象的完全所有权。这意味着发送方在传输之后不能再访问或修改该对象,而接收方可以自由地使用和修改它。
下面是一个使用[transfer]参数的例子:

// 假设我们有一个名为otherWindow的引用,它指向另一个窗口或iframe
var otherWindow = window.open('http://example.com/other-page.html', '_blank');// 创建一个ArrayBuffer对象,它包含一些数据
var arrayBuffer = new ArrayBuffer(16);
var int32View = new Int32Array(arrayBuffer);
for (var i = 0; i < int32View.length; i++) {int32View[i] = i;
}// 使用postMessage发送ArrayBuffer,并将所有权转移给接收方
otherWindow.postMessage({ data: arrayBuffer }, 'http://example.com', [arrayBuffer]);// 在发送之后,发送方不能再访问arrayBuffer,因为它的所有权已经被转移了
// 尝试访问可能会导致错误
// console.log(int32View[0]); // Uncaught TypeError: Failed to read the '0' property from 'Int32Array': The object is no longer valid.// 在接收方窗口中,可以通过监听message事件来接收消息和转移的对象
otherWindow.addEventListener('message', function(event) {if (event.origin !== 'http://your-origin.com') {return; // 忽略不是来自可信源的消息}// 接收方现在拥有arrayBuffer的所有权,并可以自由地访问和修改它var receivedArrayBuffer = event.data.data;var receivedInt32View = new Int32Array(receivedArrayBuffer);console.log(receivedInt32View[0]); // 输出:0// 接收方可以修改ArrayBuffer,但这不会影响发送方,因为所有权已经转移了receivedInt32View[0] = 42;
});

在上面的例子中,arrayBuffer的所有权被转移给了otherWindow。在发送方窗口中,一旦postMessage调用完成,arrayBuffer和任何基于它的视图(如int32View)都将变得不可用,尝试访问它们会导致错误。而在接收方窗口中,event.data.data将是一个新的ArrayBuffer对象,其内容是发送方arrayBuffer的副本,并且接收方可以自由地修改和使用它。
请注意,不是所有类型的对象都可以被传输。只有那些实现了[[Transfer]]内部方法的对象才能被传输,这通常限于上面提到的那些特定类型的对象。
尝试传输不支持的对象将导致错误。

4、targetWindow能不指定吗?

答:如果不指定接收消息的窗口,那么消息只能发往本窗口,只有本窗口才会接收到信息。

5、iframe和window都是窗口吗?

答:window对象表示浏览器窗口,是JavaScript中最顶层的对象。iframe元素是HTML中的内联框架元素,可以在当前窗口中嵌套另一个窗口,因此iframe也可以被视为一个窗口。在JavaScript中,可以通过window对象来访问和操作当前窗口,而iframe元素也有自己的window对象,可以通过iframe的contentWindow属性来访问和操作iframe窗口。 

6、监听分发的 message 事件

window.addEventListener("message", receiveMessage, false);
function receiveMessage(event) {var origin = event.origin;// 验证了所受到信息的 origin (任何时候你都应该这样做)if (origin !== "http://example.org:8080") return;console.log('来信内容如下:', event.data)// 把 event.source作为回信的对象,再把 event.origin 作为 targetOriginevent.source.postMessage("谢谢,我收到你的来信了",event.origin,);
}

event的属性有
data:
从其他window传递过来的message数据
origin:
调用 postMessage 时消息发送方窗口的 origin . 这个字符串由 协议、“://“、域名、“ : 端口号”拼接而成。
例如“https://example.org (隐含端口 443)”、“http://example.net (隐含端口 80)”、“http://example.com:8080”。
source:
对发送消息的窗口对象的引用; 你可以使用此来在具有不同 origin 的两个窗口之间建立双向通信。

主页面和iframe子页面使用postMessage通讯示例:

(注意:不要用vscode自带的live preview插件打开页面,live preview打开时发现不知从哪触发的各种postMessage信息,推荐用live server插件打开页面)

index.html

<!DOCTYPE html>
<html><head><title>Main Page</title>
</head><body><h1>Welcome to the Main Page!</h1><iframe src="iframe.html" id="myIframe" width="400" height="200"></iframe><script>// 获取iframe的引用var iframe = document.getElementById('myIframe')// 监听来自iframe的消息window.addEventListener('message', function (event) {// console.log(event)// 这里仅作示例,实际应用中务必对event.origin进行验证// if (event.origin !== 'http://example.com') {//   // 验证消息的来源//   return// }alert('Received message from iframe: ' + event.data)// 回复iframeiframe.contentWindow.postMessage('Hello from Parent!', '*')}, false)// 发送消息给iframefunction sendMessageToIframe () {iframe.contentWindow.postMessage('Hello from Parent!', '*')}</script><button onclick="sendMessageToIframe()">发送消息给iframe</button>
</body></html>

iframe.html

<!DOCTYPE html>
<html><head><title>Iframe Page</title>
</head><body><h1>Welcome to the Iframe Page!</h1><button onclick="sendMessageToParent()">发送消息给parent</button><script>function sendMessageToParent () {// 向父页面发送消息parent.postMessage('Hello from Iframe!', '*')}// 监听来自父页面的消息window.addEventListener('message', function (event) {// console.log(event)// 这里仅作示例,实际应用中务必对event.origin进行验证// if (event.origin !== 'http://example.com') {//   // 验证消息的来源//   return// }alert('Received message from parent: ' + event.data)}, false);</script>
</body></html>

这篇关于window.postMessage学习(问答式)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/802008

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

线性代数|机器学习-P36在图中找聚类

文章目录 1. 常见图结构2. 谱聚类 感觉后面几节课的内容跨越太大,需要补充太多的知识点,教授讲得内容跨越较大,一般一节课的内容是书本上的一章节内容,所以看视频比较吃力,需要先预习课本内容后才能够很好的理解教授讲解的知识点。 1. 常见图结构 假设我们有如下图结构: Adjacency Matrix:行和列表示的是节点的位置,A[i,j]表示的第 i 个节点和第 j 个

Node.js学习记录(二)

目录 一、express 1、初识express 2、安装express 3、创建并启动web服务器 4、监听 GET&POST 请求、响应内容给客户端 5、获取URL中携带的查询参数 6、获取URL中动态参数 7、静态资源托管 二、工具nodemon 三、express路由 1、express中路由 2、路由的匹配 3、路由模块化 4、路由模块添加前缀 四、中间件