微信小程序webview postmessage通信完整配置和使用

 更新时间:2026年02月04日 09:01:03   作者:零凌林  
这篇文章主要介绍了微信小程序webview postmessage通信完整配置和使用的相关资料,Webview组件作为小程序中能够与原生页面进行通信的桥梁,更是受到了广大开发者的关注,需要的朋友可以参考下

需求概述

在微信小程序中使用 web-view 组件与内嵌网页进行双向通信,主要通过 postMessage 实现。以下是完整的配置和使用方法:

通信指南

微信小程序webview官方文档

1. 基础配置

小程序端配置

// app.json 或 page.json
{
  "usingComponents": {},
  "permission": {
    "scope.webView": {
      "desc": "用于网页和小程序通信"
    }
  }
}

网页端配置

<!-- 内嵌网页需引入微信JS-SDK -->
<script src="https://res.wx.qq.com/open/js/jweixin-1.6.0.js"></script>
<!-- 或使用官方最新版本 -->

2. 通信实现

方案一:小程序向网页发送消息

小程序端代码:

// page.js
Page({
  data: {
    webViewUrl: 'https://your-domain.com/page.html'
  },

  onLoad() {
    // 通过 web-view 组件的 ref 发送消息
  },

  // 向网页发送消息
  sendToWebPage() {
    const webview = this.selectComponent('#myWebview')
    webview.postMessage({
      data: {
        type: 'from_miniprogram',
        message: 'Hello from Mini Program!',
        timestamp: Date.now()
      }
    })
  },

  // 接收网页消息
  onMessage(e) {
    console.log('收到网页消息:', e.detail.data)
    // 处理网页发送的数据
    const { type, data } = e.detail.data
    if (type === 'from_web') {
      // 处理逻辑
    }
  }
})
<!-- page.wxml -->
<web-view 
  id="myWebview"
  src="{{webViewUrl}}"
  bindmessage="onMessage"
  bindload="onWebViewLoad"
/>
<button bindtap="sendToWebPage">发送消息到网页</button>

网页端代码:

// 监听小程序消息
document.addEventListener('message', function(e) {
  const data = e.data
  console.log('收到小程序消息:', data)
  
  // 处理消息
  if (data.type === 'from_miniprogram') {
    // 执行相应操作
    
    // 回复消息给小程序
    if (window.wx && window.wx.miniProgram) {
      window.wx.miniProgram.postMessage({
        data: {
          type: 'from_web',
          reply: 'Message received!',
          original: data.message
        }
      })
    }
  }
})

// 主动发送消息到小程序
function sendToMiniProgram() {
  if (window.wx && window.wx.miniProgram) {
    window.wx.miniProgram.postMessage({
      data: {
        type: 'user_action',
        action: 'button_click',
        value: 'some_value',
        timestamp: new Date().getTime()
      }
    })
  }
}

3. 完整通信示例

双向通信封装

小程序端封装:

// utils/webviewBridge.js
class WebViewBridge {
  constructor(webviewRef) {
    this.webview = webviewRef
    this.messageHandlers = new Map()
  }

  // 发送消息到网页
  postMessage(type, data) {
    if (!this.webview) return false
    
    this.webview.postMessage({
      data: {
        type,
        payload: data,
        timestamp: Date.now(),
        source: 'miniprogram'
      }
    })
    return true
  }

  // 注册消息处理器
  onMessage(type, handler) {
    this.messageHandlers.set(type, handler)
  }

  // 处理接收到的消息
  handleMessage(event) {
    const { type, payload, source } = event.detail.data
    if (source === 'web') {
      const handler = this.messageHandlers.get(type)
      if (handler) {
        handler(payload)
      }
    }
  }

  // 移除处理器
  offMessage(type) {
    this.messageHandlers.delete(type)
  }
}

export default WebViewBridge

网页端封装:

// webview-bridge.js
class MiniProgramBridge {
  constructor() {
    this.handlers = new Map()
    this.init()
  }

