亲爱的观众老婆们好~最近在团队里组织了一次内容关于React的分享,然然后端有同窗反映未能理解为什么前端须要使用框架,框架究竟解决了什么问题。javascript
回想刚入坑前端的时候,也接触了一些PHP,当了解到其实PHP能够生成HTML模板以后,对前端的信仰几乎崩溃。既而后端就能够渲染前端的话,前端价值到底在哪?感受页面就是随便切一下就行,根本不必用框架,当时着实迷茫了好久。前端
后来对前端领域接触深了,再通过大神的指点,总算是理解为什么前端须要使用框架。根据本身的理解,再结合分享时后端同窗的疑问,因而有了这篇文章。本文相对小白向,只是经过虚拟一个项目说明问题,还望各位大神不吝赐教。java
产品: 来来来,新项目来啦。最近发现用户喜欢撸猫,咱们来个在线云撸猫!页面要求有猫图,点一下计数加一点当撸了一次!多久能上线??后端
这个简单啊!
设计模式
厉害的我连jQuery都不用,原生上!bash
<body>
<img src="1.jpg" alt="">
<p>0</p>
<script>
const img = document.querySelector('img');
const p = document.querySelector('p');
let count = 0;
img.addEventListener('click', function() {
count += 1;
p.innerHTML = count;
})
</script>
</body>复制代码
简单易懂,对不对!闭包
产品: 线上大受欢迎,然而要改需求了爸爸!如今要两只喵,点击要分开算哦,各自显示就好。你最棒了,我知道你确定能够的!app
看在叫了爸爸的份上,仍是给他改一下吧。然而第一个版本多粗糙啊,各类污染全局变量,第二版我要写得棒棒的!框架
<body>
<section class="catSection">
<img src="cat1.jpg" alt="" class="catImg">
<p class="catCount">0</p>
</section>
<section class="catSection">
<img src="cat2.jpg" alt="" class="catImg">
<p class="catCount">0</p>
</section>
<script>
(function() {
const catSection = document.querySelectorAll('.catSection');
catSection.forEach((section) => {
let count = 0;
const catImg = section.querySelector('.catImg');
const catCount = section.querySelector('.catCount');
catImg.addEventListener('click', () => {
count++;
catCount.innerHTML = count;
})
})
})()
</script>复制代码
没有污染全局变量,还能够扩展需求,后续产品就算要求再多的猫我都能hold住!
函数
产品: 父皇,你听我说啊,就最后一次,真的最后一次。如今猫两只不够啊,但太多的话展现又很差看。如今需求作一个列表,列表有N只猫,点那只猫展现那只猫,每次只展现意志,并且猫要有对应的名字哦,点击也要分开统计!其实跟之前差很少?就改改就行了。明天就要哦!
我封装得好好的功能,你跟我说改需求?并且改得面目全非跟我说差很少?
然而,吐槽归吐槽,需求仍是要作的。并且需求明天就要,代码写得再好也可能立刻改功能,代码仍是实现功能就算了吧。
具体代码由各位看官本身实现(建议先停下来,动手去实现这个需求),这里我就再也不上代码了。极可能咱们此次写的代码,就不会太考虑什么全局变量污染,也不考虑封装的问题,逐渐趋向于实现功能就行了,由于需求太多了,时间和精力限制了去写“好”代码。
固然了,产品经理说需求最后一次改,都是骗人的。下次可能会有点击到某个值时,自动切换猫,动态添加猫等等的新需求。咱们如今这样组织代码的形式,是典型的”意大利面式“代码(简单说,就是各类东西整合在一块儿,层级不分,难以维护)。这种代码写起来直观,但往后的维护是至关难的。写出上述例子后,不妨隔两天再去看看可否轻易理解那一份代码。本身写的代码尚且如此,他人的更不在话下了。
那么,这种随着项目愈来愈复杂,逻辑愈来愈多,咱们该怎么写代码呢?
前端其实有一种说法:咱们如今的”新“东西,都是其余领域玩过的。虽然看起来很气人,但这也是事实。当现状不知如何处理时,不妨参考下其余领域的解决方案。离前端比较近的就是后端了,那么后端是怎么管理的呢?最典型的设计就是MVC了。那么,前端能不能借鉴呢?
说干就干,以上面的需求为例,咱们试着用MVC的方式组织一下代码,看下和你刚才写的有什么不一样。
先来M
,也就是Model,数据层,对外提供接口能够获取相关的数据。这么组织的话,是否是蛮好懂的:
const model = (function() {
//相关数据
const _model = {
catLists: [
{
src: '1.jpg',
name: 'cat1',
count: 0
},
{
src: '2.jpg',
name: 'cat2',
count: 0
}
],
targetCatIndex: 0, //目前可被点击的是哪知喵在catLists中的索引
};
//获取getCatLists
function getCatLists() {
return _model.catLists;
}
//获取目标对象
function getTargetCatObj() {
return _model.catLists[_model.targetCatIndex];
}
//修改targetCatIndex
function setTargetCatIndex(name) {
_model.catLists.some((catObj, index) => {
if (catObj.name === name) {
_model.targetCatIndex = index;
return true
}
})
}
//目标对象点击数+1
function addTargetCatCount() {
const catObj = getTargetCatObj();
catObj.count += 1;
}
return {
getCatLists,
getTargetCatObj,
setTargetCatIndex,
addTargetCatCount
}
})();复制代码
在自执行函数中,设计了一个对象命名为_model
,经过闭包存储它。自执行函数返回一个对象,其中包含四个函数。四个函数执行后,能够返回或修改_model
中对应的数据。经过注释看其实仍是挺清楚的。
跟着是V
,也就是view层,负责页面渲染。这个可能复杂一点,但不想把它弄得太繁琐,不如就两个方法吧。就一个初始化的init()
和负责更新视图的render()
方法就好啦。
先肯定HTML模板:
<ul class="catList"></ul>
<section class="clickArea">
<img src="" class="catImage">
<p class="catCount"></p>
<p class="catName"></p>
</section>复制代码
再组织一下view层的代码:
const view = (function() {
//获取各个须要操做的DOM节点
const img = document.querySelector('.catImage');
const name = document.querySelector('.catName');
const count = document.querySelector('.catCount');
//初始化页面
function init(catLists, targetObj) {
const list = document.querySelector('.catList');
const fragment = document.createDocumentFragment();
//为ul添加对应的li
for (let i = 0, len = catLists.length; i < len; i++) {
const li = document.createElement('li');
const name = catLists[i].name;
li.innerHTML = name;
li.addEventListener('click', function() {
//以后会有controller相关的代码,其实就是换一只可点击的喵
controller.changeTargetCat(name);
});
fragment.appendChild(li);
}
list.appendChild(fragment);
img.addEventListener('click', function() {
//以后会有controller相关的代码,其实就是计数+1
controller.addCount(name);
});
render(targetObj);
}
//从新渲染页面
function render(targetObj) {
img.src = targetObj.src;
name.innerHTML = targetObj.name;
count.innerHTML = targetObj.count;
}
return {
init,
render
}
})();复制代码
view层的代码其实也很简单的,和model层的套路差很少,经过自执行函数结合闭包存储以后要操做的节点,对外暴露由两个方法组成的对象,分别是init
与render
。init
用于初始化话页面,render
用于从新渲染页面。里面调用了controller
,其实就是以后要介绍的controller。
最后是C
层,也就是controller,主要是用于逻辑相关的处理,算是整个设计里面的大脑。不过因为这项目比较简单,因此代码反而是最简单的:
const controller = {
addCount(name) {
//经过model的接口增长目标对象的计数
model.addTargetCatCount(name);
controller.renderView();
},
changeTargetCat(name) {
//经过model的接口修改目标索引
model.setTargetCatIndex(name);
controller.renderView();
},
init() {
//经过model的接口获取相关数据
const { getCatLists, getTargetCatObj } = model;
//传参并命令view层初始化
view.init(getCatLists(), getTargetCatObj());
},
renderView() {
//传参并命令view层从新渲染
view.render(model.getTargetCatObj());
}
};复制代码
controller的设计实际上是比较简陋的,只是一个包含了四个方法的对象。其中addCount
对应点击加一的操做,changeTargetCat
对应换猫的操做。上述两个方法实际上是改变了数据的,而只要数据发生了变化,一概调用renderView
从新渲染。
至此主要代码已经写完了,以后调用一下controller.init();
,就能够开心的撸猫,完成需求了。
若是以前你动手实现了上述需求的话,不妨对比一下咱们设计代码上的差异。也许你写的代码仍是以前那种“面条式”代码,但它也是可用的啊,并且还不用这么多代码呢,长得也还算能维护的样子,为什么要用这种繁琐的方式去阻止代码呢?
然而,按照以前说的,产品可能提更多的需求,下次可能会有点击到某个值时,自动切换猫,动态添加猫等等的新需求时,你如今的代码组织形式可否很快地完成需求?当往后修改某些需求时,不当心触发了潜藏的bug(常常有的状况),又是否能快速定位出问题并快速改好呢?
“面条式”代码常常是数据、视图与处理逻辑耦合起来的,很容易牵一发而动全身,当业务至关复杂的时候,开发可能还好说,维护简直是不可想象的。而你可能已经观察到了,遵循MVC设计的代码,虽然繁琐,但各层是彻底分开的,尽管数据与视图能够直接调用对方的接口进行交互,但都是必须经过控制层来作统一处理。数据、视图与处理逻辑解耦以后,代码的可阅读性与可维护性都是一个飞跃。经过牺牲空间(多写代码)来换取代码的可维护性与可拓展性,这是一笔划算的买卖。
说了半天,好像尚未说出为什么前端须要使用框架。然而在复杂的项目中,你赞成我经过MVC组织代码会比“面条式”代码好吗?若是赞成的话,将我刚才代码中不变的部分抽象起来(组件通信、报错处理等),千方百计提升渲染的性能(使用Virtual Dom),若是认为前端和其余不同,数据和视图仍是能够进行受控的交互(即MVVM),那么整合起来,不就是一个框架了吗?
其实必需要认可,人的脑力是有限的,一款产品的需求多是无限的。当这款产品已经让你没法掌握每一个细节,每位参与开发的同窗只能掌握局部细节,而其余部分只管调用是必然的事实。可是,如何肯定其余部分可信,调用不会出bug呢?这时候就该使用框架了。框架较大程度上能约束与规范每位开发者的行为,不按照框架的规定极可能就会报错,这样多人协做就有了基本的保证。
可是,不是说使用框架就是最佳实践。当项目不复杂的时候(好比一次性的活动页),咱们有足够能力去掌握项目的细节,那么使用框架反而不是好的选择。毕竟再好的框架在性能上都会有损失,而被框架的条条框框约束着,也老是使人不喜的。
简单地说:脑子不够,框架来凑;本身组织很差代码,靠框架来给咱们组织。阅读至此殊为不易,感谢各位看官,但愿本文对你有一点帮助,谢谢!