zoukankan      html  css  js  c++  java
  • 转载:如何处理浏览器的断网情况?

    本文转载自:
    如何处理浏览器的断网情况?
    https://juejin.im/post/5e81b0836fb9a03c7c4c0df6
    好的断网处理会让人很舒适:lol的断线重连,王者荣耀的断线重连 可以确保游戏的继续进行
    坏的断网处理甚至不处理会出bug:比如我手上的项目就出了个bug 业务人员表示非常苦恼
    网络问题一直是一个很值得关注的问题。
    比如在慢网情况下,增加loading避免重复发请求,使用promise顺序处理请求的返回结果,或者是增加一些友好的上传进度提示等等。
    那么大家有没有想过断网情况下该怎么做呢?比如说网络正常->断网->网络正常。
    其实我一直也没想过,直到组里的测试测出一个断网导致的bug,让我意识到重度依赖网络请求的前端,在断网情况下可能会出现严重的bug。
    因此我将在这里记录一下自己对系统断网情况下的处理,一方面避免bug产生,一方面保证用户及时在应用内知道网络已经断开连接

    概览
    用于检测浏览器是否连网的navigator.onLine
    断网事件"offline"和连网事件"online"
    断网处理项目实战

    思路和效果
    断网处理组件使用
    断网处理组件详情
    发现

    参考资料

    概览
    为了构建一个 “断网(offline)可用”的web应用,你需要知道应用在什么时候是断网(offline)的。
    不仅仅要知道什么时候断网,更要知道什么时候网络恢复正常(online)。
    可以分解陈本下面两种常见情况:

    你需要知道用户何时online,这样你可以与服务器之间re-sync(重新同步)。
    你需要知道用户何时offline,这样你可以将你未发出的请求过一段时间再向服务器发出。

    通常可以通过online/offline事件去做这个事情。
    用于检测浏览器是否连网的navigator.onLine
    navigator.onLine

    true online
    false offline

    可以通过network的online选项切换为offline,打印navigator.onLine验证。
    当浏览器不能连接到网络时,这个属性会更新。规范中是这样定义的:

    断网事件"offline"和连网事件"online"
    浏览器有两个事件:"online" 和 "offline".
    这两个事件会在浏览器在online mode和offline mode之间切换时,由页面的发射出去。
    事件会按照以下顺序冒泡:document.body -> document -> window。
    事件是不能去取消的(开发者在代码上不能手动变为online或者offline,开发时使用开发者工具可以)。
    注册上下线事件的几种方式
    最最建议window+addEventListener的组合。

    通过window或document或document.body和addEventListener(Chrome80仅window有效)
    为document或document.body的.ononline或.onoffline属性设置一个js函数。(注意,使用window.ononline和window.onoffline会有兼容性的问题)
    也可以通过标签注册事件

    例子

    <div id="status"></div>
    <div id="log"></div>
    
    window.addEventListener('load', function() {
      var status = document.getElementById("status");
      var log = document.getElementById("log");
    
      function updateOnlineStatus(event) {
        var condition = navigator.onLine ? "online" : "offline";
        status.innerHTML = condition.toUpperCase();
    
        log.insertAdjacentHTML("beforeend", "Event: " + event.type + "; Status: " + condition);
      }
    
      window.addEventListener('online',  updateOnlineStatus);
      window.addEventListener('offline', updateOnlineStatus);
    });
    
    

    其中insertAdjacentHTML是在标签节点的邻近位置插入,可以查阅:DOM进阶之insertAdjacentHTML
    断网处理项目实战
    基于vue以及iView的Spin,Notice组件封装出离线处理组件,在需要到的页面引入即可。
    思路和效果
    只要做到断网提醒+遮罩,上线提醒-遮罩即可。

    监听offline,断网给出提醒和遮罩:网络已断开,请检查网络连接。
    监听online,连网给出提醒和遮罩:网络已连接。

    断网处理组件使用

    <OfflineHandle
        :offlineTitle = "断网处理标题"
        :desc="断网处理描述"
        :onlineTitle="连网提醒"
    >
    </OfflineHandle>
    

    断网处理组件详情

    <!--OfflineHandle.vue-->
    <template>
      <div v-if="spin" class="offline-mark">
        <Spin size="large" fix>
          <h2>{{offlineTitle}}</h2>
    
          <p>{{desc}}</p>
        </Spin>
      </div>
    </template>
    
    <script>
    export default {
      name: 'offline-handle',
      props: {
        offlineTitle: {
          type: String,
          default: '网络已断开,请检查网络连接。',
        },
        onlineTitle: {
          type: String,
          default: '网络已连接',
        },
        desc: {
          type: String,
          default: '',
        },
        duration: {
          type: Number,
          default: 4.5,
        },
      },
      data() {
        return {
          spin: false,
        };
      },
      mounted() {
        window.addEventListener('offline', this.eventHandle);
        window.addEventListener('online', this.eventHandle);
      },
      beforeDestroy() {
        window.removeEventListener('offline', this.eventHandle);
        window.removeEventListener('online', this.eventHandle);
      },
      methods: {
        eventHandle(event) {
          const type = event.type === 'offline' ? 'error' : 'success';
          this.$Notice[type]({
            title: type === 'error' ? this.offlineTitle : this.onlineTitle,
            desc: type === 'error' ? this.desc : '',
            duration: this.duration,
          });
          setTimeout(() => {
            this.spin = event.type === 'offline';
          }, 1500);
        },
      },
    };
    </script>
    
    <style lang="scss" scoped>
    .offline-mark {
      position: fixed;
      top: 0;
      left: 0;
      right: 0;
      bottom: 0;
      background-color: #ccc;
      z-index: 9999;
      transition: position 2s;
    }
    /deep/.ivu-spin-fix {
      text-align: left;
      font-size: 20px;
      h2 {
        color: rgba(0, 0, 0, 0.8);
      }
      p {
        margin-top: 20px;
        color: red;
        font-weight: bold;
      }
    }
    </style>
    
    

    发现

    offline和online事件:window有效,document和document.body设置无效
    手上的项目只运行在Chrome浏览器,只有为window设置offline和online才生效。
    运行环境:"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.149 Safari/537.36
    为position增加2s的transition的避免屏闪

  • 相关阅读:
    lightoj 1094 Farthest Nodes in a Tree 【树的直径 裸题】
    nyoj 1185 最大最小值【线段树最大值最小值维护】
    nyoj 123 士兵杀敌(四) 树状数组【单点查询+区间修改】
    poj 3468 A Simple Problem with Integers【线段树区间修改】
    hdoj 1698 Just a Hook【线段树区间修改】
    hdoj 1556 Color the ball【线段树区间更新】
    hdoj 1286 找新朋友【欧拉函数】
    [LC] 303. Range Sum Query
    [LC] 79. Word Search
    [LC] 211. Add and Search Word
  • 原文地址:https://www.cnblogs.com/smart-girl/p/12602200.html
Copyright © 2011-2022 走看看