  init() {
    // 监听小程序消息
    document.addEventListener('message', (e) => {
      const { type, payload, source } = e.data
      if (source === 'miniprogram') {
        this.dispatch(type, payload)
      }
    })

    // 监听页面卸载
    window.addEventListener('beforeunload', () => {
      this.postMessage('page_unload', {})
    })
  }

  // 发送消息到小程序
  postMessage(type, data) {
    if (window.wx && window.wx.miniProgram) {
      window.wx.miniProgram.postMessage({
        data: {
          type,
          payload: data,
          timestamp: Date.now(),
          source: 'web'
        }
      })
      return true
    }
    return false
  }

  // 注册消息处理器
  on(type, handler) {
    if (!this.handlers.has(type)) {
      this.handlers.set(type, [])
    }
    this.handlers.get(type).push(handler)
  }

  // 分发消息
  dispatch(type, data) {
    const typeHandlers = this.handlers.get(type)
    if (typeHandlers) {
      typeHandlers.forEach(handler => handler(data))
    }
  }

  // 移除处理器
  off(type, handler) {
    const typeHandlers = this.handlers.get(type)
    if (typeHandlers) {
      const index = typeHandlers.indexOf(handler)
      if (index > -1) {
        typeHandlers.splice(index, 1)
      }
    }
  }
}

// 创建全局实例
window.MiniProgramBridge = new MiniProgramBridge()

4. 使用示例

小程序页面使用:

import WebViewBridge from '../../utils/webviewBridge'

Page({
  data: {
    url: 'https://example.com'
  },

  onLoad() {
    // 在 web-view 加载完成后初始化
  },

  onWebViewLoad() {
    const webview = this.selectComponent('#webview')
    this.bridge = new WebViewBridge(webview)
    
    // 注册消息处理器
    this.bridge.onMessage('user_login', (data) => {
      console.log('用户登录:', data)
      // 处理登录逻辑
    })

    this.bridge.onMessage('payment_success', (data) => {
      console.log('支付成功:', data)
      wx.showToast({
        title: '支付成功'
      })
    })
  },

  // 发送用户信息到网页
  sendUserInfo() {
    this.bridge.postMessage('user_info', {
      userId: '123',
      nickname: '张三',
      avatar: 'url'
    })
  }
})

网页端使用:

<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
</head>
<body>
  <button onclick="sendMessage()">发送消息到小程序</button>
  
  <script src="https://res.wx.qq.com/open/js/jweixin-1.6.0.js"></script>
  <script src="webview-bridge.js"></script>
  <script>
    // 监听小程序消息
    MiniProgramBridge.on('user_info', (data) => {
      console.log('收到用户信息:', data)
      document.getElementById('user-name').innerText = data.nickname
    })

    // 发送消息到小程序
    function sendMessage() {
      MiniProgramBridge.postMessage('button_click', {
        buttonId: 'submit',
        value: 'confirmed'
      })
    }

    // 页面加载完成通知小程序
    window.addEventListener('load', () => {
      MiniProgramBridge.postMessage('page_loaded', {
        title: document.title,
        url: window.location.href
      })
    })
  </script>
</body>
</html>

5. 最佳实践

// 1. 添加消息类型常量
const MessageTypes = {
  USER_INFO: 'user_info',
  PAYMENT: 'payment',
  NAVIGATION: 'navigation',
  ERROR: 'error'
}

// 2. 添加超时处理
function postMessageWithTimeout(type, data, timeout = 5000) {
  return new Promise((resolve, reject) => {
    const timer = setTimeout(() => {
      reject(new Error('Message timeout'))
    }, timeout)
    
    // 发送消息逻辑
  })
}

// 3. 错误处理
try {
  webview.postMessage(data)
} catch (error) {
  console.error('发送消息失败:', error)
  // 重试或降级处理
}

6. 注意事项

1.确保网页域名已在小程序后台配置

2.处于安全性的考虑,始终验证接收到的数据,并且避免执行不可信的脚本,防止xss攻击

3.避免高频发送(>100条/秒)

4.确保网页端微信JS-SDK版本兼容

