vue3自定义hooks遇到的问题

news2025/3/12 23:13:22

问题

写了一个输入查询参数和url返回加载中状态、请求方法、接口返回列表的hooks,出现的结果是只有请求方法有效,加载状态无效,接口返回了数据,页面却不显示数据。

代码如下

只展示部分关键代码

import { ref, toRefs, toRef, onMounted, onActivated, reactive } from 'vue';
import baseService from '@/service/baseService';
export default (props) => {
  const defaultOptions = {
    getListURL: '',
    dataForm: {},
	loading: false,
	dataList: []
  };
  const state = ref({ ...defaultOptions, ...props });
  const fetchList = () => {
    if (!state.value.getListURL) {
      return;
    }
    state.value.loading = true;
    baseService
      .get(state.value.getListURL, { ...state.value.dataForm })
      .then((res) => {
        if (res.code === 0) {
          state.value.dataList = [...res.data ];
        }
      })
      .finally(() => {
        state.value.loading = false;
      });
  };

  onMounted(() => {
     fetchList();
  });
  return {
    fetchList,
    dataList: state.value.dataList,
    loading: state.value.loading,
  };
};

页面使用

<el-row :gutter="20" v-loading="loading">
   <el-col :span="12" v-for="item in dataList" :key="item.id">
      <div class="item">
         {{ item.name }}
      </div>
   </el-col>
</el-row>

const params = ref({
  name: '', // 搜索框名称
});
const { loading, fetchList, dataList } = useDataList({
  dataForm: params.value,
  getListURL: '/page/list',
});

分析

好久没用vue3,写出来的代码让自己一时找不出来问题,唉。。。

第一次尝试

在使用文件下面打印列表查看一下

const { loading, fetchList, dataList } = useDataList({
  dataForm: params.value,
  getListURL: '/page/list',
});
console.log(dataList);

是Proxy包裹的空数组,这应该是响应的数据,不像解构赋值出现的响应丢失啊。

为防止真的是这个问题,在hooks导出时使用toRefs包裹一下看看

import { ref, toRefs, toRef, onMounted, onActivated, reactive } from 'vue';
import baseService from '@/service/baseService';
export default (props) => {
  const defaultOptions = {
    getListURL: '',
    dataForm: {},
	loading: false,
	dataList: []
  };
  const state = ref({ ...defaultOptions, ...props });
  const fetchList = () => {
    if (!state.value.getListURL) {
      return;
    }
    state.value.loading = true;
    baseService
      .get(state.value.getListURL, { ...state.value.dataForm })
      .then((res) => {
        if (res.code === 0) {
          state.value.dataList = [...res.data ];
        }
      })
      .finally(() => {
        state.value.loading = false;
      });
  };

  onMounted(() => {
     fetchList();
  });
  return {
    fetchList,
	...toRefs({
		dataList: state.value.dataList,
    	loading: state.value.loading,
	})
  };
};

这下真好,打印的是ObjectRefImpl,然而查看里面看到还是空数组。

第二次尝试

接口响应里面有问题吗?那就在接口成功里面打印一下试试看

import { ref, toRefs, toRef, onMounted, onActivated, reactive } from 'vue';
import baseService from '@/service/baseService';
export default (props) => {
  const defaultOptions = {
    getListURL: '',
    dataForm: {},
	loading: false,
	dataList: []
  };
  const state = ref({ ...defaultOptions, ...props });
  const fetchList = () => {
    if (!state.value.getListURL) {
      return;
    }
    state.value.loading = true;
    baseService
      .get(state.value.getListURL, { ...state.value.dataForm })
      .then((res) => {
        if (res.code === 0) {
          state.value.dataList = [...res.data ];
          console.log(state.value.dataList);
        }
      })
      .finally(() => {
        state.value.loading = false;
      });
  };

  onMounted(() => {
     fetchList();
  });
  return {
    fetchList,
    dataList: state.value.dataList,
    loading: state.value.loading,
  };
};

发现这里打印是有数据的,那在hooks方法里面返回前打印看看

