网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
- 当输入框的值发生变化时,调用
suggest
函数,用于显示/隐藏智能提示数据,参数items
为一个字符串数组。 - 当
items
中的字符串和输入框的值匹配时,将匹配的数据依次渲染在ul
下的li
节点中,并显示.js-suggest
节点,否则移除ul
下的所有li
节点,并隐藏.js-suggest
节点 - 输入框的值需要移除两侧空白再进行匹配
- 输入框的值为空时,按照全部不匹配处理
- 字符串使用模糊匹配,比如"北大"能匹配"北大"和"北京大学",但不能匹配"大北京",即按照
/北.*?大.*?/
这个正则进行匹配 - 通过在
.js-suggest
节点上添加/移除hide
这个class
来控制该节点的隐藏/显示
JavaScript实现
function suggest(items) {
const val = document.getElementsByClassName('js-input')[0].value.trim()
const suggest = document.getElementsByClassName('js-suggest')[0]
// 创建输入内容的正表达式:使用split将字符串转换成数组 -> 使用map映射生成新数组 -> 使用join将数组连接成字符串
const valReg = new RegExp(val.split('').map(v => special(v)).join(''))
// 通过filter方法过滤出items中符合的项:符合的条件是用户输入内容不为空(val != '')并且与用户输入的内容匹配
const item = items.filter(i => val != '' && valReg.test(i))
// 如果item.length不为0,代表有匹配的数据,则执行:
// suggest.classList['remove']('hide')相当于suggest.classList.remove('hide') 效果是删除hide这个class
suggest.classList[item.length ? 'remove' : 'add']('hide')
// 渲染列表
suggest.children[0].innerHTML = item.map(i => `<li>${i}</li>`).join('')
}
// 对特殊字符的处理
function special(val) {
// 如果val是()[].+/?\*这类的特殊字符,则在它前面加上转义字符:\
// 为什么是\\${val},两个\?因为在模板字符串``中\也需要使用\转义
return `${'()[].+/?\*'.indexOf(val) === -1 ? val : `\\${val}`}.\*?`
}
这个案例中需要注意的地方就是不要忘记对特殊字符的转义(实现special
函数)。
知识点:
- RegExp(正则表达式)
- split() 方法使用指定的分隔符字符串将一个
String
对象分割成子字符串数组,以一个指定的分割字串来决定每个拆分的位置。 - map() 方法创建一个新数组,这个新数组由原数组中的每个元素都调用一次提供的函数后的返回值组成。
- join() 方法将一个数组(或一个类数组对象)的所有元素连接成一个字符串并返回这个字符串。如果数组只有一个项目,那么将返回该项目而不使用分隔符。
- filter() 方法创建给定数组一部分的浅拷贝,其包含通过所提供函数实现的测试的所有元素。
- test() 方法执行一个检索,用来查看正则表达式与指定的字符串是否匹配。返回
true
或false
。 - Element.classList 是一个只读属性,返回一个元素
class
属性的动态 DOMTokenList集合。这可以用于操作class
集合。 - indexOf() 方法返回在数组中可以找到给定元素的第一个索引,如果不存在,则返回 -1。
2、打字机输出
效果演示
有以下HTML
和CSS
:
HTML结构
<div class="content">
<span class="word color23">h</span>
<span class="word color22">e</span>
<span class="word color4">l</span>
<span class="word color24">l</span>
<span class="word color17">o</span>
<span class="word color2"> </span>
<span class="word color9">w</span>
<span class="word color3">o</span>
<span class="word color1">r</span>
<span class="word color23">l</span>
<span class="word color15">d</span>
<br>
<span class="word color15">你</span>
<span class="word color13">好</span>
<span class="word color16">世</span>
<span class="word color19">界</span>
<span class="blink" id="jsBlink">|</span>
</div>
CSS样式
html,
body {
margin: 0;
}
.color1 {
color: #E60012;
}
.color2 {
color: #EB6100;
}
.color3 {
color: #F39800;
}
.color4 {
color: #FCC800;
}
.color5 {
color: #FFF100;
}
.color6 {
color: #CFDB00;
}
.color7 {
color: #8FC31F;
}
.color8 {
color: #22AC38;
}
.color9 {
color: #009944;
}
.color10 {
color: #009B6B;
}
.color11 {
color: #009E96;
}
.color12 {
color: #00A0C1;
}
.color13 {
color: #00A0E9;
}
.color14 {
color: #0086D1;
}
.color15 {
color: #0068B7;
}
.color16 {
color: #00479D;
}
.color17 {
color: #1D2088;
}
.color18 {
color: #601986;
}
.color19 {
color: #920783;
}
.color20 {
color: #BE0081;
}
.color21 {
color: #E4007F;
}
.color22 {
color: #E5006A;
}
.color23 {
color: #E5004F;
}
.color24 {
color: #E60033;
}
.word {
font-size: 20px;
}
.content {
text-align: center;
font-size: 0;
}
.blink {
font-size: 20px;
animation: fade 500ms infinite;
-webkit-animation: fade 500ms infinite;
}
@keyframes fade {
from {
opacity: 1.0;
}
50% {
opacity: 0;
}
to {
opacity: 1.0;
}
}
案例需求
页面上存在id
为jsBlink
的下划线闪动节点,请按照如下需求实现 output
函数
- 函数
output
接收一个字符串参数,每隔200毫秒在闪动节点之前逐个显示字符 - 请新建
span
节点放置每个字符,其中span
必须存在class
“word
”,并随机加上color1 ~ color24
中的任一个class
(请使用系统随机函数) - 每次输出指定字符串前,请将闪动节点之前的所有其他节点移除
- 不要销毁或者重新创建闪动节点
- 如果输出字符为
空格
、<
、>
,请分别对其进行HTML
转义,如果是\n
请直接输出<br />
,其他字符不需要做处理 - 上面展示的效果为
output('hello world\n你好世界')
之后的界面
JavaScript实现
function output(str) {
const content = document.getElementsByClassName('content')[0]
const jsBlink = document.getElementById('jsBlink')
// 将闪动节点之前的所有其他节点移除
while (content.children.length > 0) {
if (content.children[0] == jsBlink) {
// 如果content第1个孩子是jsBlink,说明闪动节点之前的所有其他节点移除完毕,则跳出循环
break;
}
// 删除content中的指定节点
content.removeChild(content.children[0]);
}
let i = 0;
const stl = setInterval(() => {
if (str[i] == '\n') {
const br = document.createElement('br')
// 在content中的jsBlink之前插入节点br
content.insertBefore(br, jsBlink)
} else {
const span = document.createElement('span')
span.classList.add('word')
span.classList.add(`color${Math.floor(Math.random() \* 24 + 1)}`)
switch (str[i]) {
case ' ':
span.innerHTML = ' '
break;
case '<':
span.innerHTML = '<'
break;
case '>':
span.innerHTML = '>'
break;
default:
span.innerHTML = str[i]
break;
}
content.insertBefore(span, jsBlink)
}
i++;
if (i >= str.length) {
// 清除定时器
clearInterval(stl)



**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**
GNUKfx-1715699944969)]
[外链图片转存中...(img-KmeTYANq-1715699944970)]
[外链图片转存中...(img-smVCZV2R-1715699944970)]
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**