js中for-in的坑

在js中一般使用的循环有两种

1.常规的for(var i=0;i<length;i++)

2.for-in:for(var item in list)

但是个人更喜欢使用第一种循环,而不喜欢几乎从来不使用for-in这种写法,原因如下:

1.第一种写法能够很好的控制循环何时结束,以及对应的索引;而第二种循环无法控制

2.第二种写法在某种情况下,可能会导致一些奇怪的bug

针对原因2参见下面简单的案例:


以上是在jsBin中编辑的代码,在第一段代码中,对js的数组Array添加了一个新的方法(prototype 属性使您有能力向对象添加属性和方法),为了在某些时候能够方便使用。此时,如果用for-in循环出数组arr的内容,会发现在本来想要得到的0,1,2,3的后面会多一个function(){alert("myfunction");},这应该不是我们想要得到的结果。

注意:for..in循环会把某个类型的原型(prototype)中方法与属性给遍历出来,所以这可能会导致代码中出现意外的错误

上面只是简单的遍历一个数组,可能出现的问题还不够明显,如下所示


如果我们遍历的数组中存放的是业务对象,此时会发现,最后一次输出的是空,在被循环的数组中不存在这个名称为空的人员,此时出现的错误就导致某些业务无法正常使用或显示。

对于使用for-in可能导出的bug,有两种方式避免

1.在循环数组集合时,不使用for-in,统一使用for(var i=0;i<length;i++)这种形式;

2.在for-in循环中增加一个hasOwnProperty的判断;

hasOwnProperty函数用于指示一个对象自身(不包括原型链)是否具有指定名称的属性。如果有,返回true,否则返回false

该方法属于Object对象,由于所有的对象都"继承"了Object的对象实例,因此几乎所有的实例对象都可以使用该方法。


上述的两种方式,个人觉得第一种方式更好,因为是从根源上避免问题的发生,对于第二种方式,完全是为了解决问题而增加额外的处理完全没有这个必要。



### el-select 组件常见问题与解决方法 #### 1. 关键词提示功能的实现 为了提高用户体验,可以通过封装 `el-select` 组件来实现关键词(如经常搜索、热点搜索)提示的功能。通过使用 Element UI 的分组特性,可以将历史搜索记录按类别分组展示给用户,以便于快速选择之前使用的关键词[^1]。 ```html <template> <el-select v-model="selectedKeyword"> <!-- 历史搜索 --> <el-option-group label="历史搜索"> <el-option v-for="(item, index) in historyKeywords" :key="index" :label="item.label" :value="item.value"></el-option> </el-option-group> <!-- 热门搜索 --> <el-option-group label="热门搜索"> <el-option v-for="(item, index) in hotKeywords" :key="index" :label="item.label" :value="item.value"></el-option> </el-option-group> </el-select> </template> ``` #### 2. 多个选项默认选中的问题 当初始化页面时发现所有项均显示为已选中状态,这是因为每个 `<el-option>` 标签错误地设置了相同的 `aria-selected="true"` 属性。实际上,在任何时刻仅有一个选项应当处于选中状态,因此需确保只对实际被选中的那一项设置此属性[^2]。 ```javascript // 正确的做法是在 data 或 setup 中定义 selectedValue 变量用于存储当前选中的值 data() { return { selectedValue: null, }; } ``` #### 3. Select 组件可搜索模式下的匹配逻辑 如果希望支持模糊查询,则需要注意的是,默认情况下 `el-select` 并不会自动识别未绑定到 `label` 上的内容作为过滤条件的一部分。要使自定义字段参与筛选过程,必须显式指定这些字段名并将其关联至相应的数据源属性上[^3]。 ```html <!-- 使用 filter-method 自定义过滤函数 --> <el-select v-model="searchResult" filterable @visible-change="handleVisibleChange"> <el-option v-for="item in filteredOptions" :key="item.id" :label="item.name" :value="item.id"></el-option> </el-select> <script> export default { methods: { handleVisibleChange(visible) { if (visible && this.searchInput !== '') { // 手动触发一次过滤操作 this.$refs.select.filter(this.searchInput); } }, querySearch(queryString, cb) { let results = queryString ? this.options.filter(item => item.name.toLowerCase().indexOf(queryString.toLowerCase()) === 0) : this.options; // 调用 callback 返回建议列表的数据 cb(results); } } }; </script> ```
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值