5.微信小程序 WebView 的 bindmessage 事件有一个延迟触发机制,网页向小程序发送的消息不会立即触发 bindmessage,而是在特定时机批量触发。网页向小程序 postMessage 时,会在以下特定时机触发并收到消息:小程序后退、组件销毁、分享、复制链接。e.detail = { data },data是多次 postMessage 的参数组成的数组。微信小程序webview官方文档

7.实时通信方案

最佳实践:对于需要实时通信的场景,应该使用 WebSocket、URL 参数、Storage 轮询等替代方案,bindmessage 更适合作为最终的数据提交机制。

// 方案1:使用URL参数传递状态(实时)
// 网页端
function updateState(state) {
  // 通过修改hash或reload传递数据
  window.location.hash = 'state=' + encodeURIComponent(JSON.stringify(state))
  
  // 或者使用URL参数
  const newUrl = window.location.pathname + '?data=' + encodeURIComponent(JSON.stringify(state))
  window.history.replaceState({}, '', newUrl)
}

// 方案2:使用Storage同步(需要轮询)
// 网页端
function sendViaStorage(data) {
  localStorage.setItem('web_to_miniprogram', JSON.stringify({
    data,
    timestamp: Date.now()
  }))
}

// 小程序端(需要轮询)
setInterval(() => {
  const storage = wx.getStorageSync('web_to_miniprogram')
  if (storage) {
    this.handleMessage(storage.data)
    wx.removeStorageSync('web_to_miniprogram')
  }
}, 100)

// 方案3:使用WebSocket(真正实时)
const ws = new WebSocket('wss://your-server.com')
ws.onmessage = (e) => {
  // 小程序通过后端中转消息
}

使用URL 参数为什么能达到实时的效果?

使用 URL 参数 可以达到实时效果,是因为 WebView 的 URL 变化会立即触发小程序端的 bindload 事件(页面加载或重载),而 bindload 是实时触发的。这与 bindmessage 的延迟触发机制完全不同。

拓展

什么是webview?它的作用是什么?

WebView(网页视图)可以简单理解为一个嵌入在应用程序(App或者小程序)内部的“浏览器”。它不是一个独立的浏览器(如 Chrome 或 Safari),而是一个让 App或者小程序 能够显示和处理网页内容的控件或组件。

webview是“原生”与“网页”的结合体

在移动和桌面应用开发中,通常有两种开发方式:

  • 原生开发 (Native): 为特定平台(如 iOS、Android或者小程序)专门编写的代码,性能最好,但开发成本高。
  • Web 开发: 使用 HTML、CSS 和 JavaScript 编写的网页,跨平台能力强,但性能相对较弱。

WebView 处于两者之间。它允许开发者在原生 App或者小程序 中嵌入网页内容。例如,你在一个电商 App 里看到的商品详情页,其实可能就是一个加载了网页的 WebView,而不是用原生代码画出来的界面。

WebView 的作用非常广泛,主要体现在以下几个方面:

实现混合式开发 (Hybrid Development)

这是 WebView 最主要的作用。很多 App或者小程序的页面并不是完全用原生代码写的,而是部分界面使用 WebView 加载网页。

  • 优势: 开发者可以使用一套网页代码(HTML/JS),同时在 Android 和 iOS 或者小程序上运行,大大降低了开发和维护成本。
  • 常见框架: 像 uni-app、React Native 等框架底层都利用了 WebView 技术。

动态更新,无需发版

这是 WebView 相比原生代码的巨大优势。

  • 原生 App或者小程序: 如果想修改界面或逻辑,必须修改代码 -> 重新打包 -> 提交平台审核 -> 用户下载更新。
  • WebView: 只需要在服务器端更新网页代码(HTML/JS),下次用户打开页面时,看到的就是最新的内容,用户无需重新下载安装 App。这非常适合需要频繁更新活动页面(如促销、广告页)的场景。

强大的交互能力

