你是否遇到过断网检测的需求?

你也碰到断网检测的需求啦?

一般的断网检测需求,大部分都是在用户网络状态不好或者网络掉线时,我们给用户一个提示或者引导界面,防止用户不知道自己卡了在那里一直等待。

方案1,轮询请求

直接说最有效的方案,我们通过轮训请求来检测网络是否可用,比如加载图片或者访问接口等...

下面我以加载图片为例,搞个小demo:

  1. 首先尽可能的找一个小的图片,不要让图片的请求堵塞我们的其他功能使用。

    推荐一个图片在线的压缩的网站: https://www.yalijuda.com/,然后把图片上到内部的服务器上

  2. 既然搞了,就搞个通用的,使用tsup我们搞个npm包,然后想一想我们要搞的功能,先把入口函数的出入参类型定了。我们既然要做轮训请求图片,我们首先需要一个图片地址,然后请求后的回调事件,甚至可能需要一些控制参数,那我们的入口代码也就有了。

    js 复制代码
    const request = (imgUrl) => {
      // do something
    };
    
    type CheckNetworkOptionsType = {
      interval: number; // 循环时间 单位ms
    };
    
    type CheckNetworkType = (
      imgUrl: string, // 图片url
      callback: (isOnline: boolean) => void,  // 回调,返回网络是否在线
      options?: CheckNetworkOptionsType, // 配置项
    ) => void;
    
    const checkNetwork: CheckNetworkType = (imgUrl, callback, options) => {
      const { interval = 30_000 } = options || {};
      const timer = setInterval(() => {
        request(imgUrl)
      }, interval);
      return timer
    };
    
    export default checkNetwork
  3. 接下来我们要考虑一下如何进行请求,我们需要一个创一个promiseimg标签resove 出去 onloadonerror 对应的在线和离线状态。

    js 复制代码
    const request = (imgUrl) => {
      return new Promise((resolve) => {
        let imgRef = null;
        let isRespond = false;
        imgRef = document.createElement('img');
        imgRef.onerror = () => {
          isRespond = true;
          resolve(false);
          return '';
        };
        imgRef.onload = () => {
          isRespond = true;
          resolve(true);
          return '';
        };
        imgRef.src = `${imgUrl}?time=${new Date().toLocaleString()}`;
    
      });
    };
    
    type CheckNetworkOptionsType = {
      interval: number; // 循环时间 单位ms
    };
    
    type CheckNetworkType = (
      imgUrl: string, // 图片url
      callback: (isOnline: boolean) => void,  // 回调,返回网络是否在线
      options?: CheckNetworkOptionsType, // 配置项
    ) => void;
    
    const checkNetwork: CheckNetworkType = (imgUrl, callback, options) => {
      const { interval = 30_000 } = options || {};
      const timer = setInterval(async () => {
        const status = (await request(imgUrl)) as boolean;
        callback(status);
      }, interval);
      return timer
    };
    
    export default checkNetwork
  4. 这样基本的功能似乎就差不多了,但是感觉好像少点什么?比如服务器就是返回图片资源慢?那我们是不是可以加个超时时间?又或者是不是可以可以让用户手动取消循环?

    js 复制代码
    const request = (imgUrl) => {
      return new Promise((resolve) => {
        let imgRef = null;
        let isRespond = false;
        imgRef = document.createElement('img');
        imgRef.onerror = () => {
          isRespond = true;
          resolve(false);
          return '';
        };
        imgRef.onload = () => {
          isRespond = true;
          resolve(true);
          return '';
        };
        // 加个参数,防止浏览器缓存
        imgRef.src = `${imgUrl}?time=${new Date().toLocaleString()}`;
    
      });
    };
    
    type CheckNetworkOptionsType = {
      interval: number; // 循环时间 单位ms
    };
    
    type CheckNetworkType = (
      imgUrl: string, // 图片url
      callback: (isOnline: boolean) => void,  // 回调,返回网络是否在线
      options?: CheckNetworkOptionsType, // 配置项
    ) => void;
    
    const checkNetwork: CheckNetworkType = (imgUrl, callback, options) => {
      const { interval = 30_000 } = options || {};
      const timer = setInterval(async () => {
        const status = (await request(imgUrl)) as boolean;
        callback(status);
      }, interval);
      return timer
    };
    
    export default checkNetwork
  5. 完整的代码就完整了,具体用的时间还是建议大家关键模块来进行断网检测。不要一个后台配置表单都弄检测,这种完全可以在提交表单的时候接口响应进行处理,断网检测一般都是用在需要实时监控之类的。不多说了,我们来体验下:

6. 没问题,一切ok,发个包,就叫network-watcher吧,欢迎大家star!
github.com/waltiu/netw...

方案2,直接调api

首先说下这种方案不推荐,其一浏览器兼容性有问题,其二只能检测到是否网络有连接,但是不能检测是否可用,这种实用性真的很差。

浏览器也提供了navigator.onLinenavigator.connection 可以直接查询网络状态,我们可以监听网络状态的变化。

js 复制代码
    window.addEventListener('online',function () {
        alert("正常上网");
    })
    window.addEventListener('offline',function () {
        alert('无网络');
    })

这种实用性真的很差,用户网络连接但是没有网或者网很慢,实际上都会影响用户体验!!

用户的体验永远是NO.1

相关推荐
VT.馒头5 小时前
【力扣】2625. 扁平化嵌套数组
前端·javascript·算法·leetcode·职场和发展·typescript
San30.8 小时前
从零构建坚固的前端堡垒:TypeScript 与 React 实战深度指南
前端·react.js·typescript
VT.馒头10 小时前
【力扣】2694. 事件发射器
前端·javascript·算法·leetcode·职场和发展·typescript
止观止11 小时前
像三元表达式一样写类型?深入理解 TS 条件类型与 `infer` 推断
前端·typescript
We་ct12 小时前
LeetCode 73. 矩阵置零:原地算法实现与优化解析
前端·算法·leetcode·矩阵·typescript
We་ct12 小时前
LeetCode 48. 旋转图像:原地旋转最优解法
前端·算法·leetcode·typescript
We་ct1 天前
LeetCode 54. 螺旋矩阵:两种解法吃透顺时针遍历逻辑
前端·算法·leetcode·矩阵·typescript
We་ct1 天前
LeetCode 36. 有效的数独:Set实现哈希表最优解
前端·算法·leetcode·typescript·散列表
Async Cipher2 天前
TypeScript 的用法
前端·typescript
We་ct2 天前
LeetCode 30. 串联所有单词的子串:从暴力到高效,滑动窗口优化详解
前端·算法·leetcode·typescript