import { ref, toRefs, toRef, onMounted, onActivated, reactive } from 'vue';
import baseService from '@/service/baseService';
export default (props) => {
  const defaultOptions = {
    getListURL: '',
    dataForm: {},
	loading: false,
	dataList: []
  };
  const state = ref({ ...defaultOptions, ...props });
  const fetchList = () => {
    if (!state.value.getListURL) {
      return;
    }
    state.value.loading = true;
    baseService
      .get(state.value.getListURL, { ...state.value.dataForm })
      .then((res) => {
        if (res.code === 0) {
          state.value.dataList = [...res.data ];
        }
      })
      .finally(() => {
        state.value.loading = false;
      });
  };

  onMounted(() => {
     fetchList();
  });
  console.log(state.value.dataList);
  return {
    fetchList,
    dataList: state.value.dataList,
    loading: state.value.loading,
  };
};

啊?这里也是空对象,看来问题还是出现在hooks里面啊。直接打印state看看吧(应该上一次打印,这两个都打印试试的)

import { ref, toRefs, toRef, onMounted, onActivated, reactive } from 'vue';
import baseService from '@/service/baseService';
export default (props) => {
  const defaultOptions = {
    getListURL: '',
    dataForm: {},
	loading: false,
	dataList: []
  };
  const state = ref({ ...defaultOptions, ...props });
  const fetchList = () => {
    if (!state.value.getListURL) {
      return;
    }
    state.value.loading = true;
    baseService
      .get(state.value.getListURL, { ...state.value.dataForm })
      .then((res) => {
        if (res.code === 0) {
          state.value.dataList = [...res.data ];
        }
      })
      .finally(() => {
        state.value.loading = false;
      });
  };

  onMounted(() => {
     fetchList();
  });
  console.log(state.value.dataList);
  console.log(state);
  return {
    fetchList,
    dataList: state.value.dataList,
    loading: state.value.loading,
  };
};

此时可以看到state里面是有接口返回的数据的,这样应该就说明不能使用.value获取之后返回使用,这时候的数据是接口响应前的。

那就改造一下直接返回refreactive的数据结构赋值之后是响应丢失的)

import { ref, toRefs, toRef, onMounted, onActivated, reactive } from 'vue';
import baseService from '@/service/baseService';
export default (props) => {
  const defaultOptions = {
    getListURL: '',
    dataForm: {},
  };
  const state = ref({ ...defaultOptions, ...props });
  const loading = ref(false);
  const dataList = ref({});
  const fetchList = () => {
    if (!state.value.getListURL) {
      return;
    }
    loading.value = true;
    baseService
      .get(state.value.getListURL, { ...state.value.dataForm })
      .then((res) => {
        if (res.code === 0) {
          dataList.value = [...res.data ];
        }
      })
      .finally(() => {
        loading.value = false;
      });
  };

  onMounted(() => {
     fetchList();
  });
  return {
    fetchList,
    dataList,
    loading,
  };
};
<el-row :gutter="20" v-loading="loading.value">
   <el-col :span="12" v-for="item in dataList.value" :key="item.id">
      <div class="item">
         {{ item.name }}
      </div>
   </el-col>
</el-row>



const params = ref({
  name: '', // 搜索框名称
});
const { loading, fetchList, dataList } = useDataList({
  dataForm: params.value,
  getListURL: '/page/list',
});

但是这是为什么,还是有问题啊?崩溃啊!

第三次尝试

在使用里面打印一下

<el-row :gutter="20" v-loading="loading.value">
   <el-col :span="12" v-for="item in dataList.value" :key="item.id">
      <div class="item">
         {{ item.name }}
      </div>
   </el-col>
</el-row>



const params = ref({
  name: '', // 搜索框名称
});
const { loading, fetchList, dataList } = useDataList({
  dataForm: params.value,
  getListURL: '/page/list',
});
console.log(dataList)

发现数据已经更新到接口返回的了,怎么页面没反应呢?额。。。好像页面模版里面可以省略.value,修改一下看看

<el-row :gutter="20" v-loading="loading">
   <el-col :span="12" v-for="item in dataList" :key="item.id">
      <div class="item">
         {{ item.name }}
      </div>
   </el-col>
</el-row>



const params = ref({
  name: '', // 搜索框名称
});
const { loading, fetchList, dataList } = useDataList({
  dataForm: params.value,
  getListURL: '/page/list',
});
console.log(dataList)

撒花,终于解决完问题了(刚才不手欠加个.value,就不用耽误一次修改了)

结尾

  1. 自定义hooks,要想返回响应式ref数据,不要使用.value转一层,直接返回ref数据。
  2. 模板里面使用ref数据时候,不要加.value

