vue3+vant4 二次封装IndexBar城市列表组件以及性能优化

前言

二次封装城市列表以及对静态资源的引入做异步引入优化。
版本号: vue3.3 + vant4

效果图

在这里插入图片描述

数据源

城市列表 city.json 永久网盘链接
链接:https://pan.baidu.com/s/10E-b441-4P7mjvomlJhm0g
提取码:m92c

字段大致讲解
indexList :每个字母是所属城市外层盒子唯一的key。
van-index-anchor: 是页面的每个标题,例如A、B。
van-cell:title就是A下面的每个城市
进来先用computed把城市数据源中的首个字母取出来。数据源长这个样子
在这里插入图片描述

性能优化点

此组件做了一处优化
city.json 同步引入改为异步引入。
作用:引入的数据量较大时 页面无需因为它而加载不出来,同步的还是照样展示,数据量大的让其加入到异步队列即可,因为我们都知道,同步永远比异步优先执行。
效果:如此可达到快速加载页面、丝毫不影响用户体验,还是挺不错的。

优化前效果图

进入的时候明显要等几秒钟,因为引入方式是同步的,同步需要页面完全加载完才会展示,这是同步引入的弊端。
在这里插入图片描述

优化前代码

<script lang="ts" setup>
import { ref, computed, onMounted} from 'vue'
import { useRouter } from 'vue-router'
import { city } from '@/assets/json/city.json' // 同步导入 JSON 数据

const router = useRouter()
const cityData = city
// const indexList = ['A', 'B', 'C']
//获取所有城市首字母
const indexList = computed(() => {
	return cityData.map(item => item.initial)
})
</script>
优化后效果

立马就进入,依旧优先加载同步,但是city.json改为异步了,已加入异步队列,所以永远在同步后面才执行。
在这里插入图片描述

优化后代码

<script lang="ts" setup>
import { ref, computed, onMounted } from 'vue'
const cityData = ref([])

// const indexList = ['A', 'B', 'C']
//获取所有城市首字母
const indexList = computed(() => {
	return cityData.value.map(item => item.initial)
})

// 异步加载城市数据 性能优化
async function loadCities() {
	try {
		const data = await import('@/assets/json/city.json') // 动态导入 JSON 数据
		cityData.value = data.city // 更新 cities 的值
		// console.log(cityData, 'cityData.value')
	} catch (error) {
		console.error('Failed to load cities:', error)
	}
}

// 在组件挂载后加载数据
onMounted(() => {
	loadCities()
})
</script>

完整源码

City.vue 子组件

<script lang="ts">
export default {
	name: 'CityList',
}
</script>
<script lang="ts" setup>
import { ref, computed, onMounted, } from 'vue'
import { useRouter } from 'vue-router'

const router = useRouter()
const cityData = ref([])

// const indexList = ['A', 'B', 'C']
//获取所有城市首字母
const indexList = computed(() => {
	return cityData.value.map(item => item.initial)
})

// 选择城市回首页
const backHome = cityname => {
	// router.push({ path: '/', query: { cityname: cityname + '市' } })
	console.log(cityname + '市', '选中的城市')
}
// 异步加载城市数据 性能优化
async function loadCities() {
	try {
		const data = await import('@/assets/json/city.json') // 动态导入 JSON 数据
		cityData.value = data.city // 更新 cities 的值
		// console.log(cityData.value, 'cityData.value')
	} catch (error) {
		console.error('Failed to load cities:', error)
	}
}

// 在组件挂载后加载数据
onMounted(() => {
	loadCities()
})
</script>
<template>
	<div class="city-list-container">
		<div class="hotcity">所有城市</div>
		<van-index-bar :index-list="indexList" highlight-color="#1989fa">
			<div v-for="(city, idx1) in cityData" :key="idx1">
				<van-index-anchor :index="city.initial"></van-index-anchor>
				<van-cell v-for="(item, idx2) in city.list" :key="idx2" :title="item.name" @click="backHome(item.name)"></van-cell>
			</div>
		</van-index-bar>
	</div>
</template>
<style scoped lang="scss">
.hotcity {
	padding: 30px;
	font-size: 28px;
	font-weight: bold;
}
::v-deep(.van-index-bar) {
	background-color: #fff;
}
::v-deep(.van-index-bar__sidebar) {
	top: 72%;
}
</style>

如何使用

以SearchList.vue 父组件为例

<template>
	<div>
		<CityList />
	</div>
</template>
  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
可以参考以下步骤: 1. 首先安装 axios 和 qs(如果需要) 2. 在 main.js 中引入 axios 并进行二次封装: ``` import axios from 'axios' import qs from 'qs' axios.defaults.baseURL = 'http://api.xxx.com' // 设置接口基地址 axios.defaults.timeout = 10000 // 设置超时时间 axios.defaults.headers.post['Content-Type'] = 'application/x-www-form-urlencoded;charset=UTF-8' // 设置默认请求头 axios.interceptors.request.use(config => { // 在请求发送之前做一些处理 config.headers.token = localStorage.getItem('token') || '' return config }, error => { // 出错处理 return Promise.reject(error) }) axios.interceptors.response.use(response => { // 在响应成功处理之前做一些处理 return response }, error => { // 响应错误处理 return Promise.reject(error) }) function get (url, params) { return new Promise((resolve, reject) => { axios.get(url, { params }).then(res => { resolve(res.data) }).catch(err => { reject(err.data) }) }) } function post (url, data) { return new Promise((resolve, reject) => { axios.post(url, qs.stringify(data)).then(res => { resolve(res.data) }).catch(err => { reject(err.data) }) }) } export default { get, post } ``` 3. 在组件中使用二次封装后的 axios: ``` import request from '@/utils/request' methods: { getData () { request.get('/api/getData', { id: '123456' }).then(res => { console.log(res) }).catch(err => { console.log(err) }) } } ``` 以上是对于vue3 + vant + ts 配置axios的二次封装的一种实现,仅供参考。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值