WebView 不仅仅是一个“看”网页的工具,它还能和原生系统进行深度交互(Native-Web 互操作):

  • 调用系统功能: 网页可以通过 WebView 调用手机的摄像头、相册、地理位置(GPS)等原生功能。
  • 数据互通: 原生代码可以往网页里传数据,网页也可以把用户操作的结果回传给原生代码。

跨平台兼容

在 Windows、macOS、Android、iOS 等不同平台上,WebView 都能提供一致的网页渲染体验。例如,Microsoft Edge WebView2 控件允许开发者在桌面应用中使用最新的 Chromium 引擎渲染网页。

简单来说,WebView 就是 App 里的一个“窗口”,用来展示网页内容。它的核心价值在于:让 App 既能拥有原生的系统权限和性能,又能具备网页的灵活更新和跨平台能力。你在日常使用 App 时,那些偶尔需要“加载”的页面,或者下拉刷新就能看到新内容的界面,很大概率都是 WebView 在背后工作。

总结

到此这篇关于微信小程序webview postmessage通信完整配置和使用的文章就介绍到这了,更多相关微信小程序webview postmessage通信内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Python中如何处理HTTPS请求和SSL证书验证详解

    Python中如何处理HTTPS请求和SSL证书验证详解

    Python 的ssl模块提供了 SSL/TLS 协议的实现,用于加密网络通信,这篇文章主要介绍了Python中如何处理HTTPS请求和SSL证书验证的相关资料,需要的朋友可以参考下
    2025-06-06
  • 一文让你彻底弄懂js中undefined和null的区别

    一文让你彻底弄懂js中undefined和null的区别

    JavaScript是一门动态类型语言,元素除了表示存在的空值外,还有可能根本就不存在,这就是undefined存在的原因,这篇文章主要给大家介绍了关于undefined和null区别的相关资料,需要的朋友可以参考下
    2022-03-03
  • 何时使用Map来代替普通的JS对象

    何时使用Map来代替普通的JS对象

    这篇文章主要介绍了何时使用Map来代替普通的JS对象,对Map感兴趣的同学,可以参考下
    2021-04-04
  • 解决layUI的页面显示不全的问题

    解决layUI的页面显示不全的问题

    今天小编就为大家分享一篇解决layUI的页面显示不全的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-09-09
  • html的DOM中Event对象onabort事件用法实例

    html的DOM中Event对象onabort事件用法实例

    这篇文章主要介绍了html的DOM中Event对象onabort事件用法,实例分析了onabort事件的适用范围与对应的javascript使用技巧,需要的朋友可以参考下
    2015-01-01
  • 微信小程序使用第三方库Immutable.js实例详解

    微信小程序使用第三方库Immutable.js实例详解

    Immutable 是 Facebook 开发的不可变数据集合。不可变数据一旦创建就不能被修改,是的应用开发更简单,允许使用函数式编程技术,比如惰性评估。微信小程序无法直接使用Immutable.js,下面就来说说微信小程序如何使用第三方库Immutable.js。
    2016-09-09
  • js之如何筛选出两个数组相同的值

    js之如何筛选出两个数组相同的值

    这篇文章主要介绍了js之如何筛选出两个数组相同的值问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-06-06
  • JS实现头条新闻的经典轮播图效果示例

    JS实现头条新闻的经典轮播图效果示例

    这篇文章主要介绍了JS实现头条新闻的经典轮播图效果,涉及javascript图片轮播切换相关实现技巧,需要的朋友可以参考下
    2019-01-01
  • JavaScript中常用的字符串方法函数操作方法总结

    JavaScript中常用的字符串方法函数操作方法总结

    这篇文章主要介绍了JavaScript中所有的字符串函数操作方法整理汇总,包括字符串的长度、连接、查找、截取、替换、分隔、转换等处理方法,以及网址中获取文件名等等,需要的朋友可以参考下
    2023-12-12
  • 基于Bootstrap框架菜鸟入门教程(推荐)

    基于Bootstrap框架菜鸟入门教程(推荐)

    下面小编就为大家带来一篇基于Bootstrap框架菜鸟入门教程(推荐)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-09-09

最新评论