彩蛋

使用css实现多行超出显示…无效

超出一行没换行

有的换行,但是超出没显示…

overflow:hidden; 
text-overflow:ellipsis;
display:-webkit-box; 
-webkit-box-orient:vertical;
-webkit-line-clamp:2; 

这可能是英文文字太长导致的,这需要加上强制换行的css就可以

word-break: break-all;

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2313980.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

liunx磁盘挂载和jar启动命令

一、磁盘挂载 查看历史磁盘挂载命令&#xff1a;history | grep mount 查看所有挂载硬盘命令&#xff1a;mount 磁盘挂载命令&#xff1a;mount -t cifs -o usernamesh**,passwordP!ss**** //192.168.1.2/attachmentfilesShare2.2/pdfCert /home/nybzg/cnfai1/pdfCert 二、j…

gbase8s rss集群通信流程

什么是rss RSS是一种将数据从主服务器复制到备服务器的方法 实例级别的复制 (所有启用日志记录功能的数据库) 基于逻辑日志的复制技术&#xff0c;需要传输大量的逻辑日志,数据库需启用日志模式 通过网络持续将数据复制到备节点 如果主服务器发生故障&#xff0c;那么备用服务…

如何调用 DeepSeek 的自然语言处理 API 接口并集成到在线客服系统

我在业余时间开发了一款自己的独立产品&#xff1a;升讯威在线客服与营销系统。陆陆续续开发了几年&#xff0c;从一开始的偶有用户尝试&#xff0c;到如今线上环境和私有化部署均有了越来越多的稳定用户。 随时近来 AI 大模型的火热&#xff0c;越来越多的客户&#xff0c;问…

【AI智能体报告】开源AI助手的革命:OpenManus深度使用报告

一、引言&#xff1a;当开源智能体走进生活 2025年3月&#xff0c;MetaGPT团队用一场"开源闪电战"改写了AI Agent的竞争格局。面对商业产品Manus高达10万元的邀请码炒作&#xff0c;他们仅用3小时便推出开源替代品OpenManus&#xff0c;首日即登顶GitHub趋势榜。 …

DeepSeek+Maxkb+Ollama+Docker搭建一个AI问答系统

DeepSeekMaxkbOllamaDocker搭建一个AI问答系统 文章目录 DeepSeekMaxkbOllamaDocker搭建一个AI问答系统前言一、创建同一内网的网络二、拉取两个镜像三、启动Ollama以及调试Maxkb4.Maxkb创建一个应用并建立知识库5、应用效果总结 前言 我觉得只要是使用Docker技术&#xff0c;…

江科大51单片机笔记【12】DS18B20温度传感器(上)

写在前言 此为博主自学江科大51单片机&#xff08;B站&#xff09;的笔记&#xff0c;方便后续重温知识 在后面的章节中&#xff0c;为了防止篇幅过长和易于查找&#xff0c;我把一个小节分成两部分来发&#xff0c;上章节主要是关于本节课的硬件介绍、电路图、原理图等理论…

P8662 [蓝桥杯 2018 省 AB] 全球变暖--DFS

P8662 [蓝桥杯 2018 省 AB] 全球变暖--dfs 题目 解析讲下DFS代码 题目 解析 这道题的思路就是遍历所有岛屿&#xff0c;判断每一块陆地是否会沉没。对于这种图的遍历&#xff0c;我们首先应该想到DFS。 代码的注意思想就是&#xff0c;在主函数中遍历找出所有岛屿&#xff0c…

Vue 侧边栏导航栏 el-menu单个item和多个item

在固钉的下面去写菜单导航栏。 <el-menu class"aside-menu" router :default-active"$route.path" :collapse"isCollapse" background-color"#131b27" text-color"#bfcbd9" active-text-color"#20a0ff" :defau…

Unity Dots从入门到精通之 Prefab引用 转 实体引用

文章目录 前言安装 DOTS 包实体引用Authoring 前言 DOTS&#xff08;面向数据的技术堆栈&#xff09;是一套由 Unity 提供支持的技术&#xff0c;用于提供高性能游戏开发解决方案&#xff0c;特别适合需要处理大量数据的游戏&#xff0c;例如大型开放世界游戏。 本文讲解我在…

无人机避障——XTDrone中运行VINS-Fusion+Ego-planner进行路径规划

