当前位置:   article > 正文

微信小程序页面之间传参的几种方式_微信小程序方法传参

微信小程序方法传参

目录

前言

第一种:url传值

url传值使用详细说明

api跳转

组件跳转

第二种:将值缓存在本地,再从本地取值

第三种:全局传值(应用实例传值)

第四种:组件传值

第五种:使用通信通道(通信通道是wx.navitageTo()独有的)

第六中:使用页面栈(只对当前页面栈中存在的页面生效)

总结


前言

  •  由于经常需要进行页面间传参且各种传参的业务场景也不相同,根据官方文档和日常工作进行了总结,共有六种传参方式。

概览

方式优点缺点
globalData双向传参、全应用可用不及时
storage双向传参、全应用可用不及时
路由简单方便、及时正向传参
通信通道双向传参、及时仅 wx.navagateTo() 接口调用才可用
页面栈可操作数据和函数、及时反向传参、仅 wx.navagateTo() 和<navigator open-type="navigateTo" url="/bbb?id=1"></navigator〉才可用

第一种:url传值

  • A页面部分js代码

  1.  Page({
  2.    toDetailPage: function (e) {
  3.      var name = "斗帝蓝电霸王龙Pro";
  4.      wx.navigateTo({
  5.        url: '/pages/detail/detail?name =' + name,
  6.     }
  7.  })
  • B页面部分js代码

  1.  Page({
  2.    /**
  3.     * onLoad生命周期函数--监听页面加载
  4.     */
  5.    onLoad: function (option) {
  6.      console.log(option); // 斗帝蓝电霸王龙Pro
  7.     /**
  8.     * 这种普通函数中this的指向是动态的,为了保证函数内部可以访问到组件实例
  9.     * 这里将this赋值给that
  10.     * 如果你想直接使用this,可以使用箭头函数,箭头函数没有自己的this
  11.     * 他会继承外层作用域的this
  12.     * 在箭头函数中this的指向是定义的上下文,而不是执行时的上下文(`不清楚的可以看001例子说明`)
  13.     */
  14.      let that = this
  15.      that.setData({
  16.        name:option
  17.     })
  18.   },
  19.  })
  1.  // 001
  2.  // 举个例子来说明:
  3.  const obj = {
  4.    name: '斗帝蓝电霸王龙Pro',
  5.    sayHello: function() {
  6.      console.log(`Hello, ${this.name}!`);
  7.   },
  8.    sayHelloArrow: () => {
  9.      console.log(`Hello, ${this.name}!`);
  10.   }
  11.  };
  12.  ​
  13.  obj.sayHello(); // 输出:Hello, 斗帝蓝电霸王龙Pro!
  14.  obj.sayHelloArrow(); // 输出:Hello, undefined!
  15.  ​
  16.  // 下面我来分析一下这段代码
  17.     /**
  18.     * sayHelloArrow是一个箭头函数。当我们调用obj.sayHello()时,
  19.     * this指向的是调用该方法的对象obj,所以输出的结果是Hello, 斗帝蓝电霸王龙Pro!。
  20.     * 而当我们调用obj.sayHelloArrow()时,由于箭头函数的this是在定义时确定的,
  21.     * 而定义sayHelloArrow时的上下文是全局上下文,
  22.     * 所以this.name实际上是指向全局对象(在浏览器中是window对象)的name属性,而全局对象
  23.     * 没有name属性,所以输出的结果是Hello, undefined!。`
  24.     */

url传值使用详细说明

  • 说明:url上直接携带参数长度是有限的且不支持特殊符号,可用编码、解码的方式解决, 在传参之前进行编码encodeURIComponent(), 接收的时候进行解码decodeURIComponent()。

api跳转

  • A页面部分js代码

  1.  //跳转到 tabBar 页面,并关闭其他所有非 tabBar 页面
  2.  wx.switchTab({
  3.      url:'/pages/detail/detail?id=5213828',
  4.  })
  5.  //关闭所有页面,打开到应用内的某个页面
  6.  wx.reLaunch({
  7.    url: '/pages/detail/detail?id=5213828',
  8.  })
  9.  //关闭当前页面,跳转到应用内的某个页面。但是不允许跳转到 tabbar 页面
  10.  wx.redirectTo({
  11.    url: '/pages/detail/detail?id=5213828',
  12.  })
  13.  //保留当前页面,跳转到应用内的某个页面。但是不能跳到 tabbar 页面。
  14.  wx.navigateTo({
  15.    url: '/pages/detail/detail?id='+ encodeURIComponent(this.data.id),
  16.  })
  17.  //以上四种路由方式在传参和接参上没有任何区别
  • B页面部分js代码

  1.  Page({
  2.    //地址传参带过来的参数只能在该方法的options参数中获取
  3.    onLoad:function(options){
  4.      console.log(decodeURIComponent(options.id)) // '5213828' ,无论传的变量是数字、布尔还是对象接收的时候都是字符串
  5.   }
  6.  })

组件跳转

  • A页面部分wxml

  1.  <view>
  2.    <navigator open-type="switchTab" url="/pages/detail/detail?id=5213828">
  3.    跳转到 tabBar 页面,并关闭其他所有非 tabBar 页面
  4.    </navigator>
  5.    <navigator open-type="reLaunch" url="/pages/detail/detail?id=5213828">
  6.    关闭所有页面,打开到应用内的某个页面
  7.    </navigator>
  8.    <navigator open-type="redirectTo" url="/pages/detail/detail?id=5213828">
  9.    关闭当前页面,跳转到应用内的某个页面。但是不允许跳转到 tabbar 页面
  10.    </navigator>
  11.    <navigator open-type="navigateTo" url="/pages/detail/detail?id=5213828">
  12.    保留当前页面,跳转到应用内的某个页面。但是不能跳到 tabbar 页面
  13.    </navigator>
  14.  </view>
  15.  //以上四种路由方式在传参和接参上没有任何区别
  • B页面部分js代码

  1.  Page({
  2.    //地址传参带过来的参数只能在该方法的options参数中获取
  3.    onLoad:function(options){
  4.      console.log(options.id) //'5213828' ,无论传的变量是数字、布尔还是对象接收的时候都是字符串
  5.   }
  6.  })

第二种:将值缓存在本地,再从本地取值

  • A页面部分js代码

  1.  Page({
  2.    toDetailPage: function (e) {
  3.      var name = "斗帝蓝电霸王龙Pro";
  4.      wx.setStorageSync("name", name);
  5.      wx.navigateTo({
  6.        url: '/pages/detail/detail'
  7.     })
  8.   }
  9.  })
  • B页面部分js代码

  1.  Page({
  2.   /**
  3.     * onLoad生命周期函数--监听页面加载
  4.     */
  5.    onLoad: function (option) {
  6.      var name = wx.getStorageSync("name");
  7.      console.log(name); // 斗帝蓝电霸王龙Pro
  8.   }
  9.  })

第三种:全局传值(应用实例传值)

  • app.js页面代码

  1.  App({
  2.    globalData: {
  3.      name: '斗帝蓝电霸王龙'
  4.   },
  5.    // onLaunch 小程序启动时执行1次,常用于获取场景值或者启动时的一些参数(如自定义分享)
  6.    onLaunch() {
  7.    // 读取 token
  8.      this.getToken()
  9.   },
  10.    getToken() {
  11.      // 读取本地的token
  12.      // this 指向应用实例本身
  13.      this.token = wx.getStorageSync('token')
  14.      this.refreshToken = wx.getStorageSync('refreshToken')
  15.   },
  16.    setToken(key, token) {
  17.      // 存储到应例实例当中
  18.      this[key] = token
  19.      // 存储到本地存储当中
  20.      wx.setStorageSync(key, token)
  21.   }
  22.  })
  • 其他页面js代码

  1.  // 获取应用实例
  2.  const app = getApp();
  3.  Page({
  4.    onLoad: function (option) {
  5.      console.log(app.globalData.name); // 斗帝蓝电霸王龙
  6.      // 重新存储新的 token
  7.      app.setToken('token', '斗帝蓝电霸王龙Pro')
  8.      app.setToken('refreshToken', '斗帝蓝电霸王龙Pro')
  9.   },
  10.    getToken() {
  11.      console.log(app.token);  // 斗帝蓝电霸王龙Pro
  12.      console.log(app.refreshToken); // 斗帝蓝电霸王龙Pro
  13.   }
  14.  })

第四种:组件传值

  • 父传子

    • 父组件

      1.  // 父组件的wxml
      2.  <classDetail  detail="{{name}}"></classDetail>
      3.  // 父组件的js
      4.  data:{
      5.      name:"父组件---斗帝蓝电霸王龙Pro"
      6.   }
      7.  ​
      8.  // 父组件的json(要把子组件注册进去)
      9.  "usingComponents": {
      10.      "classDetail": "./components/classDetail/index"
      11.   }
    • 子组件

      1.  // 子组件的js
      2.    properties:{
      3.      detail: String
      4.   }
      5.  ​
      6.  // 子组件的wxml
      7.  <view>{{detail}}<view>  // 父组件---斗帝蓝电霸王龙Pro子传父
  • 子传父

    • 子组件

      1.  // 子组件.wxml
      2.  <button bindtap="changeSelect">添加</button>
      3.  ​
      4.  // 子组件的js
      5.  changeSelect(){
      6.    // 在子组件当中通过this.triggerEvent("自定义事件名",要传递的数据)
      7.    this.triggerEvent("updataSelect", '子组件---斗帝蓝电霸王龙Pro')
      8.  }
    • 父组件

      1.  // 父组件的wxml
      2.  // 在父组件的子标签中声明自定义事件 bind:自定义事件名="事件名"
      3.  // 注意,这里的changeSelect是父组件js里的方法名称,updataSelect是子组件js里传递的方法名称
      4.  <shudanbook index="{{ index }}" bind:updataSelect="changeSelect" ></shudanbook>
      5.  ​
      6.  // 父组件的js
      7.    changeSelect(e){
      8.    //获得传递过来的值
      9.      console.log(e.detail) // 子组件---斗帝蓝电霸王龙Pro
      10.   }

第五种:使用通信通道(通信通道是wx.navitageTo()独有的)

  • A页面部分js代码

  1.  wx.navigateTo({
  2.    url: "/pages/detail/detail?name='斗帝蓝电霸王龙Pro'",
  3.    events: {
  4.      // 为指定事件添加一个监听器,获取被打开页面传送到当前页面的数据
  5.      acceptDataFromOpenedPage: function(data) { //参数名字随便起,前后页面对应上即可
  6.      //对发送回来的数据进行处理
  7.        console.log(data)         // 斗帝蓝电霸王龙Plus
  8.     },
  9.      someEvent: function(data) { // 参数名字随便起,前后页面对应上即可
  10.        console.log(data)        // 斗帝蓝电霸王龙Pro max
  11.     }
  12.   },
  13.    success: function(res) {
  14.      // 通过eventChannel向被打开页面传送数据
  15.      res.eventChannel.emit('acceptDataFromOpenerPage', { data: '斗帝蓝电霸王龙Pro' })//参数名字随便起,前后页面对应上即可
  16.   }
  17.  })
  • B页面部分js代码

  1.  Page({
  2.    onLoad: function(option){
  3.      //获取通信通道
  4.      const eventChannel = this.getOpenerEventChannel()
  5.      // 监听acceptDataFromOpenerPage事件,获取上一页面通过eventChannel传送到当前页面的数据
  6.      eventChannel.on('acceptDataFromOpenerPage', function(data) {
  7.        //对发送过来的数据进行处理
  8.        console.log(data)   // 斗帝蓝电霸王龙Pro
  9.     })
  10.      //向上一页面发送数据
  11.      eventChannel.emit('acceptDataFromOpenedPage', {data: '斗帝蓝电霸王龙Plus'});
  12.      eventChannel.emit('someEvent', {data: '斗帝蓝电霸王龙Pro max'});
  13.   }
  14.  })

第六中:使用页面栈(只对当前页面栈中存在的页面生效)

  • A页面部分js代码

  1.  Page({
  2.    data:{
  3.      name:'斗帝蓝电霸王龙Pro',
  4.      age:22
  5.   },
  6.    eatFood:function(food){
  7.      console.log('eating '+ food)    
  8.   }
  9.  })
  • B页面部分js代码

  1.  Page({
  2.    onLoad: function (options) {
  3.      // 获取当前页面栈
  4.      const pages = getCurrentPages();
  5.      // 获取上一页面对象
  6.      let prePage = pages[pages.length - 2];
  7.      console.log(prePage.data.name) // 斗帝蓝电霸王龙Pro
  8.      prePage.eatFood('斗帝蓝电霸王龙Pro') // eating 斗帝蓝电霸王龙Pro
  9.   }
  10.  })

总结

  1. globalData与storage原理相同都是将数据放在一个公共的地方全应用随意取用,但是他们的数据不推送也不关联,即globalData和storage中的数据更新了,但已经获取过值的页面其对应的值并 不会更新。
  2. 解决办法:将各页面获取值的事件放到onShow()中,每次页面进入前台的时候都会重新从globalData和storage中取值。
  3. 路由携带参数简单方便,但是地址长度有限不能携带大量参数和特殊符号。
  4. 解决办法:在传参之前进行编码encodeURIComponent(),接收的时候进行解码 decodeURIComponent()。如此大量参数可以携带但是只能单方面传递参数,即只能a向b传,反之则不行。
  5. 通信通道,页面可以互相传参,但是该通道仅存在于wx.navagateTo()的接口调用时才有效。
  6. 页面栈,这是一个极其强大功能,它本质上不是向页面传参而是直接修改页面参数和方法。通过页面栈我们可以拿到该页面的对象,然后就可以对该页面的各项数据和函数进行操作。但是这种方法只能在b页面去操作a页面,并不能在a页面操作b页面,因为此时页面栈中还没有b。并且该方法也仅限于通过wx.navagateTo()或 <navigator open-type="navigateTo" url="/pages/detail/detail?id=5213828">跳转</navigator> 这两种方式进入b页面才可使用,因为其他方式跳转到b页面时已经销毁了其他所有页面,页面栈中已经没有上一个页面了。
     
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/寸_铁/article/detail/769889
推荐阅读
相关标签
  

闽ICP备14008679号