本文聚焦于无人机避障技术领域的经典方案&#xff0c;重点探讨视觉双目VINS-Fusion建图与Ego-planner路径规划的组合应用。通过视觉双目VINS-Fusion实现精准的环境建图与自身定位&#xff0c;结合Ego-planner的高效路径规划能力&#xff0c;使无人机在复杂环境中实现自主避障飞…

【沐渥科技】氮气柜日常如何维护?

氮气柜的维护是确保其长期稳定运行、延长使用寿命和保持环境控制精度的关键。以下是沐渥氮气柜的日常维护和定期保养指南&#xff1a; 一、日常维护 柜体清洁 定期用软布擦拭柜体表面和内部&#xff0c;避免灰尘堆积。避免使用腐蚀性清洁剂&#xff0c;防止损伤密封条或传感器。…

MATLAB 控制系统设计与仿真 - 24

PID 控制器分析- 控制器的形式 连续控制器的结构&#xff1a; 为滤波时间常数&#xff0c;这类PID控制器在MATLAB系统控制工具箱称为并联PID控制器&#xff0c;可由MATLAB提供的pid函数直接输入&#xff0c;格式为&#xff1a; 其他类型的控制器也可以由该函数直接输入&#x…

linux(权限)

sudo 主要用来短暂的提权 权限 就是 >角色目标属性 这里面的角色就是---拥有者----所属组----other 所属组的目的&#xff1f; 更细化的管理 chmod 就是修改权限制 我们要是想要切换到体育的账号&#xff0c;我们可以去看一下有几个账号,我…

《OpenCV》—— dlib(换脸操作)

文章目录 dlib换脸介绍仿射变换在 dlib 换脸中的应用 换脸操作 dlib换脸介绍 dlib 换脸是基于 dlib 库实现的一种人脸替换技术&#xff0c;以下是关于它的详细介绍&#xff1a; 原理 人脸检测&#xff1a;dlib 库中包含先进的人脸检测器&#xff0c;如基于 HOG&#xff08;方向…

修改Flutter项目使用的JAVA版本

使用Android studio开发Flutter过程中&#xff0c;会默认使用Android studio自带的JDK。因为新版Android studio中的JDK版本过高&#xff0c;导致项目编译时总是无法完成&#xff0c;报【 unsupported class file major version 65】错误&#xff0c;如下&#xff1a; 解决这个…

虚拟dom的diff中的双端比较算法

‌双端比较算法是Vue中用于高效比较新旧VNode子节点的一种策略‌。该算法的核心思想是&#xff0c;通过从新旧VNode子节点的两端开始比较&#xff0c;逐步向中间靠拢&#xff0c;以找到最小的差异并据此更新DOM。以下是双端比较算法的大致流程&#xff1a; ‌初始化指针‌&…

VMware安装Windows server 2016

1、新建虚拟机&#xff0c;选择自定义模式 2、选择兼容性 4、命名虚拟机 5、固件类型 EFI 虚拟磁盘类型&#xff0c;不同电脑推荐的类型不同&#xff0c;用默认的就行 删除声卡和打印机 检查网络配置 选择本地的Windows server 2016的系统镜像&#xff0c;系统镜像可以去Window…

HippoRAG 2 原理精读

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 整体流程离线索引阶段在线检索和问答阶段 总结 整体流程 从上图可以看出&#xff0c;整个流程分为两个阶段 1、离线索引阶段 2、在线检索和问答阶段 离线索引阶段…

三:FFMPEG拉流读取模块的讲解

FFMPEG拉流读取模块在远程监控项目最核心的作用是读取UVC摄像头传输的H264码流&#xff0c;并对其码流进行帧的提取&#xff0c;提取完成之后则把数据传输到VDEC解码模块进行解码。而在我们这个项目中&#xff0c;UVC推流的功能由FFMPEG的命令完成。 FFMPEG拉流读取模块的API…

《苍穹外卖》SpringBoot后端开发项目核心知识点与常见问题整理(DAY1 to DAY3)

目录 一、在本地部署并启动Nginx服务1. 解压Nginx压缩包2. 启动Nginx服务3. 验证Nginx是否启动成功&#xff1a; 二、导入接口文档1. 黑马程序员提供的YApi平台2. YApi Pro平台3. 推荐工具&#xff1a;Apifox 三、Swagger1. 常用注解1.1 Api与ApiModel1.2 ApiModelProperty与Ap…