-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsearch.xml
72 lines (33 loc) · 547 KB
/
search.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
<?xml version="1.0" encoding="utf-8"?>
<search>
<entry>
<title>前端学习</title>
<link href="/2020/051612750.html"/>
<url>/2020/051612750.html</url>
<content type="html"><![CDATA[<h1 id="前端学习"><a href="#前端学习" class="headerlink" title="前端学习"></a>前端学习</h1><h1 id="HTML标签"><a href="#HTML标签" class="headerlink" title="HTML标签"></a>HTML标签</h1><h2 id="Web标准"><a href="#Web标准" class="headerlink" title="Web标准"></a>Web标准</h2><p>结构:用于对网页元素进行整理和分类</p><p>表现:用于设置网页元素的格式,颜色,大小等外观样式</p><p>行为:指网页模型的定义以及交互的编写.</p><p>最佳体验方案:结构,样式,行为相分离</p><h2 id="HTML基本结构标签"><a href="#HTML基本结构标签" class="headerlink" title="HTML基本结构标签"></a>HTML基本结构标签</h2><html></html> 根标签<head><link rel="stylesheet" href="/css/prism-tomorrow.css" type="text/css"></head> 文档的头部<p><titile></titile>文档的标题</p><body><script>!function(e){var c=Array.prototype.slice.call(document.querySelectorAll("img[data-original]"));function i(){for(var r=0;r<c.length;r++)t=c[r],0<=(n=t.getBoundingClientRect()).bottom&&0<=n.left&&n.top<=(e.innerHeight||document.documentElement.clientHeight)&&function(){var t,n,e,i,o=c[r];t=o,n=function(){c=c.filter(function(t){return o!==t})},e=new Image,i=t.getAttribute("data-original"),e.onload=function(){t.src=i,n&&n()},e.src=i}();var t,n}i(),e.addEventListener("scroll",function(){var t,n;t=i,n=e,clearTimeout(t.tId),t.tId=setTimeout(function(){t.call(n)},500)})}(this);</script><script>window.addEventListener("load",function(){var t=/\.(gif|jpg|jpeg|tiff|png)$/i,r=/^data:image\/[a-z]+;base64,/;Array.prototype.slice.call(document.querySelectorAll("img[data-original]")).forEach(function(a){var e=a.parentNode;"A"===e.tagName&&(e.href.match(t)||e.href.match(r))&&(e.href=a.dataset.original)})});</script></body> 文档的主体<h2 id="VS-Code-推荐插件"><a href="#VS-Code-推荐插件" class="headerlink" title="VS Code 推荐插件"></a>VS Code 推荐插件</h2><p>Open in Browser : 右击选择浏览器打开html文件</p><p>js-css-html formatter: 每次保存.都会自动格式化js,css,和html代码</p><p>atuo rename tag :自动重命名配对的html/xml标签</p><p>css peek :追踪至样式</p><h2 id="网页开发工具"><a href="#网页开发工具" class="headerlink" title="网页开发工具"></a>网页开发工具</h2><h3 id="1-文档类型声明标签"><a href="#1-文档类型声明标签" class="headerlink" title="1. 文档类型声明标签"></a>1. 文档类型声明标签</h3><p></!DOCTYPE></p><p>意思:当前页面采用的是html5版本来显示网页的</p><h3 id="2-lang语言种类"><a href="#2-lang语言种类" class="headerlink" title="2. lang语言种类"></a>2. lang语言种类</h3><html lang="en"> <p>告诉浏览器或者搜索引擎这是一个英文网站</p><h3 id="3-字符集"><a href="#3-字符集" class="headerlink" title="3.字符集"></a>3.字符集</h3><meta charset="utf-8"/><h2 id="HTML常用标签"><a href="#HTML常用标签" class="headerlink" title="HTML常用标签"></a>HTML常用标签</h2><h3 id="1-标题标签"><a href="#1-标题标签" class="headerlink" title="1.标题标签 - "></a>1.标题标签<h1> - <h6></h3><p>/<h1> 我是一级标题</h1></p><h3 id="2-段落标签"><a href="#2-段落标签" class="headerlink" title="2.段落标签"></a>2.段落标签<p></p></h3><p>/<p>我是一个标签段落</p></p><h3 id="3-换行标签"><a href="#3-换行标签" class="headerlink" title="3.换行标签"></a>3.换行标签<br/></h3><h3 id="4-文本格式化标签"><a href="#4-文本格式化标签" class="headerlink" title="4.文本格式化标签"></a>4.文本格式化标签</h3><p>加粗: <strong></strong>或者<b></b></p><p>倾斜:<em></em>或者<i></i></p><p>删除线:<del></del>或者<s></s></p><p>下划线:<ins></ins>或者<u></u></p><h3 id="5-和-标签"><a href="#5-和-标签" class="headerlink" title="5.和 标签"></a>5.<div>和<span> 标签</h3><p>用来布局</p><p>/<div>一行只能放一个<div><br>/一行可以放多个<span></p><h3 id="6-图像标签和路径"><a href="#6-图像标签和路径" class="headerlink" title="6.图像标签和路径"></a>6.图像标签和路径</h3><img><p>图片标签的属性</p><p>src 图片路径</p><p>alt 替换文本,图像不能显示时,显示的文字</p><p>title 提示文本,鼠标放在图像上,显示的文字</p><h3 id="7-超链接标签"><a href="#7-超链接标签" class="headerlink" title="7.超链接标签"></a>7.超链接标签</h3><a><p>属性:<br>href 用于指定链接目标的url地址</p><p>target 用于指定链接页面的打开方式</p><p>_self 当前窗口打开页面</p><p>_blank 新窗口打开页面</p><p>链接分类:外部链接,内部链接,空链接,下载链接,网页元素链接</p><p>锚点链接</p><p>/<a href="#XXX"></p><p>会跳转到id为xxx的位置去</p><h2 id="注释"><a href="#注释" class="headerlink" title="注释"></a>注释</h2><!-- 注释语句 --><p>快捷键 ctrl + /</p><h2 id="特殊字符"><a href="#特殊字符" class="headerlink" title="特殊字符"></a>特殊字符</h2><p>空格符 & nbsp;</p><p>大于号 & gt;</p><p>小于号 & lt;</p><h2 id="表格标签"><a href="#表格标签" class="headerlink" title="表格标签"></a>表格标签</h2><p>/<table>定义表格标签</p><p>/<tr>行 <th></th>表头单元格 </tr></p><p>/<tr>行 <td></td>列 </tr></p></table><h3 id="表格属性-不常用"><a href="#表格属性-不常用" class="headerlink" title="表格属性(不常用)"></a>表格属性(不常用)</h3><p>align left,right,center 规定表格相对周围元素的对齐方式</p><p>border 1或””” 是否有边框</p><p>cellpadding 像素值 规定单元边沿与其内容之间的空白</p><p>cellspacing 像素值 单元格之间的空白\</p><p>width 像素值或百分比 表格的宽度 </p><h3 id="表格结构标签"><a href="#表格结构标签" class="headerlink" title="表格结构标签"></a>表格结构标签</h3><thead> 表格的头部区域<tbody> 表格的主体区域<h3 id="合并单元格"><a href="#合并单元格" class="headerlink" title="合并单元格"></a>合并单元格</h3><p>跨行合并: rowspan”合并单元格的个数”</p><p>跨列合并: colspan”合并单元格的个数”</p><h3 id="三部曲"><a href="#三部曲" class="headerlink" title="三部曲"></a>三部曲</h3><ol><li>确定跨行还是跨列</li><li>找到目标单元格,写上合并方式</li><li>删除多余的单元格</li></ol><h2 id="列表标签"><a href="#列表标签" class="headerlink" title="列表标签"></a>列表标签</h2><h3 id="无序列表"><a href="#无序列表" class="headerlink" title="无序列表"></a>无序列表</h3><p>/<ul><br> <li>列表项</li></p><p> <li>列表项</li></p><p> <li>列表项</li></p></ul><h3 id="有序列表"><a href="#有序列表" class="headerlink" title="有序列表"></a>有序列表</h3><p>/<ol></p><p> <li>列表项</li></p><p> <li>列表项</li></p><p> <li>列表项</li></p></ol><h3 id="自定义列表"><a href="#自定义列表" class="headerlink" title="自定义列表"></a>自定义列表</h3><p>/<dl></p><dt>名词1</dt><dd>名词1解释1</dd><dd>名词1解释2</dd></dl><h2 id="表单标签"><a href="#表单标签" class="headerlink" title="表单标签"></a>表单标签</h2><h3 id="表单域"><a href="#表单域" class="headerlink" title="表单域"></a>表单域</h3><form> 会把它范围内的表单元素信息提交给服务器<h4 id="常用属性"><a href="#常用属性" class="headerlink" title="常用属性"></a>常用属性</h4><p>action url地址 指定接收并处理表单数据的服务器程序的url地址</p><p>method get/post 设置表单数据的提交方式</p><p>name 名称 用于指定表单的名称,以区一个页面中的多个表单域</p><h3 id="表单控件"><a href="#表单控件" class="headerlink" title="表单控件"></a>表单控件</h3><ol><li><p>input输入表单元素</p><p>type属性:</p></li></ol><p> button 按钮</p><p> checkbox 复选框</p><p> file 定义输入字段和浏览按钮,供文件上传</p><p> hidden 隐藏的输入字段</p><p> image 定义图像形式的提交按钮</p><p> password 定义密码字段</p><p> radio 定义单选按钮</p><p> reset 定义重置按钮</p><p> submit 定义提交按钮</p><p> text 定义单行的输入字段</p><p> 其他属性</p><p> name 定义input名称</p><p> value 规定input元素的值</p><p> checked 规定input元素首次加载时应当被选中</p><p> maxlength 规定输入字段中的字符的最大长度</p><p> <label>标签 绑定一个表单元素,当点击标签内的文本时,浏览器会自动将焦点转到或者选择对应的表单元素上</p><p> for属性要和input属性中的id名字一致</p><ol start="2"><li><p>select下拉表单元素</p><p>/<select></p><p> <option>选项1</option></p><p> <option>选项2</option></p><p> <option>选项3</option></p><p> <option>选项4</option></p><p> <option>选项5</option></p></select><p>selected=”selected” 当前项为默认选中项</p></li><li><p>textarea文本域元素</p></li></ol><h2 id="CSS-层叠样式表"><a href="#CSS-层叠样式表" class="headerlink" title="CSS 层叠样式表"></a>CSS 层叠样式表</h2><h3 id="一-简介"><a href="#一-简介" class="headerlink" title="一 简介"></a>一 简介</h3><p>主要设置html页面中的文本内容(字体,大小,对齐方式),图片的外形以及版面的布局和外观显示样式</p><h4 id="CSS语法规范"><a href="#CSS语法规范" class="headerlink" title="CSS语法规范"></a>CSS语法规范</h4><p>CSS规则有两个主要部分构成:选择器以及一条或多条声明</p><h3 id="二-CSS基础选择器"><a href="#二-CSS基础选择器" class="headerlink" title="二 CSS基础选择器"></a>二 CSS基础选择器</h3><h4 id="1标签选择器"><a href="#1标签选择器" class="headerlink" title="1标签选择器"></a>1标签选择器</h4><h4 id="2类选择器"><a href="#2类选择器" class="headerlink" title="2类选择器"></a>2类选择器</h4><p>语法</p><p>需要设置class属性</p><p>.class类名{</p><p>color: red</p><p>}</p><h5 id="多类名"><a href="#多类名" class="headerlink" title="多类名"></a>多类名</h5><p>/<div class="red font20">亚瑟</div></p><p>中间必须用空格分开</p><h4 id="3id选择器"><a href="#3id选择器" class="headerlink" title="3id选择器"></a>3id选择器</h4><p>#id名</p><p>{</p><p> 属性1:属性值1;</p><p>}</p><h4 id="4通配符选择器"><a href="#4通配符选择器" class="headerlink" title="4通配符选择器"></a>4通配符选择器</h4><p>表示选取页面中的所有元素</p><p>*{</p><p> 属性1:属性值1;</p><p>}</p><h3 id="三-字体属性"><a href="#三-字体属性" class="headerlink" title="三 ,字体属性"></a>三 ,字体属性</h3><p>字体系列</p><p>font-famliy: “微软雅黑 ,Microsoft YaHei”</p><p>字体大小</p><p>font-size: 30px;</p><p>字体粗细</p><p>font-weight: bold;</p><p>font-weight: 700;</p><p>文字样式</p><p>font-style: normal/italic(斜体);</p><p>复合属性写法</p><p>font: font-style font-weight font-size/line-height font-familiy;</p><p>eg: font: italic 700 16px ‘Microsoft yahei’;</p><p>必须保留大小和字体</p><h3 id="四-文本属性"><a href="#四-文本属性" class="headerlink" title="四,文本属性"></a>四,文本属性</h3><p>文本颜色</p><p>1.预定义的颜色值</p><p>2.十六进制</p><p>3.RGB代码</p><p>对齐文本</p><p>text-align: center/left/right;</p><p>装饰文本</p><p>text-decoration: underline/overline/line-through/none;</p><p>文本缩进</p><p>段落的首行缩进</p><p>text-indent: 2em;</p><p>em是一个相对单位,就是当前元素一个文字的大小</p><p>行间距</p><p>line-height设置行间的距离.可以控制文字行与行之间的距离</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587378250993.png" alt="1587378250993"></p><h3 id="五-CSS的引入方式"><a href="#五-CSS的引入方式" class="headerlink" title="五,CSS的引入方式"></a>五,CSS的引入方式</h3><p>内部样式表</p><p>就是写到html页面的内部.将所有的css代码抽取出来.单独放到一个<style>标签中</p><p>行内样式表</p><p>/<div style="color: red; font-size: 12px;">xxxxxx</div></p><p>外部样式表</p><p>单独写到css文件中,之后把css文件引入到html页面中使用</p><p>方式:</p><p>1.新建一个后缀名为.css的样式文件,把所有css代码都放入此文件中</p><p>2.在html页面中,使用<link>标签引入这个文件</p><h2 id="六-Emmet语法"><a href="#六-Emmet语法" class="headerlink" title="六,Emmet语法"></a>六,Emmet语法</h2><h3 id="1-快速生成HTML结构语法"><a href="#1-快速生成HTML结构语法" class="headerlink" title="1.快速生成HTML结构语法"></a>1.快速生成HTML结构语法</h3><ol><li>生成标签 直接输入标签名按tab键即可</li><li>如果想要生成多个相同标签,加上* 如:div*3+tab</li><li>如果有父子级关系的标签,可用> 比如ul>li</li><li>如果有兄弟关系,可用+ 比如div+p</li><li>如果生成带有类名或者id名字的,直接写.demo或者#two tab即可</li><li>如果生成的div类名是有序的,可以用自增符号$</li><li>如果想在生成的标签内部写内容可以用{}表示</li></ol><h3 id="2-快速生成CSS样式语法"><a href="#2-快速生成CSS样式语法" class="headerlink" title="2.快速生成CSS样式语法"></a>2.快速生成CSS样式语法</h3><p>CSS基本采取简写形式即可</p><p>例如 w200可生成width: 200px</p><p> lh26可生成line-height: 26px;</p><h3 id="3-快速格式化代码"><a href="#3-快速格式化代码" class="headerlink" title="3.快速格式化代码"></a>3.快速格式化代码</h3><h2 id="七-CSS的复合选择器"><a href="#七-CSS的复合选择器" class="headerlink" title="七,CSS的复合选择器"></a>七,CSS的复合选择器</h2><p>由多个基础选择器通过不同方式组合而成的</p><p>包括:后代选择器,子选择器,并集选择器,伪类选择器</p><h3 id="后代选择器"><a href="#后代选择器" class="headerlink" title="后代选择器"></a>后代选择器</h3><p>语法:</p><p>元素1 元素2 { 样式声明 }</p><p>表示选择元素1里面所有元素2</p><p>例如:</p><p>ol li a {</p><p> color: red</p><p>}</p><p>.nav ol li a</p><p>{</p><p> color: yellow</p><p>}</p><h3 id="子选择器"><a href="#子选择器" class="headerlink" title="子选择器"></a>子选择器</h3><p>只能选择作为某元素的最近一级子元素,</p><p>语法:</p><p>元素1 > 元素2 { 样式声明 }</p><p>.nav > a{ color: red}</p><h3 id="并集选择器"><a href="#并集选择器" class="headerlink" title="并集选择器"></a>并集选择器</h3><p>各选择器通过英文逗号,连接而成</p><p>元素1,元素2 { 样式声明 }</p><p>可以选择多组标签,同时定义相同的样式</p><h3 id="伪类选择器"><a href="#伪类选择器" class="headerlink" title="伪类选择器"></a>伪类选择器</h3><p>用于向某些选择器添加特殊的效果,比如给链接添加特殊效果或选第一个,第n个元素.</p><p>用冒号表示 如:hover </p><h4 id="链接伪类选择器"><a href="#链接伪类选择器" class="headerlink" title="链接伪类选择器"></a>链接伪类选择器</h4><p>a:link 选择所有未被访问的链接</p><p>a:visited 选择所有已被访问的链接 </p><p>a:hover 选择鼠标指针位于其上的链接</p><p>a:active 选择活动链接</p><p>注意事项</p><p> 保证生效,按照LVHA的顺序声明</p><h4 id="focus伪类选择器"><a href="#focus伪类选择器" class="headerlink" title=":focus伪类选择器"></a>:focus伪类选择器</h4><p>用于选取获得焦点的表单元素</p><p>input :focus{ xxxx}</p><h2 id="八-CSS的元素显示模式"><a href="#八-CSS的元素显示模式" class="headerlink" title="八,CSS的元素显示模式"></a>八,CSS的元素显示模式</h2><p>元素显示模式是元素以什么方式进行显示</p><p>HTML元素一般分为块元素和行内元素</p><h4 id="块元素"><a href="#块元素" class="headerlink" title="块元素"></a>块元素</h4><p>/<div> <ol> <ul> <li> <h1>-<h6> <p><br>特点:</p><ol><li>独占一行</li><li>高,宽,外边距,内边距可以控制</li><li>宽度是容器的100%</li><li>是一个容器及盒子,可以放行内或者块元素</li></ol><p>文字元素里面不可以放块级元素</p><h4 id="行内元素"><a href="#行内元素" class="headerlink" title="行内元素"></a>行内元素</h4><p><a> <strong> <b> <em> <i> <del> <s> <ins> <u> <span></p><p>特点:</p><ol><li>相邻行内元素在一行上,一行可以显示多个</li><li>高,宽接设置是无效的</li><li>默认宽度就是它本身内容的宽度</li><li>行内元素只能容纳文本或其他行内元素</li></ol><h4 id="行内块元素"><a href="#行内块元素" class="headerlink" title="行内块元素"></a>行内块元素</h4><p><img /> <input /> <td ></p><p>特点:</p><ol><li>一行可以显示多个</li><li>默认宽度是它本身内容的宽度</li><li>高度,行高,外边距以及内边距都可以控制</li></ol><h4 id="模式转换"><a href="#模式转换" class="headerlink" title="模式转换"></a>模式转换</h4><p>转化为块级元素 display:block</p><p>转化为行内元素 display:inline</p><p>转化为行内块 display: inline-block</p><h3 id="Snipaste基本使用"><a href="#Snipaste基本使用" class="headerlink" title="Snipaste基本使用"></a>Snipaste基本使用</h3><ol><li>F1截图</li><li>F3在桌面置顶显示</li><li>alt可以取色(Shift切换)</li><li>esc取消图片显示</li></ol><h4 id="垂直居中"><a href="#垂直居中" class="headerlink" title="垂直居中"></a>垂直居中</h4><p>文字的行高等于盒子的高度</p><h2 id="九-CSS的背景"><a href="#九-CSS的背景" class="headerlink" title="九,CSS的背景"></a>九,CSS的背景</h2><h4 id="背景颜色"><a href="#背景颜色" class="headerlink" title="背景颜色"></a>背景颜色</h4><h4 id="背景图片"><a href="#背景图片" class="headerlink" title="背景图片"></a>背景图片</h4><p>background-image: none | url(url)</p><h4 id="背景平铺"><a href="#背景平铺" class="headerlink" title="背景平铺"></a>背景平铺</h4><p>background-repeat: repeat | no-repeat | repeat-x | repeat-y;</p><h4 id="背景位置"><a href="#背景位置" class="headerlink" title="背景位置"></a>背景位置</h4><p>background-position: x y;</p><p>length 百分数</p><p>position top|center|bottom|left|right</p><p>填了第一个参数,第二个参数默认居中</p><p>x,y可以混合单位</p><p>例如: background-position: center top</p><h4 id="背景固定"><a href="#背景固定" class="headerlink" title="背景固定"></a>背景固定</h4><p>background-attachment: scroll | fixed</p><p>scroll 随内容滚动</p><p>fixed 图像固定</p><h4 id="复合写法"><a href="#复合写法" class="headerlink" title="复合写法"></a>复合写法</h4><p>例如:</p><p>background: 背景颜色 背景图片 背景平铺 背景图像滚动 背景图片位置;</p><h4 id="背景色半透明"><a href="#背景色半透明" class="headerlink" title="背景色半透明"></a>背景色半透明</h4><p>background: rgba(0,0,0,0.3);</p><h2 id="十-CSS的三大特性"><a href="#十-CSS的三大特性" class="headerlink" title="十,CSS的三大特性"></a>十,CSS的三大特性</h2><h3 id="1-层叠性"><a href="#1-层叠性" class="headerlink" title="1.层叠性"></a>1.层叠性</h3><p>原则:就近原则</p><h3 id="2-继承性"><a href="#2-继承性" class="headerlink" title="2.继承性"></a>2.继承性</h3><p>行高的继承性</p><p>body{ font:12px/1.5 Microsoft YaHei;}</p><p>此时子元素的行高是:当前子元素的文字大小*1.5</p><p>优势:子元素可以根据自己的文字大小自动调整行高</p><h3 id="3-优先级"><a href="#3-优先级" class="headerlink" title="3.优先级"></a>3.优先级</h3><p>继承或者* 0.0.0.0</p><p>元素选择器 0.0.0.1</p><p>类选择器,伪类选择器 0.0.1.0</p><p>id选择器 0.1.0.0</p><p>行内样式 style=”” 1.0.0.0</p><p>!important重要的 无穷大</p><h4 id="权重叠加"><a href="#权重叠加" class="headerlink" title="权重叠加"></a>权重叠加</h4><p>如果是复合选择器,则会有权重叠加</p><p>div ul li 0,0,0,3</p><p>.nav ul li 0,0,1,2</p><p>a:hover 0,0,1,1</p><p>.nav a 0,0,1,1</p><h2 id="盒子模型"><a href="#盒子模型" class="headerlink" title="盒子模型"></a>盒子模型</h2><h4 id="网页布局的本质"><a href="#网页布局的本质" class="headerlink" title="网页布局的本质"></a>网页布局的本质</h4><ol><li>准备好相关的网页元素</li><li>利用css设置好盒子样式,摆放到相应的位置</li><li>往盒子里装内容</li></ol><h4 id="盒子组成"><a href="#盒子组成" class="headerlink" title="盒子组成"></a>盒子组成</h4><p>包括:边框,外边距,内边距,实际内容</p><h3 id="边框"><a href="#边框" class="headerlink" title="边框"></a>边框</h3><p>border: border-width || border-style ||border-color;</p><p>实线solid,虚线dashed,点线dotted</p><h3 id="边框的复合写法"><a href="#边框的复合写法" class="headerlink" title="边框的复合写法"></a>边框的复合写法</h3><p>border: 1px solid red;没有顺序</p><p>分开写法</p><p>border-top: 1px solid red;</p><h3 id="表格细线边框"><a href="#表格细线边框" class="headerlink" title="表格细线边框"></a>表格细线边框</h3><p>border-collapse: collapse;</p><h3 id="边框会影响盒子的大小"><a href="#边框会影响盒子的大小" class="headerlink" title="边框会影响盒子的大小"></a>边框会影响盒子的大小</h3><h3 id="内边距padding"><a href="#内边距padding" class="headerlink" title="内边距padding"></a>内边距padding</h3><p>padding-left,padding-right,padding-top,padding-bottom</p><h4 id="padding-简写"><a href="#padding-简写" class="headerlink" title="padding 简写"></a>padding 简写</h4><p>padding: 5px; 上下左右5px内边距</p><p>padding: 5px 10px; 上下5px,左右10px</p><p>padding: 5px 10px 20px 上,左右,下</p><p>padding: 5px 10px 20px 30px 上 右 下 左</p><h4 id="padding会影响盒子模型的大小"><a href="#padding会影响盒子模型的大小" class="headerlink" title="padding会影响盒子模型的大小"></a>padding会影响盒子模型的大小</h4><p>但如果盒子本身没有指定width/height属性,则此时padding不会撑开盒子大小</p><h3 id="外边距"><a href="#外边距" class="headerlink" title="外边距"></a>外边距</h3><p>margin:设置外边距,控制盒子与盒子之间的距离</p><p>margin-left,margin-right,margin-top,margin-bottom</p><h4 id="水平居中"><a href="#水平居中" class="headerlink" title="水平居中"></a>水平居中</h4><p>需要满足的条件</p><ol><li>盒子必须指定了宽度</li><li>盒子左右的外边距都设置为auto</li></ol><p>常见的写法:</p><p>margin-left auot;margin-right: auto;</p><p>margin: auto;</p><p>margin:0 auto;</p><p>而行内元素和行内块元素水平居中给其父元素添加text-align:center即可</p><h4 id="外边距合并问题"><a href="#外边距合并问题" class="headerlink" title="外边距合并问题"></a>外边距合并问题</h4><p>嵌套块元素垂直外边距的塌陷</p><p>解决方法:</p><ol><li>可以为父元素定义上边框</li><li>可以为父元素定义内边距</li><li>可以为父元素添加overflow:hidden</li></ol><h4 id="清楚内外边距"><a href="#清楚内外边距" class="headerlink" title="清楚内外边距"></a>清楚内外边距</h4><p>网页元素很多都有默认的内外边距,而且不同浏览器默认的也不一致,因此在我们布局前首先要清除网页元素的内外边距</p><pre class=" language-css"><code class="language-css"><span class="token selector">*</span><span class="token punctuation">{</span> <span class="token property">padding</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">;</span> <span class="token property">margin</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">;</span><span class="token punctuation">}</span></code></pre><p>注意:行内元素尽量只设置左右内外边距</p><h2 id="PS基本操作"><a href="#PS基本操作" class="headerlink" title="PS基本操作"></a>PS基本操作</h2><p>Ctrl+R: 打开标尺</p><p>右击标尺,把单位改为像素</p><p>Ctrl+加号或者减号,放大和缩小视图</p><p>按住空格,拖动视图</p><p>用选区拖动,可以测量大小</p><p>Ctrl+D可以取消选区,或者在旁边空白处点击一下也可以取消选区</p><h3 id="圆角边框"><a href="#圆角边框" class="headerlink" title="圆角边框"></a>圆角边框</h3><hr><blockquote><p>border-radius: length;</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587914968836.png" alt="1587914968836"></p><p>圆形</p><blockquote><p>boder-radius:50%;</p></blockquote><p>length可以有四个值,分别代表左上,右上,右下,左下</p><h3 id="盒子阴影"><a href="#盒子阴影" class="headerlink" title="盒子阴影"></a>盒子阴影</h3><hr><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587915439949.png" alt="1587915439949"></p><h3 id="文字阴影"><a href="#文字阴影" class="headerlink" title="文字阴影"></a>文字阴影</h3><hr><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587915708829.png" alt="1587915708829"></p><h1 id="浮动"><a href="#浮动" class="headerlink" title="浮动"></a>浮动</h1><hr><blockquote><p> css提供了三种布局方式</p></blockquote><ul><li><p>普通流</p></li><li><p>浮动</p></li><li><p>定位</p></li></ul><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587916101627.png" alt="1587916101627"></p><h3 id="为什么需要浮动"><a href="#为什么需要浮动" class="headerlink" title="为什么需要浮动"></a>为什么需要浮动</h3><hr><p>浮动可以改变标签的默认排列方式</p><p>经典应用:让多个块级元素一行内排列显示</p><blockquote><p>第一准则: 多个块级元素纵向排列找标准流,多个块级元素横向排列找浮动</p></blockquote><h3 id="什么是浮动"><a href="#什么是浮动" class="headerlink" title="什么是浮动"></a>什么是浮动</h3><hr><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587916606194.png" alt="1587916606194"></p><h3 id="浮动的特性"><a href="#浮动的特性" class="headerlink" title="浮动的特性"></a>浮动的特性</h3><hr><ol><li><p>浮动元素会脱离标准流</p><ul><li>元素浮动到指定位置</li><li>元素原先位置不会保留(其他元素会占据)</li></ul></li><li><p>浮动的元素会在一行内显示,并且元素顶部对其</p></li><li><p>浮动的元素具有行内块元素的特性</p><p>如果块级元素没有设置宽度,默认宽度和父级元素一样宽,但是添加浮动后大小根据内容来决定</p></li></ol><blockquote><p>先用标准流的父元素排列上下位置,之后内部元素采取浮动排列左右位置</p></blockquote><blockquote><p>第二准则:先设置盒子的大小,后设置盒子的位置</p></blockquote><h3 id="清除浮动"><a href="#清除浮动" class="headerlink" title="清除浮动"></a>清除浮动</h3><hr><blockquote><p>为什么要清除浮动?</p></blockquote><p>由于父级盒子很多情况下,不方便给高度,但是子盒子浮动又不占有位置,最后父盒子高度为0,会影响到下面的标准流盒子</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587921902605.png" alt="1587921902605"></p><blockquote><p>清除浮动的方法</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587922013642.png" alt="1587922013642"></p><p>1.额外标签法</p><p>在浮动元素末尾添加一个空的标签.如<div style="clear:both"></div>之类的</p><p>添加的元素必须是块级元素</p><p>2.父级添加overflow</p><p>属性值可以是hiden,auto,scroll</p><p>缺点:无法显示溢出的部分</p><p>3 :after伪元素法</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587922545803.png" alt="1587922545803"></p><p>4.双伪元素清除浮动</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587922784625.png" alt="1587922784625"></p><blockquote><p> 总结</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587922980952.png" alt="1587922980952"></p><h3 id="PS切图"><a href="#PS切图" class="headerlink" title="PS切图"></a>PS切图</h3><hr><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587923074634.png" alt="1587923074634"></p><h1 id="定位"><a href="#定位" class="headerlink" title="定位"></a>定位</h1><hr><blockquote><p>为什么需要定位</p></blockquote><p>定位可以让盒子自由的在某个盒子内移动位置或者固定屏幕中某个位置,并且可以压住其他盒子</p><blockquote><p>定位组成</p></blockquote><p>定位=定位模式+边偏移</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587963095749.png" alt="1587963095749"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587963180127.png" alt="1587963180127"></p><blockquote><p>静态定位 static</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587963287263.png" alt="1587963287263"></p><blockquote><p>相对定位 relative</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587963557031.png" alt="1587963557031"></p><blockquote><p>绝对定位</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587964151826.png" alt="1587964151826"></p><h3 id="子绝父相"><a href="#子绝父相" class="headerlink" title="子绝父相"></a><strong>子绝父相</strong></h3><blockquote><p>固定定位</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587964945781.png" alt="1587964945781"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587965241390.png" alt="1587965241390"></p><blockquote><p>粘性定位</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587965457956.png" alt="1587965457956"></p><blockquote><p>总结</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587965538975.png" alt="1587965538975"></p><blockquote><p>定位叠放次序</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587965941318.png" alt="1587965941318"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587966267790.png" alt="1587966267790"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587966427719.png" alt="1587966427719"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587966668006.png" alt="1587966668006"></p><h3 id="元素的显示和隐藏"><a href="#元素的显示和隐藏" class="headerlink" title="元素的显示和隐藏"></a>元素的显示和隐藏</h3><hr><ol><li>display</li></ol><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587970222338.png" alt="1587970222338"></p><ol start="2"><li>visibility</li></ol><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587970340569.png" alt="1587970340569"></p><ol start="3"><li>overflow</li></ol><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587970692228.png" alt="1587970692228"></p><p>小结</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587970857981.png" alt="1587970857981"></p><h1 id="CSS高级"><a href="#CSS高级" class="headerlink" title="CSS高级"></a>CSS高级</h1><hr><h2 id="精灵图"><a href="#精灵图" class="headerlink" title="精灵图"></a>精灵图</h2><hr><h3 id="为什么需要精灵图"><a href="#为什么需要精灵图" class="headerlink" title="为什么需要精灵图"></a>为什么需要精灵图</h3><p> 一个网页中往往会应用很多小的背景图片作为修饰,当网页中的图像过多时,服务器就会频繁地接收和发送请求图片,造成服务器请求压力过大,这将大大降低页面的加载速度.</p><p> 因此,为了有效地减少服务器接收和发送请求的次数,提高页面的加载速度,出现了CSS精灵技术</p><p>核心原理:将网页中的一些小背景图像整合到一张大图中,这样服务器只需要一次请求就可以了.</p><h3 id="精灵图-sprites-的使用"><a href="#精灵图-sprites-的使用" class="headerlink" title="精灵图(sprites)的使用"></a>精灵图(sprites)的使用</h3><h4 id="精灵图的核心总结"><a href="#精灵图的核心总结" class="headerlink" title="精灵图的核心总结"></a>精灵图的核心总结</h4><ol><li>精灵图主要针对小的背景图片使用</li><li>主要借助于背景位置来实现, background-position</li><li>一般情况下精灵图都是负值</li></ol><h3 id="精灵图的缺点"><a href="#精灵图的缺点" class="headerlink" title="精灵图的缺点"></a>精灵图的缺点</h3><ol><li>图片文件比较大</li><li>图片本身放大缩小会失真</li><li>图片制作完毕更换复杂</li></ol><h2 id="字体图标"><a href="#字体图标" class="headerlink" title="字体图标"></a>字体图标</h2><hr><p>展示的是图标,本质属于字体</p><h3 id="字体图标的优点"><a href="#字体图标的优点" class="headerlink" title="字体图标的优点"></a>字体图标的优点</h3><p>轻量级:一个图标字体比一系列图像小</p><p>灵活性:本质是文字,可以随意改变颜色,产生阴影,透明,旋转效果</p><p>兼容性:基本兼容所有浏览器</p><p>注意:字体图标不能代替精灵技术,只是对图标部分的提升和优化</p><h3 id="字体图标的使用"><a href="#字体图标的使用" class="headerlink" title="字体图标的使用"></a>字体图标的使用</h3><p>先在一个.css文件中复制</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588764009055.png" alt="1588764009055"></p><p>再在style中添加</p><pre class=" language-css"><code class="language-css"><span class="token property">font-family</span><span class="token punctuation">:</span> <span class="token string">'iconfont'</span><span class="token punctuation">;</span></code></pre><h3 id="字体图标的追加"><a href="#字体图标的追加" class="headerlink" title="字体图标的追加"></a>字体图标的追加</h3><p>把压缩包的selection.json重新上传,添加新的图标之后再下载即可</p><h2 id="CSS三角"><a href="#CSS三角" class="headerlink" title="CSS三角"></a>CSS三角</h2><hr><pre class=" language-css"><code class="language-css"><span class="token selector"><span class="token class">.box1</span> </span><span class="token punctuation">{</span> <span class="token property">height</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">;</span> <span class="token property">width</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">;</span> <span class="token comment" spellcheck="true">/* 下左右透明,上保留颜色,即可实现三角 */</span> <span class="token property">border</span><span class="token punctuation">:</span> <span class="token number">50</span>px solid transparent<span class="token punctuation">;</span> <span class="token property">border-top-color</span><span class="token punctuation">:</span> blue<span class="token punctuation">;</span> <span class="token property">margin</span><span class="token punctuation">:</span> <span class="token number">100</span>px auto<span class="token punctuation">;</span> <span class="token punctuation">}</span></code></pre><h2 id="CSS用户样式界面"><a href="#CSS用户样式界面" class="headerlink" title="CSS用户样式界面"></a>CSS用户样式界面</h2><hr><h3 id="什么是界面样式"><a href="#什么是界面样式" class="headerlink" title="什么是界面样式"></a>什么是界面样式</h3><p>更改用户的鼠标样式</p><p>表单轮廓</p><p>防止表单域拖拽</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588853104453.png" alt="1588853104453"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588853368773.png" alt="1588853368773"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588853510266.png" alt="1588853510266"></p><h3 id="vertical-align"><a href="#vertical-align" class="headerlink" title="vertical align"></a>vertical align</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588853630086.png" alt="1588853630086"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588854084281.png" alt="1588854084281"></p><h3 id="溢出的文字行内块显示"><a href="#溢出的文字行内块显示" class="headerlink" title="溢出的文字行内块显示"></a>溢出的文字行内块显示</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588854366041.png" alt="1588854366041"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588854504558.png" alt="1588854504558"></p><h3 id="常见的布局技巧"><a href="#常见的布局技巧" class="headerlink" title="常见的布局技巧"></a>常见的布局技巧</h3><p>margin负值的运用</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588855134627.png" alt="1588855134627"></p><p>文字围绕浮动元素</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588855388282.png" alt="1588855388282"></p><p>行块的巧妙运用</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588855972239.png" alt="1588855972239"></p><p>css三角强化</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588857756785.png" alt="1588857756785"></p><h3 id="CSS初始化"><a href="#CSS初始化" class="headerlink" title="CSS初始化"></a>CSS初始化</h3><p>不同浏览器对有些标签的默认值是不同的,为了消除不同浏览器对html文本呈现的差异,照顾浏览器的兼容,我们需要对css初始化</p><p>京东页面初始化</p><pre class=" language-css"><code class="language-css"><span class="token comment" spellcheck="true">/* 把我们所有标签的内外边距清零 */</span><span class="token selector">* </span><span class="token punctuation">{</span> <span class="token property">margin</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">;</span> <span class="token property">padding</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">}</span><span class="token comment" spellcheck="true">/* em 和 i 斜体的文字不倾斜 */</span><span class="token selector">em,i </span><span class="token punctuation">{</span> <span class="token property">font-style</span><span class="token punctuation">:</span> normal<span class="token punctuation">}</span><span class="token comment" spellcheck="true">/* 去掉li 的小圆点 */</span><span class="token selector">li </span><span class="token punctuation">{</span> <span class="token property">list-style</span><span class="token punctuation">:</span> none<span class="token punctuation">}</span><span class="token selector">img </span><span class="token punctuation">{</span> <span class="token comment" spellcheck="true">/* border 0 照顾低版本浏览器 如果 图片外面包含了链接会有边框的问题 */</span> <span class="token property">border</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">;</span> <span class="token comment" spellcheck="true">/* 取消图片底侧有空白缝隙的问题 */</span> <span class="token property">vertical-align</span><span class="token punctuation">:</span> middle<span class="token punctuation">}</span><span class="token selector">button </span><span class="token punctuation">{</span> <span class="token comment" spellcheck="true">/* 当我们鼠标经过button 按钮的时候,鼠标变成小手 */</span> <span class="token property">cursor</span><span class="token punctuation">:</span> pointer<span class="token punctuation">}</span><span class="token selector">a </span><span class="token punctuation">{</span> <span class="token property">color</span><span class="token punctuation">:</span> <span class="token hexcode">#666</span><span class="token punctuation">;</span> <span class="token property">text-decoration</span><span class="token punctuation">:</span> none<span class="token punctuation">}</span><span class="token selector">a<span class="token pseudo-class">:hover</span> </span><span class="token punctuation">{</span> <span class="token property">color</span><span class="token punctuation">:</span> <span class="token hexcode">#c81623</span><span class="token punctuation">}</span><span class="token selector">button,input </span><span class="token punctuation">{</span> <span class="token comment" spellcheck="true">/* "\5B8B\4F53" 就是宋体的意思 这样浏览器兼容性比较好 */</span> <span class="token property">font-family</span><span class="token punctuation">:</span> Microsoft YaHei, Heiti SC, tahoma, arial, Hiragino Sans GB, <span class="token string">"\5B8B\4F53"</span>, sans-serif<span class="token punctuation">}</span><span class="token selector">body </span><span class="token punctuation">{</span> <span class="token comment" spellcheck="true">/* CSS3 抗锯齿形 让文字显示的更加清晰 */</span> <span class="token property">-webkit-font-smoothing</span><span class="token punctuation">:</span> antialiased<span class="token punctuation">;</span> <span class="token property">background-color</span><span class="token punctuation">:</span> <span class="token hexcode">#fff</span><span class="token punctuation">;</span> <span class="token property">font</span><span class="token punctuation">:</span> <span class="token number">12</span>px/<span class="token number">1.5</span> Microsoft YaHei, Heiti SC, tahoma, arial, Hiragino Sans GB, <span class="token string">"\5B8B\4F53"</span>, sans-serif<span class="token punctuation">;</span> <span class="token property">color</span><span class="token punctuation">:</span> <span class="token hexcode">#666</span><span class="token punctuation">}</span><span class="token selector"><span class="token class">.hide</span>,<span class="token class">.none</span> </span><span class="token punctuation">{</span> <span class="token property">display</span><span class="token punctuation">:</span> none<span class="token punctuation">}</span><span class="token comment" spellcheck="true">/* 清除浮动 */</span><span class="token selector"><span class="token class">.clearfix</span><span class="token pseudo-element">:after</span> </span><span class="token punctuation">{</span> <span class="token property">visibility</span><span class="token punctuation">:</span> hidden<span class="token punctuation">;</span> <span class="token property">clear</span><span class="token punctuation">:</span> both<span class="token punctuation">;</span> <span class="token property">display</span><span class="token punctuation">:</span> block<span class="token punctuation">;</span> <span class="token property">content</span><span class="token punctuation">:</span> <span class="token string">"."</span><span class="token punctuation">;</span> <span class="token property">height</span><span class="token punctuation">:</span> <span class="token number">0</span><span class="token punctuation">}</span><span class="token selector"><span class="token class">.clearfix</span> </span><span class="token punctuation">{</span> *<span class="token property">zoom</span><span class="token punctuation">:</span> <span class="token number">1</span><span class="token punctuation">}</span></code></pre><h1 id="HTML5-新特性"><a href="#HTML5-新特性" class="headerlink" title="HTML5 新特性"></a>HTML5 新特性</h1><h3 id="语言标签"><a href="#语言标签" class="headerlink" title="语言标签"></a>语言标签</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588862518617.png" alt="1588862518617"></p><h3 id="多媒体标签"><a href="#多媒体标签" class="headerlink" title="多媒体标签"></a>多媒体标签</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588863055085.png" alt="1588863055085"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588863234101.png" alt="1588863234101"></p><h3 id="Input表单"><a href="#Input表单" class="headerlink" title="Input表单"></a>Input表单</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588863551121.png" alt="1588863551121"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588863929750.png" alt="1588863929750"></p><h1 id="CSS3新特性"><a href="#CSS3新特性" class="headerlink" title="CSS3新特性"></a>CSS3新特性</h1><h3 id="新增选择性"><a href="#新增选择性" class="headerlink" title="新增选择性"></a>新增选择性</h3><h3 id="属性选择器"><a href="#属性选择器" class="headerlink" title="属性选择器"></a>属性选择器</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588864403854.png" alt="1588864403854"></p><h3 id="结构伪类选择器"><a href="#结构伪类选择器" class="headerlink" title="结构伪类选择器"></a>结构伪类选择器</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588865095706.png" alt="1588865095706"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588865684472.png" alt="1588865684472"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588866357778.png" alt="1588866357778"></p><h3 id="伪元素选择器"><a href="#伪元素选择器" class="headerlink" title="伪元素选择器"></a>伪元素选择器</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588866539921.png" alt="1588866539921"></p><h3 id="使用场景"><a href="#使用场景" class="headerlink" title="使用场景"></a>使用场景</h3><p>伪元素字体图标</p><p>仿土豆效果</p><p>清除浮动</p><h3 id="盒子模型-1"><a href="#盒子模型-1" class="headerlink" title="盒子模型"></a>盒子模型</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588913902316.png" alt="1588913902316"></p><h3 id="其他特性"><a href="#其他特性" class="headerlink" title="其他特性"></a>其他特性</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588914086390.png" alt="1588914086390"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588914260485.png" alt="1588914260485"></p><h3 id="过渡"><a href="#过渡" class="headerlink" title="过渡"></a>过渡</h3><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588914384648.png" alt="1588914384648"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588914623090.png" alt="1588914623090"></p><h3 id="进度条的布局"><a href="#进度条的布局" class="headerlink" title="进度条的布局"></a>进度条的布局</h3>]]></content>
</entry>
<entry>
<title>POI和easyExcel</title>
<link href="/2020/051661781.html"/>
<url>/2020/051661781.html</url>
<content type="html"><![CDATA[<h1 id="POI和easyExcel"><a href="#POI和easyExcel" class="headerlink" title="POI和easyExcel"></a>POI和easyExcel</h1><blockquote><p>常用场景</p></blockquote><p>1,将用户信息导出为excel表格</p><p>2,将excel表格中的信息录入到网站数据库</p><blockquote><p>Apache POI</p></blockquote><p>Apache POI官网:<a href="https://poi.apache.org/" target="_blank" rel="noopener">https://poi.apache.org/</a> </p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588096839977.png" alt="1588096839977"></p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588096956164.png" alt="1588096956164"></p><blockquote><p>easyExcel</p></blockquote><p>官网 <a href="https://alibaba-easyexcel.github.io/" target="_blank" rel="noopener">https://alibaba-easyexcel.github.io/</a></p><p>EasyExcel 是阿里巴巴开源的一个excel处理框架,<strong>以使用简单、节省内存著称</strong>。 </p><p>EasyExcel 能大大减少占用内存的主要原因是在解析 Excel 时没有将文件数据一次性全部加载到内存中, </p><p>而是从磁盘上一行行读取数据,逐个解析</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1588097473980.png" alt="1588097473980"></p><p>需要注意:2003 版本和 2007 版本存在兼容性的问题!03最多只有 65535 行!</p><p>03版本:</p><pre class=" language-java"><code class="language-java"><span class="token comment" spellcheck="true">// 创建新的Excel 工作簿 </span>Workbook workbook <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">HSSFWorkbook</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token comment" spellcheck="true">// 在Excel工作簿中建一工作表,其名为缺省值 Sheet0 </span>Sheet sheet <span class="token operator">=</span> workbook<span class="token punctuation">.</span><span class="token function">createSheet</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span><span class="token comment" spellcheck="true">// 创建行(row 1)</span>Row row1 <span class="token operator">=</span> sheet<span class="token punctuation">.</span><span class="token function">createRow</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span><span class="token comment" spellcheck="true">// 创建单元格(col 1-1)</span>Cell cell11 <span class="token operator">=</span> row1<span class="token punctuation">.</span><span class="token function">createCell</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span>cell11<span class="token punctuation">.</span><span class="token function">setCellValue</span><span class="token punctuation">(</span><span class="token string">"今日新增关注"</span><span class="token punctuation">)</span><span class="token punctuation">;</span></code></pre><p>07版本:</p><pre class=" language-java"><code class="language-java"><span class="token comment" spellcheck="true">// 创建新的Excel 工作簿, 只有对象变了</span>Workbook workbook <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">XSSFWorkbook</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span><span class="token comment" spellcheck="true">//其余一样</span></code></pre><blockquote><p>大文件写HSSF </p></blockquote><p>缺点:最多只能处理65536行,否则会抛出异常 </p><p>优点:过程中写入缓存,不操作磁盘,最后一次性写入磁盘,速度快</p><blockquote><p>大文件写XSSF </p></blockquote><p>缺点:写数据时速度非常慢,非常耗内存,也会发生内存溢出,如100万条 </p><p>优点:可以写较大的数据量,如20万条</p><blockquote><p>大文件写SXSSF</p></blockquote><p>优点:可以写非常大的数据量,如100万条甚至更多条,写数据速度快,占用更少的内存 </p><pre class=" language-java"><code class="language-java"><span class="token annotation punctuation">@Test</span> <span class="token keyword">public</span> <span class="token keyword">void</span> <span class="token function">test03S</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token keyword">throws</span> Exception<span class="token punctuation">{</span> <span class="token keyword">long</span> began<span class="token operator">=</span>System<span class="token punctuation">.</span><span class="token function">currentTimeMillis</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Workbook workbook <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">SXSSFWorkbook</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Sheet sheet <span class="token operator">=</span> workbook<span class="token punctuation">.</span><span class="token function">createSheet</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span> <span class="token punctuation">(</span><span class="token keyword">int</span> rowNum <span class="token operator">=</span> <span class="token number">0</span><span class="token punctuation">;</span> rowNum <span class="token operator"><</span> <span class="token number">100000</span><span class="token punctuation">;</span> rowNum<span class="token operator">++</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> <span class="token comment" spellcheck="true">//创建一个行</span> Row row <span class="token operator">=</span> sheet<span class="token punctuation">.</span><span class="token function">createRow</span><span class="token punctuation">(</span>rowNum<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span> <span class="token punctuation">(</span><span class="token keyword">int</span> cellNum <span class="token operator">=</span> <span class="token number">0</span><span class="token punctuation">;</span> cellNum <span class="token operator"><</span> <span class="token number">10</span><span class="token punctuation">;</span> cellNum<span class="token operator">++</span><span class="token punctuation">)</span> <span class="token punctuation">{</span><span class="token comment" spellcheck="true">//创建单元格</span> Cell cell <span class="token operator">=</span> row<span class="token punctuation">.</span><span class="token function">createCell</span><span class="token punctuation">(</span>cellNum<span class="token punctuation">)</span><span class="token punctuation">;</span> cell<span class="token punctuation">.</span><span class="token function">setCellValue</span><span class="token punctuation">(</span>cellNum<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span><span class="token string">"over"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> FileOutputStream fos<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">FileOutputStream</span><span class="token punctuation">(</span><span class="token keyword">new</span> <span class="token class-name">File</span><span class="token punctuation">(</span>path<span class="token operator">+</span><span class="token string">"test03SS"</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> workbook<span class="token punctuation">.</span><span class="token function">write</span><span class="token punctuation">(</span>fos<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">long</span> end<span class="token operator">=</span>System<span class="token punctuation">.</span><span class="token function">currentTimeMillis</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> fos<span class="token punctuation">.</span><span class="token function">close</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token comment" spellcheck="true">//清楚临时文件</span> <span class="token punctuation">(</span><span class="token punctuation">(</span>SXSSFWorkbook<span class="token punctuation">)</span>workbook<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">dispose</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span><span class="token punctuation">(</span><span class="token keyword">double</span><span class="token punctuation">)</span><span class="token punctuation">(</span>end<span class="token operator">-</span>began<span class="token punctuation">)</span><span class="token operator">/</span><span class="token number">1000</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span></code></pre><p><strong>注意:</strong> </p><p>过程中会产生临时文件,需要清理临时文件 </p><p>默认由100条记录被保存在内存中,如果超过这数量,则最前面的数据被写入临时文件 </p><p>如果想自定义内存中数据的数量,可以使用new SXSSFWorkbook ( 数量 )</p><blockquote><p>POI-EXCEL读</p></blockquote><pre class=" language-java"><code class="language-java"><span class="token annotation punctuation">@Test</span> <span class="token keyword">public</span> <span class="token keyword">void</span> <span class="token function">test03</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> Exception <span class="token punctuation">{</span> FileInputStream in<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">FileInputStream</span><span class="token punctuation">(</span>path<span class="token operator">+</span><span class="token string">"test03"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Workbook workbook <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">XSSFWorkbook</span><span class="token punctuation">(</span>in<span class="token punctuation">)</span><span class="token punctuation">;</span> Sheet sheet <span class="token operator">=</span> workbook<span class="token punctuation">.</span><span class="token function">getSheetAt</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Row row <span class="token operator">=</span> sheet<span class="token punctuation">.</span><span class="token function">getRow</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Cell cell <span class="token operator">=</span> row<span class="token punctuation">.</span><span class="token function">getCell</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>cell<span class="token punctuation">.</span><span class="token function">getNumericCellValue</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span></code></pre><blockquote><p>不同数据类型的读取</p></blockquote><pre class=" language-java"><code class="language-java"><span class="token annotation punctuation">@Test</span> <span class="token keyword">public</span> <span class="token keyword">void</span> <span class="token function">testDifType</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> Exception <span class="token punctuation">{</span> FileInputStream in<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">FileInputStream</span><span class="token punctuation">(</span>path<span class="token operator">+</span><span class="token string">"DifTest.xls"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Workbook workbook<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">HSSFWorkbook</span><span class="token punctuation">(</span>in<span class="token punctuation">)</span><span class="token punctuation">;</span> Sheet sheet <span class="token operator">=</span> workbook<span class="token punctuation">.</span><span class="token function">getSheetAt</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token comment" spellcheck="true">//获取标题</span> Row rowTitel <span class="token operator">=</span> sheet<span class="token punctuation">.</span><span class="token function">getRow</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">if</span><span class="token punctuation">(</span>rowTitel<span class="token operator">!=</span>null<span class="token punctuation">)</span> <span class="token punctuation">{</span> <span class="token keyword">int</span> cellCount <span class="token operator">=</span> rowTitel<span class="token punctuation">.</span><span class="token function">getPhysicalNumberOfCells</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span><span class="token punctuation">(</span><span class="token keyword">int</span> cellNum<span class="token operator">=</span><span class="token number">0</span><span class="token punctuation">;</span>cellNum<span class="token operator"><</span>cellCount<span class="token punctuation">;</span>cellNum<span class="token operator">++</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> Cell cell <span class="token operator">=</span> rowTitel<span class="token punctuation">.</span><span class="token function">getCell</span><span class="token punctuation">(</span>cellNum<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">if</span><span class="token punctuation">(</span>cell<span class="token operator">!=</span>null<span class="token punctuation">)</span> <span class="token punctuation">{</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span>cell<span class="token punctuation">.</span><span class="token function">getStringCellValue</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token operator">+</span><span class="token string">" | "</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token keyword">int</span> rowCount <span class="token operator">=</span> sheet<span class="token punctuation">.</span><span class="token function">getPhysicalNumberOfRows</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span><span class="token punctuation">(</span><span class="token keyword">int</span> rowNum<span class="token operator">=</span><span class="token number">1</span><span class="token punctuation">;</span>rowNum<span class="token operator"><</span>rowCount<span class="token punctuation">;</span>rowNum<span class="token operator">++</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> Row rowData <span class="token operator">=</span> sheet<span class="token punctuation">.</span><span class="token function">getRow</span><span class="token punctuation">(</span>rowNum<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">if</span><span class="token punctuation">(</span>rowData<span class="token operator">!=</span>null<span class="token punctuation">)</span> <span class="token punctuation">{</span> <span class="token keyword">int</span> cellCount <span class="token operator">=</span> rowData<span class="token punctuation">.</span><span class="token function">getPhysicalNumberOfCells</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span><span class="token punctuation">(</span><span class="token keyword">int</span> cellNum<span class="token operator">=</span><span class="token number">0</span><span class="token punctuation">;</span>cellNum<span class="token operator"><</span>cellCount<span class="token punctuation">;</span>cellNum<span class="token operator">++</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> Cell cell <span class="token operator">=</span> rowData<span class="token punctuation">.</span><span class="token function">getCell</span><span class="token punctuation">(</span>cellNum<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">if</span><span class="token punctuation">(</span>cell<span class="token operator">!=</span>null<span class="token punctuation">)</span> <span class="token punctuation">{</span> <span class="token keyword">int</span> cellType <span class="token operator">=</span> cell<span class="token punctuation">.</span><span class="token function">getCellType</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> String cellValue<span class="token operator">=</span><span class="token string">""</span><span class="token punctuation">;</span> <span class="token comment" spellcheck="true">//判断数据类型</span> <span class="token keyword">switch</span> <span class="token punctuation">(</span>cellType<span class="token punctuation">)</span><span class="token punctuation">{</span> <span class="token keyword">case</span> HSSFCell<span class="token punctuation">.</span>CELL_TYPE_STRING<span class="token operator">:</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span><span class="token string">"【STRING】"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> cellValue<span class="token operator">=</span>cell<span class="token punctuation">.</span><span class="token function">getStringCellValue</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">break</span><span class="token punctuation">;</span> <span class="token keyword">case</span> HSSFCell<span class="token punctuation">.</span>CELL_TYPE_BOOLEAN<span class="token operator">:</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span><span class="token string">"【BOOLEAN】"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> cellValue<span class="token operator">=</span>String<span class="token punctuation">.</span><span class="token function">valueOf</span><span class="token punctuation">(</span>cell<span class="token punctuation">.</span><span class="token function">getBooleanCellValue</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">break</span><span class="token punctuation">;</span> <span class="token keyword">case</span> HSSFCell<span class="token punctuation">.</span>CELL_TYPE_BLANK<span class="token operator">:</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span><span class="token string">"【BLANK】"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">break</span><span class="token punctuation">;</span> <span class="token keyword">case</span> HSSFCell<span class="token punctuation">.</span>CELL_TYPE_NUMERIC<span class="token operator">:</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span><span class="token string">"【NUMERIC】"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">if</span><span class="token punctuation">(</span>HSSFDateUtil<span class="token punctuation">.</span><span class="token function">isCellDateFormatted</span><span class="token punctuation">(</span>cell<span class="token punctuation">)</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> <span class="token comment" spellcheck="true">//日期得转换格式</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span><span class="token string">"【日期】"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Date date <span class="token operator">=</span> cell<span class="token punctuation">.</span><span class="token function">getDateCellValue</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> cellValue<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">DateTime</span><span class="token punctuation">(</span>date<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">toString</span><span class="token punctuation">(</span><span class="token string">"yyyy-MM-dd"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span><span class="token keyword">else</span> <span class="token punctuation">{</span> <span class="token comment" spellcheck="true">//用字符串方式防止数据过大</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span><span class="token string">"转化成字符串"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> cell<span class="token punctuation">.</span><span class="token function">setCellType</span><span class="token punctuation">(</span>HSSFCell<span class="token punctuation">.</span>CELL_TYPE_STRING<span class="token punctuation">)</span><span class="token punctuation">;</span> cellValue<span class="token operator">=</span>cell<span class="token punctuation">.</span><span class="token function">toString</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token keyword">break</span><span class="token punctuation">;</span> <span class="token keyword">case</span> HSSFCell<span class="token punctuation">.</span>CELL_TYPE_ERROR<span class="token operator">:</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">print</span><span class="token punctuation">(</span><span class="token string">"【数据类型错误】"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">break</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>cellValue<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span> in<span class="token punctuation">.</span><span class="token function">close</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span></code></pre><p>获取公式</p><pre class=" language-java"><code class="language-java"><span class="token annotation punctuation">@Test</span> <span class="token keyword">public</span> <span class="token keyword">void</span> <span class="token function">testFormula</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token keyword">throws</span> Exception <span class="token punctuation">{</span> FileInputStream in <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">FileInputStream</span><span class="token punctuation">(</span>path<span class="token operator">+</span><span class="token string">"公式.xls"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Workbook workbook<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">HSSFWorkbook</span><span class="token punctuation">(</span>in<span class="token punctuation">)</span><span class="token punctuation">;</span> Sheet sheet <span class="token operator">=</span> workbook<span class="token punctuation">.</span><span class="token function">getSheetAt</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Row row <span class="token operator">=</span> sheet<span class="token punctuation">.</span><span class="token function">getRow</span><span class="token punctuation">(</span><span class="token number">3</span><span class="token punctuation">)</span><span class="token punctuation">;</span> Cell cell <span class="token operator">=</span> row<span class="token punctuation">.</span><span class="token function">getCell</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token comment" spellcheck="true">//公式计算器</span> FormulaEvaluator formulaEvaluator<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">HSSFFormulaEvaluator</span><span class="token punctuation">(</span><span class="token punctuation">(</span>HSSFWorkbook<span class="token punctuation">)</span> workbook<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">if</span><span class="token punctuation">(</span>cell<span class="token operator">!=</span>null<span class="token punctuation">)</span> <span class="token punctuation">{</span> <span class="token keyword">int</span> cellType <span class="token operator">=</span> cell<span class="token punctuation">.</span><span class="token function">getCellType</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">switch</span> <span class="token punctuation">(</span>cellType<span class="token punctuation">)</span><span class="token punctuation">{</span> <span class="token keyword">case</span> HSSFCell<span class="token punctuation">.</span>CELL_TYPE_FORMULA<span class="token operator">:</span> String Formula <span class="token operator">=</span> cell<span class="token punctuation">.</span><span class="token function">getCellFormula</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>Formula<span class="token punctuation">)</span><span class="token punctuation">;</span> CellValue evaluate <span class="token operator">=</span> formulaEvaluator<span class="token punctuation">.</span><span class="token function">evaluate</span><span class="token punctuation">(</span>cell<span class="token punctuation">)</span><span class="token punctuation">;</span> String cellValue <span class="token operator">=</span> evaluate<span class="token punctuation">.</span><span class="token function">formatAsString</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>cellValue<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">break</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span></code></pre><blockquote><p>easyExcel</p></blockquote><p>简单的写入</p><pre class=" language-java"><code class="language-java"><span class="token annotation punctuation">@Data</span><span class="token keyword">public</span> <span class="token keyword">class</span> <span class="token class-name">DemoData</span> <span class="token punctuation">{</span> <span class="token annotation punctuation">@ExcelProperty</span><span class="token punctuation">(</span><span class="token string">"字符串标题"</span><span class="token punctuation">)</span> <span class="token keyword">private</span> String string<span class="token punctuation">;</span> <span class="token annotation punctuation">@ExcelProperty</span><span class="token punctuation">(</span><span class="token string">"日期标题"</span><span class="token punctuation">)</span> <span class="token keyword">private</span> Date date<span class="token punctuation">;</span> <span class="token annotation punctuation">@ExcelProperty</span><span class="token punctuation">(</span><span class="token string">"数字标题"</span><span class="token punctuation">)</span> <span class="token keyword">private</span> Double doubleData<span class="token punctuation">;</span> <span class="token comment" spellcheck="true">/** * 忽略这个字段 */</span> <span class="token annotation punctuation">@ExcelIgnore</span> <span class="token keyword">private</span> String ignore<span class="token punctuation">;</span><span class="token punctuation">}</span></code></pre><pre class=" language-java"><code class="language-java"><span class="token keyword">static</span> String path<span class="token operator">=</span><span class="token string">"C:\\Users\\CR553\\IdeaProjects\\excel-poi\\"</span><span class="token punctuation">;</span> <span class="token keyword">private</span> List<span class="token operator"><</span>DemoData<span class="token operator">></span> <span class="token function">data</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> List<span class="token operator"><</span>DemoData<span class="token operator">></span> list <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">ArrayList</span><span class="token operator"><</span>DemoData<span class="token operator">></span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span> <span class="token punctuation">(</span><span class="token keyword">int</span> i <span class="token operator">=</span> <span class="token number">0</span><span class="token punctuation">;</span> i <span class="token operator"><</span> <span class="token number">10</span><span class="token punctuation">;</span> i<span class="token operator">++</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> DemoData data <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">DemoData</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> data<span class="token punctuation">.</span><span class="token function">setString</span><span class="token punctuation">(</span><span class="token string">"字符串"</span> <span class="token operator">+</span> i<span class="token punctuation">)</span><span class="token punctuation">;</span> data<span class="token punctuation">.</span><span class="token function">setDate</span><span class="token punctuation">(</span><span class="token keyword">new</span> <span class="token class-name">Date</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> data<span class="token punctuation">.</span><span class="token function">setDoubleData</span><span class="token punctuation">(</span><span class="token number">0.56</span><span class="token punctuation">)</span><span class="token punctuation">;</span> list<span class="token punctuation">.</span><span class="token function">add</span><span class="token punctuation">(</span>data<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token keyword">return</span> list<span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">public</span> <span class="token keyword">void</span> <span class="token function">simpleWrite</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> <span class="token comment" spellcheck="true">// 写法1</span> String fileName <span class="token operator">=</span> path<span class="token operator">+</span><span class="token string">"easyE.xlsx"</span><span class="token punctuation">;</span> EasyExcel<span class="token punctuation">.</span><span class="token function">write</span><span class="token punctuation">(</span>fileName<span class="token punctuation">,</span> DemoData<span class="token punctuation">.</span><span class="token keyword">class</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">sheet</span><span class="token punctuation">(</span><span class="token string">"模板"</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">doWrite</span><span class="token punctuation">(</span><span class="token function">data</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span></code></pre><p>具体参照官方文档</p><p><a href="https://alibaba-easyexcel.github.io/index.html" target="_blank" rel="noopener">https://alibaba-easyexcel.github.io/index.html</a></p><p>任务:</p><p>阅读easyExcel的官方文档</p><p>测试easyExcel的APi</p>]]></content>
</entry>
<entry>
<title>JavaScript</title>
<link href="/2020/051618006.html"/>
<url>/2020/051618006.html</url>
<content type="html"><![CDATA[<h1 id="JavaScript"><a href="#JavaScript" class="headerlink" title="JavaScript"></a>JavaScript</h1><h1 id="快速入门"><a href="#快速入门" class="headerlink" title="快速入门"></a>快速入门</h1><h2 id="strict模式"><a href="#strict模式" class="headerlink" title="strict模式"></a>strict模式</h2><p>如果一个变量没有通过<code>var</code>申明就被使用,那么该变量就自动被申明为全局变量:</p><pre><code>i = 10; // i现在是全局变量</code></pre><p>在同一个页面的不同的JavaScript文件中,如果都不用<code>var</code>申明,恰好都使用了变量<code>i</code>,将造成变量<code>i</code>互相影响,产生难以调试的错误结果。</p><p>启用strict模式的方法是在JavaScript代码的第一行写上:</p><pre><code>'use strict';</code></pre><h2 id="字符串"><a href="#字符串" class="headerlink" title="字符串"></a>字符串</h2><p>JavaScript的字符串就是用<code>''</code>或<code>""</code>括起来的字符表示。</p><p>如果字符串内部既包含<code>'</code>又包含<code>"</code>怎么办?可以用转义字符<code>\</code>来标识,比如:</p><pre><code>'I\'m \"OK\"!';</code></pre><p>表示的字符串内容是:<code>I'm "OK"!</code></p><p>由于多行字符串用<code>\n</code>写起来比较费事,所以最新的ES6标准新增了一种多行字符串的表示方法,用反引号 ** … ** 表示:</p><pre><code>`这是一个多行字符串`;</code></pre><p>ES6新增了一种模板字符串,表示方法和上面的多行字符串一样,但是它会自动替换字符串中的变量:</p><pre><code>var name = '小明';var age = 20;var message = `你好, ${name}, 你今年${age}岁了!`;alert(message);</code></pre><h3 id="操作字符串"><a href="#操作字符串" class="headerlink" title="操作字符串"></a>操作字符串</h3><p><em>需要特别注意的是</em>,字符串是不可变的,如果对字符串的某个索引赋值,不会有任何错误,但是,也没有任何效果:</p><pre><code>var s = 'Test';s[0] = 'X';alert(s); /</code></pre><h3 id="toUpperCase"><a href="#toUpperCase" class="headerlink" title="toUpperCase"></a>toUpperCase</h3><p><code>toUpperCase()</code>把一个字符串全部变为大写:</p><pre><code>var s = 'Hello';s.toUpperCase(); // 返回'HELLO'</code></pre><h3 id="toLowerCase"><a href="#toLowerCase" class="headerlink" title="toLowerCase"></a>toLowerCase</h3><p><code>toLowerCase()</code>把一个字符串全部变为小写:</p><pre><code>var s = 'Hello';var lower = s.toLowerCase(); // 返回'hello'并赋值给变量lowerlower; // 'hello'</code></pre><h3 id="indexOf"><a href="#indexOf" class="headerlink" title="indexOf"></a>indexOf</h3><p><code>indexOf()</code>会搜索指定字符串出现的位置:</p><pre><code>var s = 'hello, world';s.indexOf('world'); // 返回7s.indexOf('World'); // 没有找到指定的子串,返回-1</code></pre><h3 id="substring"><a href="#substring" class="headerlink" title="substring"></a>substring</h3><p><code>substring()</code>返回指定索引区间的子串:</p><pre><code>var s = 'hello, world's.substring(0, 5); // 从索引0开始到5(不包括5),返回'hello's.substring(7); // 从索引7开始到结束,返回'world'</code></pre><h2 id="数组"><a href="#数组" class="headerlink" title="数组"></a>数组</h2><p>JavaScript的<code>Array</code>可以包含任意数据类型,并通过索引来访问每个元素。</p><p>要取得<code>Array</code>的长度,直接访问<code>length</code>属性:</p><pre><code>var arr = [1, 2, 3.14, 'Hello', null, true];arr.length; // 6</code></pre><p><em>请注意</em>,直接给<code>Array</code>的<code>length</code>赋一个新的值会导致<code>Array</code>大小的变化:</p><pre><code>var arr = [1, 2, 3];arr.length; // 3arr.length = 6;arr; // arr变为[1, 2, 3, undefined, undefined, undefined]arr.length = 2;arr; // arr变为[1, 2]</code></pre><p><code>Array</code>可以通过索引把对应的元素修改为新的值,因此,对<code>Array</code>的索引进行赋值会直接修改这个<code>Array</code>:</p><pre><code>var arr = ['A', 'B', 'C'];arr[1] = 99;arr; // arr现在变为['A', 99, 'C']</code></pre><p><em>请注意</em>,如果通过索引赋值时,索引超过了范围,同样会引起<code>Array</code>大小的变化:</p><pre><code>var arr = [1, 2, 3];arr[5] = 'x';arr; // arr变为[1, 2, 3, undefined, undefined, 'x']</code></pre><h3 id="indexOf-1"><a href="#indexOf-1" class="headerlink" title="indexOf"></a>indexOf</h3><p>与String类似,<code>Array</code>也可以通过<code>indexOf()</code>来搜索一个指定的元素的位置:</p><pre><code>var arr = [10, 20, '30', 'xyz'];arr.indexOf(10); // 元素10的索引为0arr.indexOf(20); // 元素20的索引为1arr.indexOf(30); // 元素30没有找到,返回-1arr.indexOf('30'); // 元素'30'的索引为2</code></pre><p>注意了,数字<code>30</code>和字符串<code>'30'</code>是不同的元素。</p><h3 id="slice"><a href="#slice" class="headerlink" title="slice"></a>slice</h3><p><code>slice()</code>就是对应String的<code>substring()</code>版本,它截取<code>Array</code>的部分元素,然后返回一个新的<code>Array</code>:</p><pre><code>var arr = ['A', 'B', 'C', 'D', 'E', 'F', 'G'];arr.slice(0, 3); // 从索引0开始,到索引3结束,但不包括索引3: ['A', 'B', 'C']arr.slice(3); // 从索引3开始到结束: ['D', 'E', 'F', 'G']</code></pre><p>注意到<code>slice()</code>的起止参数包括开始索引,不包括结束索引。</p><p>如果不给<code>slice()</code>传递任何参数,它就会从头到尾截取所有元素。利用这一点,我们可以很容易地复制一个<code>Array</code>:</p><pre><code>var arr = ['A', 'B', 'C', 'D', 'E', 'F', 'G'];var aCopy = arr.slice();aCopy; // ['A', 'B', 'C', 'D', 'E', 'F', 'G']aCopy === arr; // false</code></pre><h3 id="push和pop"><a href="#push和pop" class="headerlink" title="push和pop"></a>push和pop</h3><p><code>push()</code>向<code>Array</code>的末尾添加若干元素,<code>pop()</code>则把<code>Array</code>的最后一个元素删除掉:</p><pre><code>var arr = [1, 2];arr.push('A', 'B'); // 返回Array新的长度: 4arr; // [1, 2, 'A', 'B']arr.pop(); // pop()返回'B'arr; // [1, 2, 'A']arr.pop(); arr.pop(); arr.pop(); // 连续pop 3次arr; // []arr.pop(); // 空数组继续pop不会报错,而是返回undefinedarr; // []</code></pre><h3 id="unshift和shift"><a href="#unshift和shift" class="headerlink" title="unshift和shift"></a>unshift和shift</h3><p>如果要往<code>Array</code>的头部添加若干元素,使用<code>unshift()</code>方法,<code>shift()</code>方法则把<code>Array</code>的第一个元素删掉:</p><pre><code>var arr = [1, 2];arr.unshift('A', 'B'); // 返回Array新的长度: 4arr; // ['A', 'B', 1, 2]arr.shift(); // 'A'arr; // ['B', 1, 2]arr.shift(); arr.shift(); arr.shift(); // 连续shift 3次arr; // []arr.shift(); // 空数组继续shift不会报错,而是返回undefinedarr; // []</code></pre><h3 id="sort"><a href="#sort" class="headerlink" title="sort"></a>sort</h3><p><code>sort()</code>可以对当前<code>Array</code>进行排序,它会直接修改当前<code>Array</code>的元素位置,直接调用时,按照默认顺序排序:</p><pre><code>var arr = ['B', 'C', 'A'];arr.sort();arr; // ['A', 'B', 'C']</code></pre><p>能否按照我们自己指定的顺序排序呢?完全可以,我们将在后面的函数中讲到。</p><h3 id="reverse"><a href="#reverse" class="headerlink" title="reverse"></a>reverse</h3><p><code>reverse()</code>把整个<code>Array</code>的元素给掉个个,也就是反转:</p><pre><code>var arr = ['one', 'two', 'three'];arr.reverse(); arr; // ['three', 'two', 'one']</code></pre><h3 id="splice"><a href="#splice" class="headerlink" title="splice"></a>splice</h3><p><code>splice()</code>方法是修改<code>Array</code>的“万能方法”,它可以从指定的索引开始删除若干元素,然后再从该位置添加若干元素:</p><pre><code>var arr = ['Microsoft', 'Apple', 'Yahoo', 'AOL', 'Excite', 'Oracle'];// 从索引2开始删除3个元素,然后再添加两个元素:arr.splice(2, 3, 'Google', 'Facebook'); // 返回删除的元素 ['Yahoo', 'AOL', 'Excite']arr; // ['Microsoft', 'Apple', 'Google', 'Facebook', 'Oracle']// 只删除,不添加:arr.splice(2, 2); // ['Google', 'Facebook']arr; // ['Microsoft', 'Apple', 'Oracle']// 只添加,不删除:arr.splice(2, 0, 'Google', 'Facebook'); // 返回[],因为没有删除任何元素arr; // ['Microsoft', 'Apple', 'Google', 'Facebook', 'Oracle']</code></pre><h3 id="concat"><a href="#concat" class="headerlink" title="concat"></a>concat</h3><p><code>concat()</code>方法把当前的<code>Array</code>和另一个<code>Array</code>连接起来,并返回一个新的<code>Array</code>:</p><pre><code>var arr = ['A', 'B', 'C'];var added = arr.concat([1, 2, 3]);added; // ['A', 'B', 'C', 1, 2, 3]arr; // ['A', 'B', 'C']</code></pre><p><em>请注意</em>,<code>concat()</code>方法并没有修改当前<code>Array</code>,而是返回了一个新的<code>Array</code>。</p><p>实际上,<code>concat()</code>方法可以接收任意个元素和<code>Array</code>,并且自动把<code>Array</code>拆开,然后全部添加到新的<code>Array</code>里:</p><pre><code>var arr = ['A', 'B', 'C'];arr.concat(1, 2, [3, 4]); // ['A', 'B', 'C', 1, 2, 3, 4]</code></pre><h3 id="join"><a href="#join" class="headerlink" title="join"></a>join</h3><p><code>join()</code>方法是一个非常实用的方法,它把当前<code>Array</code>的每个元素都用指定的字符串连接起来,然后返回连接后的字符串:</p><pre><code>var arr = ['A', 'B', 'C', 1, 2, 3];arr.join('-'); // 'A-B-C-1-2-3'</code></pre><p>如果<code>Array</code>的元素不是字符串,将自动转换为字符串后再连接。</p><h3 id="多维数组"><a href="#多维数组" class="headerlink" title="多维数组"></a>多维数组</h3><p>如果数组的某个元素又是一个<code>Array</code>,则可以形成多维数组,例如:</p><pre><code>var arr = [[1, 2, 3], [400, 500, 600], '-'];</code></pre><p>上述<code>Array</code>包含3个元素,其中头两个元素本身也是<code>Array</code>。</p><h2 id="对象"><a href="#对象" class="headerlink" title="对象"></a>对象</h2><p>JavaScript的对象是一种无序的集合数据类型,它由若干键值对组成</p><pre><code>var xiaoming = { name: '小明', birth: 1990, school: 'No.1 Middle School', height: 1.70, weight: 65, score: null};</code></pre><p>JavaScript用一个<code>{...}</code>表示一个对象,键值对以<code>xxx: xxx</code>形式申明,用<code>,</code>隔开。注意,最后一个键值对不需要在末尾加<code>,</code>,如果加了,有的浏览器(如低版本的IE)将报错。</p><pre><code>var xiaohong = { name: '小红', 'middle-school': 'No.1 Middle School'};</code></pre><p><code>xiaohong</code>的属性名<code>middle-school</code>不是一个有效的变量,就需要用<code>''</code>括起来。访问这个属性也无法使用<code>.</code>操作符,必须用<code>['xxx']</code>来访问:</p><pre><code>xiaohong['middle-school']; // 'No.1 Middle School'xiaohong['name']; // '小红'xiaohong.name; // '小红'</code></pre><p>也可以用<code>xiaohong['name']</code>来访问<code>xiaohong</code>的<code>name</code>属性,不过<code>xiaohong.name</code>的写法更简洁。我们在编写JavaScript代码的时候,属性名尽量使用标准的变量名,这样就可以直接通过<code>object.prop</code>的形式访问一个属性了。</p><p>由于JavaScript的对象是动态类型,你可以自由地给一个对象添加或删除属性:</p><pre><code>var xiaoming = { name: '小明'};xiaoming.age; // undefinedxiaoming.age = 18; // 新增一个age属性xiaoming.age; // 18delete xiaoming.age; // 删除age属性xiaoming.age; // undefineddelete xiaoming['name']; // 删除name属性xiaoming.name; // undefineddelete xiaoming.school; // 删除一个不存在的school属性也不会报错</code></pre><p>要判断一个属性是否是<code>xiaoming</code>自身拥有的,而不是继承得到的,可以用<code>hasOwnProperty()</code>方法:</p><pre><code>var xiaoming = { name: '小明'};xiaoming.hasOwnProperty('name'); // truexiaoming.hasOwnProperty('toString'); // false</code></pre><h2 id="条件判断"><a href="#条件判断" class="headerlink" title="条件判断"></a>条件判断</h2><p>JavaScript使用<code>if () { ... } else { ... }</code>来进行条件判断。例如,根据年龄显示不同内容,可以用<code>if</code>语句实现如下:</p><pre><code>var age = 20;if (age >= 18) { // 如果age >= 18为true,则执行if语句块 alert('adult');} else { // 否则执行else语句块 alert('teenager');}</code></pre><h2 id="循环"><a href="#循环" class="headerlink" title="循环"></a>循环</h2><p><code>for</code>循环最常用的地方是利用索引来遍历数组:</p><pre><code>var arr = ['Apple', 'Google', 'Microsoft'];var i, x;for (i=0; i<arr.length; i++) { x = arr[i]; console.log(x);}</code></pre><p><code>for</code>循环的3个条件都是可以省略的,如果没有退出循环的判断条件,就必须使用<code>break</code>语句退出循环,否则就是死循环:</p><pre><code>var x = 0;for (;;) { // 将无限循环下去 if (x > 100) { break; // 通过if判断来退出循环 } x ++;}</code></pre><p><code>while</code>循环只有一个判断条件,条件满足,就不断循环,条件不满足时则退出循环。比如我们要计算100以内所有奇数之和,可以用while循环实现:</p><pre><code>var x = 0;var n = 99;while (n > 0) { x = x + n; n = n - 2;}x; // 2500</code></pre><h2 id="Map和Sert"><a href="#Map和Sert" class="headerlink" title="Map和Sert"></a>Map和Sert</h2><p>JavaScript的默认对象表示方式<code>{}</code>可以视为其他语言中的<code>Map</code>或<code>Dictionary</code>的数据结构,即一组键值对。</p><p>但是JavaScript的对象有个小问题,就是键必须是字符串。但实际上Number或者其他数据类型作为键也是非常合理的。</p><p>初始化<code>Map</code>需要一个二维数组,或者直接初始化一个空<code>Map</code>。<code>Map</code>具有以下方法:</p><pre><code>var m = new Map(); // 空Mapm.set('Adam', 67); // 添加新的key-valuem.set('Bob', 59);m.has('Adam'); // 是否存在key 'Adam': truem.get('Adam'); // 67m.delete('Adam'); // 删除key 'Adam'm.get('Adam'); // undefined</code></pre><p>由于一个key只能对应一个value,所以,多次对一个key放入value,后面的值会把前面的值冲掉:</p><pre><code>var m = new Map();m.set('Adam', 67);m.set('Adam', 88);m.get('Adam'); // 88</code></pre><h3 id="Set"><a href="#Set" class="headerlink" title="Set"></a>Set</h3><pre><code>var s1 = new Set(); // 空Setvar s2 = new Set([1, 2, 3]); // 含1, 2, 3</code></pre><p>重复元素在<code>Set</code>中自动被过滤:</p><pre><code>var s = new Set([1, 2, 3, 3, '3']);s; // Set {1, 2, 3, "3"}</code></pre><p>注意数字<code>3</code>和字符串<code>'3'</code>是不同的元素。</p><p>通过<code>add(key)</code>方法可以添加元素到<code>Set</code>中,可以重复添加,但不会有效果:</p><pre><code>s.add(4);s; // Set {1, 2, 3, 4}s.add(4);s; // 仍然是 Set {1, 2, 3, 4}</code></pre><p>通过<code>delete(key)</code>方法可以删除元素:</p><pre><code>var s = new Set([1, 2, 3]);s; // Set {1, 2, 3}s.delete(3);s; // Set {1, 2}</code></pre><h2 id="iterable"><a href="#iterable" class="headerlink" title="iterable"></a>iterable</h2><p>用<code>for ... of</code>循环遍历集合,用法如下:</p><pre><code>var a = ['A', 'B', 'C'];var s = new Set(['A', 'B', 'C']);var m = new Map([[1, 'x'], [2, 'y'], [3, 'z']]);for (var x of a) { // 遍历Array console.log(x);}for (var x of s) { // 遍历Set console.log(x);}for (var x of m) { // 遍历Map console.log(x[0] + '=' + x[1]);}</code></pre><p>然而,更好的方式是直接使用<code>iterable</code>内置的<code>forEach</code>方法,它接收一个函数,每次迭代就自动回调该函数。以<code>Array</code>为例:</p><pre><code>'use strict'; var a = ['A', 'B', 'C']; </code></pre><pre><code>a.forEach(function (element, index, array) { // element: 指向当前元素的值 // index: 指向当前索引 // array: 指向Array对象本身 console.log(element + ', index = ' + index);});</code></pre><p><code>Set</code>与<code>Array</code>类似,但<code>Set</code>没有索引,因此回调函数的前两个参数都是元素本身:</p><pre><code>var s = new Set(['A', 'B', 'C']);s.forEach(function (element, sameElement, set) { console.log(element);});</code></pre><p><code>Map</code>的回调函数参数依次为<code>value</code>、<code>key</code>和<code>map</code>本身:</p><pre><code>var m = new Map([[1, 'x'], [2, 'y'], [3, 'z']]);m.forEach(function (value, key, map) { console.log(value);});</code></pre><p>如果对某些参数不感兴趣,由于JavaScript的函数调用不要求参数必须一致,因此可以忽略它们。例如,只需要获得<code>Array</code>的<code>element</code>:</p><pre><code>var a = ['A', 'B', 'C'];a.forEach(function (element) { console.log(element);});</code></pre><h2 id="函数"><a href="#函数" class="headerlink" title="函数"></a>函数</h2><h3 id="定义函数"><a href="#定义函数" class="headerlink" title="定义函数"></a>定义函数</h3><p>在JavaScript中,定义函数的方式如下:</p><pre><code>function abs(x) { if (x >= 0) { return x; } else { return -x; }}</code></pre><p>上述<code>abs()</code>函数的定义如下:</p><ul><li><code>function</code>指出这是一个函数定义;</li><li><code>abs</code>是函数的名称;</li><li><code>(x)</code>括号内列出函数的参数,多个参数以<code>,</code>分隔;</li><li><code>{ ... }</code>之间的代码是函数体,可以包含若干语句,甚至可以没有任何语句。</li></ul><p>请注意,函数体内部的语句在执行时,一旦执行到<code>return</code>时,函数就执行完毕,并将结果返回。因此,函数内部通过条件判断和循环可以实现非常复杂的逻辑。</p><p>如果没有<code>return</code>语句,函数执行完毕后也会返回结果,只是结果为<code>undefined</code>。</p><p>由于JavaScript的函数也是一个对象,上述定义的<code>abs()</code>函数实际上是一个函数对象,而函数名<code>abs</code>可以视为指向该函数的变量。</p><p>因此,第二种定义函数的方式如下:</p><pre><code>var abs = function (x) { if (x >= 0) { return x; } else { return -x; }};</code></pre><p>在这种方式下,<code>function (x) { ... }</code>是一个匿名函数,它没有函数名。但是,这个匿名函数赋值给了变量<code>abs</code>,所以,通过变量<code>abs</code>就可以调用该函数。</p><p>上述两种定义<em>完全等价</em>,注意第二种方式按照完整语法需要在函数体末尾加一个<code>;</code>,表示赋值语句结束。</p><h3 id="arguments"><a href="#arguments" class="headerlink" title="arguments"></a>arguments</h3><p>JavaScript还有一个免费赠送的关键字<code>arguments</code>,它只在函数内部起作用,并且永远指向当前函数的调用者传入的所有参数。<code>arguments</code>类似<code>Array</code>但它不是一个<code>Array</code>:</p><pre><code>function foo(x) { console.log('x = ' + x); // 10 for (var i=0; i<arguments.length; i++) { console.log('arg ' + i + ' = ' + arguments[i]); // 10, 20, 30 }}foo(10, 20, 30);</code></pre><p>利用<code>arguments</code>,你可以获得调用者传入的所有参数。也就是说,即使函数不定义任何参数,还是可以拿到参数的值:</p><pre><code>function abs() { if (arguments.length === 0) { return 0; } var x = arguments[0]; return x >= 0 ? x : -x;}abs(); // 0abs(10); // 10abs(-9); // 9</code></pre><p>实际上<code>arguments</code>最常用于判断传入参数的个数。你可能会看到这样的写法:</p><pre><code>// foo(a[, b], c)// 接收2~3个参数,b是可选参数,如果只传2个参数,b默认为null:function foo(a, b, c) { if (arguments.length === 2) { // 实际拿到的参数是a和b,c为undefined c = b; // 把b赋给c b = null; // b变为默认值 } // ...}</code></pre><p>要把中间的参数<code>b</code>变为“可选”参数,就只能通过<code>arguments</code>判断,然后重新调整参数并赋值。</p><p>ES6标准引入了rest参数,上面的函数可以改写为:</p><pre><code>function foo(a, b, ...rest) { console.log('a = ' + a); console.log('b = ' + b); console.log(rest);}foo(1, 2, 3, 4, 5);// 结果:// a = 1// b = 2// Array [ 3, 4, 5 ]foo(1);// 结果:// a = 1// b = undefined// Array []</code></pre><p>rest参数只能写在最后,前面用<code>...</code>标识,从运行结果可知,传入的参数先绑定<code>a</code>、<code>b</code>,多余的参数以数组形式交给变量<code>rest</code>,所以,不再需要<code>arguments</code>我们就获取了全部参数。</p><p>如果传入的参数连正常定义的参数都没填满,也不要紧,rest参数会接收一个空数组(注意不是<code>undefined</code>)。</p><h3 id="变量作用与解构赋值"><a href="#变量作用与解构赋值" class="headerlink" title="变量作用与解构赋值"></a>变量作用与解构赋值</h3><h4 id="变量提升"><a href="#变量提升" class="headerlink" title="变量提升"></a>变量提升</h4><p>JavaScript的函数定义有个特点,它会先扫描整个函数体的语句,把所有申明的变量“提升”到函数顶部:</p><pre><code>'use strict';function foo() { var x = 'Hello, ' + y; console.log(x); var y = 'Bob';}foo();</code></pre><p>虽然是strict模式,但语句<code>var x = 'Hello, ' + y;</code>并不报错,原因是变量<code>y</code>在稍后申明了。但是<code>console.log</code>显示<code>Hello, undefined</code>,说明变量<code>y</code>的值为<code>undefined</code>。这正是因为JavaScript引擎自动提升了变量<code>y</code>的声明,但不会提升变量<code>y</code>的赋值。</p><p>对于上述<code>foo()</code>函数,JavaScript引擎看到的代码相当于:</p><pre><code>function foo() { var y; // 提升变量y的申明,此时y为undefined var x = 'Hello, ' + y; console.log(x); y = 'Bob';}</code></pre><p>由于JavaScript的这一怪异的“特性”,我们在函数内部定义变量时,请严格遵守“在函数内部首先申明所有变量”这一规则。最常见的做法是用一个<code>var</code>申明函数内部用到的所有变量:</p><pre><code>function foo() { var x = 1, // x初始化为1 y = x + 1, // y初始化为2 z, i; // z和i为undefined // 其他语句: for (i=0; i<100; i++) { ... }}</code></pre><h4 id="全局作用域"><a href="#全局作用域" class="headerlink" title="全局作用域"></a>全局作用域</h4><p>不在任何函数内定义的变量就具有全局作用域。实际上,JavaScript默认有一个全局对象<code>window</code>,全局作用域的变量实际上被绑定到<code>window</code>的一个属性:</p><p>JavaScript实际上只有一个全局作用域。任何变量(函数也视为变量),如果没有在当前函数作用域中找到,就会继续往上查找,最后如果在全局作用域中也没有找到,则报<code>ReferenceError</code>错误。</p><h4 id="名字空间"><a href="#名字空间" class="headerlink" title="名字空间"></a>名字空间</h4><p>全局变量会绑定到<code>window</code>上,不同的JavaScript文件如果使用了相同的全局变量,或者定义了相同名字的顶层函数,都会造成命名冲突,并且很难被发现。</p><p>减少冲突的一个方法是把自己的所有变量和函数全部绑定到一个全局变量中。例如:</p><pre><code>// 唯一的全局变量MYAPP:var MYAPP = {};// 其他变量:MYAPP.name = 'myapp';MYAPP.version = 1.0;// 其他函数:MYAPP.foo = function () { return 'foo';};</code></pre><p>把自己的代码全部放入唯一的名字空间<code>MYAPP</code>中,会大大减少全局变量冲突的可能。</p><p>许多著名的JavaScript库都是这么干的:jQuery,YUI,underscore等等。</p><h4 id="局部作用域"><a href="#局部作用域" class="headerlink" title="局部作用域"></a>局部作用域</h4><p>由于JavaScript的变量作用域实际上是函数内部,我们在<code>for</code>循环等语句块中是无法定义具有局部作用域的变量的:</p><pre><code>'use strict';function foo() { for (var i=0; i<100; i++) { // } i += 100; // 仍然可以引用变量i}</code></pre><p>为了解决块级作用域,ES6引入了新的关键字<code>let</code>,用<code>let</code>替代<code>var</code>可以申明一个块级作用域的变量:</p><pre><code>'use strict';function foo() { var sum = 0; for (let i=0; i<100; i++) { sum += i; } // SyntaxError: i += 1;}</code></pre><h4 id="常量"><a href="#常量" class="headerlink" title="常量"></a>常量</h4><p>由于<code>var</code>和<code>let</code>申明的是变量,如果要申明一个常量,在ES6之前是不行的,我们通常用全部大写的变量来表示“这是一个常量,不要修改它的值”:</p><pre><code>var PI = 3.14;</code></pre><p>ES6标准引入了新的关键字<code>const</code>来定义常量,<code>const</code>与<code>let</code>都具有块级作用域:</p><pre><code>'use strict';const PI = 3.14;PI = 3; // 某些浏览器不报错,但是无效果!PI; // 3.14</code></pre><h4 id="解构赋值"><a href="#解构赋值" class="headerlink" title="解构赋值"></a>解构赋值</h4><p>从ES6开始,JavaScript引入了解构赋值,可以同时对一组变量进行赋值。</p><p>什么是解构赋值?我们先看看传统的做法,如何把一个数组的元素分别赋值给几个变量:</p><pre><code>var array = ['hello', 'JavaScript', 'ES6'];var x = array[0];var y = array[1];var z = array[2];</code></pre><p>现在,在ES6中,可以使用解构赋值,直接对多个变量同时赋值:</p><pre><code>'use strict'; // 如果浏览器支持解构赋值就不会报错: var [x, y, z] = ['hello', 'JavaScript', 'ES6'];</code></pre><p>如果数组本身还有嵌套,也可以通过下面的形式进行解构赋值,注意嵌套层次和位置要保持一致:</p><pre><code>let [x, [y, z]] = ['hello', ['JavaScript', 'ES6']];x; // 'hello'y; // 'JavaScript'z; // 'ES6'</code></pre><p>解构赋值还可以忽略某些元素:</p><pre><code>let [, , z] = ['hello', 'JavaScript', 'ES6']; // 忽略前两个元素,只对z赋值第三个元素z; // 'ES6'</code></pre><p>如果需要从一个对象中取出若干属性,也可以使用解构赋值,便于快速获取对象的指定属性:</p><pre><code>'use strict'; var person = { name: '小明',age: 20, gender: 'male', passport: 'G-12345678', school: 'No.4 middle school'}; var {name, age, passport} = person;</code></pre><pre><code>var person = { name: '小明', age: 20, gender: 'male', passport: 'G-12345678', school: 'No.4 middle school', address: { city: 'Beijing', street: 'No.1 Road', zipcode: '100001' }};var {name, address: {city, zip}} = person;name; // '小明'city; // 'Beijing'zip; // undefined, 因为属性名是zipcode而不是zip// 注意: address不是变量,而是为了让city和zip获得嵌套的address对象的属性:address; // Uncaught ReferenceError: address is not defined</code></pre><p>使用解构赋值对对象属性进行赋值时,如果对应的属性不存在,变量将被赋值为<code>undefined</code>,这和引用一个不存在的属性获得<code>undefined</code>是一致的。如果要使用的变量名和属性名不一致,可以用下面的语法获取:</p><pre><code>var person = { name: '小明', age: 20, gender: 'male', passport: 'G-12345678', school: 'No.4 middle school'};// 把passport属性赋值给变量id:let {name, passport:id} = person;name; // '小明'id; // 'G-12345678'// 注意: passport不是变量,而是为了让变量id获得passport属性:passport; // Uncaught ReferenceError: passport is not defined</code></pre><p>解构赋值还可以使用默认值,这样就避免了不存在的属性返回<code>undefined</code>的问题:</p><pre><code>var person = { name: '小明', age: 20, gender: 'male', passport: 'G-12345678'};// 如果person对象没有single属性,默认赋值为true:var {name, single=true} = person;name; // '小明'single; // true</code></pre><h4 id="使用场景"><a href="#使用场景" class="headerlink" title="使用场景"></a>使用场景</h4><p>解构赋值在很多时候可以大大简化代码。例如,交换两个变量<code>x</code>和<code>y</code>的值,可以这么写,不再需要临时变量:</p><pre><code>var x=1, y=2;[x, y] = [y, x]</code></pre><h3 id="方法"><a href="#方法" class="headerlink" title="方法"></a>方法</h3><p>在一个对象中绑定函数,称为这个对象的方法。</p><pre><code>'use strict';var xiaoming = { name: '小明', birth: 1990, age: function () { var that = this; // 在方法内部一开始就捕获this function getAgeFromBirth() { var y = new Date().getFullYear(); return y - that.birth; // 用that而不是this } return getAgeFromBirth(); }};xiaoming.age(); // 25用var that = this;,你就可以放心地在方法内部定义其他函数,而不是把所有语句都堆到一个方法中。</code></pre><h4 id="apply"><a href="#apply" class="headerlink" title="apply"></a>apply</h4><p>虽然在一个独立的函数调用中,根据是否是strict模式,<code>this</code>指向<code>undefined</code>或<code>window</code>,不过,我们还是可以控制<code>this</code>的指向的!</p><p>要指定函数的<code>this</code>指向哪个对象,可以用函数本身的<code>apply</code>方法,它接收两个参数,第一个参数就是需要绑定的<code>this</code>变量,第二个参数是<code>Array</code>,表示函数本身的参数。</p><p>用<code>apply</code>修复<code>getAge()</code>调用:</p><pre><code>function getAge() { var y = new Date().getFullYear(); return y - this.birth;}var xiaoming = { name: '小明', birth: 1990, age: getAge};xiaoming.age(); // 25getAge.apply(xiaoming, []); // 25, this指向xiaoming, 参数为空</code></pre><p>另一个与<code>apply()</code>类似的方法是<code>call()</code>,唯一区别是:</p><ul><li><code>apply()</code>把参数打包成<code>Array</code>再传入;</li><li><code>call()</code>把参数按顺序传入。</li></ul><p>比如调用<code>Math.max(3, 5, 4)</code>,分别用<code>apply()</code>和<code>call()</code>实现如下:</p><pre><code>Math.max.apply(null, [3, 5, 4]); // 5Math.max.call(null, 3, 5, 4); // 5</code></pre><p>对普通函数调用,我们通常把<code>this</code>绑定为<code>null</code>。</p><h3 id="装饰器"><a href="#装饰器" class="headerlink" title="装饰器"></a>装饰器</h3><p>利用<code>apply()</code>,我们还可以动态改变函数的行为。</p><p>JavaScript的所有对象都是动态的,即使内置的函数,我们也可以重新指向新的函数。</p><p>现在假定我们想统计一下代码一共调用了多少次<code>parseInt()</code>,可以把所有的调用都找出来,然后手动加上<code>count += 1</code>,不过这样做太傻了。最佳方案是用我们自己的函数替换掉默认的<code>parseInt()</code>:</p><pre><code>'use strict'; var count = 0; var oldParseInt = parseInt; // 保存原函数 window.parseInt = function () { count += 1; return oldParseInt.apply(null, arguments); // 调用原函数 };</code></pre><h3 id="高阶函数"><a href="#高阶函数" class="headerlink" title="高阶函数"></a>高阶函数</h3><p>高阶函数英文叫Higher-order function。那么什么是高阶函数?</p><p>JavaScript的函数其实都指向某个变量。既然变量可以指向函数,函数的参数能接收变量,那么一个函数就可以接收另一个函数作为参数,这种函数就称之为高阶函数。</p><p>一个最简单的高阶函数:</p><pre><code>function add(x, y, f) { return f(x) + f(y);}</code></pre><p>当我们调用<code>add(-5, 6, Math.abs)</code>时,参数<code>x</code>,<code>y</code>和<code>f</code>分别接收<code>-5</code>,<code>6</code>和函数<code>Math.abs</code>,根据函数定义,我们可以推导计算过程为:</p><pre><code>x = -5;y = 6;f = Math.abs;f(x) + f(y) ==> Math.abs(-5) + Math.abs(6) ==> 11;return 11;</code></pre><h4 id="map"><a href="#map" class="headerlink" title="map"></a>map</h4><p>由于<code>map()</code>方法定义在JavaScript的<code>Array</code>中,我们调用<code>Array</code>的<code>map()</code>方法,传入我们自己的函数,就得到了一个新的<code>Array</code>作为结果:</p><pre><code>'use strict'; function pow(x) { return x * x; }</code></pre><pre><code>var arr = [1, 2, 3, 4, 5, 6, 7, 8, 9];var results = arr.map(pow); // [1, 4, 9, 16, 25, 36, 49, 64, 81]console.log(results);</code></pre><p>所以,<code>map()</code>作为高阶函数,事实上它把运算规则抽象了,因此,我们不但可以计算简单的f(x)=x2,还可以计算任意复杂的函数,比如,把<code>Array</code>的所有数字转为字符串:</p><pre><code>var arr = [1, 2, 3, 4, 5, 6, 7, 8, 9];arr.map(String); // ['1', '2', '3', '4', '5', '6', '7', '8', '9']</code></pre><p>只需要一行代码。</p><h4 id="reduce"><a href="#reduce" class="headerlink" title="reduce"></a>reduce</h4><p>要把<code>[1, 3, 5, 7, 9]</code>变换成整数13579,<code>reduce()</code>也能派上用场:</p><pre><code>var arr = [1, 3, 5, 7, 9];arr.reduce(function (x, y) { return x * 10 + y;}); // 13579</code></pre><h4 id="fillter"><a href="#fillter" class="headerlink" title="fillter"></a>fillter</h4><p>filter也是一个常用的操作,它用于把<code>Array</code>的某些元素过滤掉,然后返回剩下的元素。</p><p>和<code>map()</code>类似,<code>Array</code>的<code>filter()</code>也接收一个函数。和<code>map()</code>不同的是,<code>filter()</code>把传入的函数依次作用于每个元素,然后根据返回值是<code>true</code>还是<code>false</code>决定保留还是丢弃该元素。</p><p>例如,在一个<code>Array</code>中,删掉偶数,只保留奇数,可以这么写:</p><pre><code>var arr = [1, 2, 4, 5, 6, 9, 10, 15];var r = arr.filter(function (x) { return x % 2 !== 0;});r; // [1, 5, 9, 15]</code></pre><p>把一个<code>Array</code>中的空字符串删掉,可以这么写:</p><pre><code>var arr = ['A', '', 'B', null, undefined, 'C', ' '];var r = arr.filter(function (s) { return s && s.trim(); // 注意:IE9以下的版本没有trim()方法});r; // ['A', 'B', 'C']</code></pre><p>可见用<code>filter()</code>这个高阶函数,关键在于正确实现一个“筛选”函数。</p><h5 id="回调函数"><a href="#回调函数" class="headerlink" title="回调函数"></a>回调函数</h5><p><code>filter()</code>接收的回调函数,其实可以有多个参数。通常我们仅使用第一个参数,表示<code>Array</code>的某个元素。回调函数还可以接收另外两个参数,表示元素的位置和数组本身:</p><pre><code>var arr = ['A', 'B', 'C'];var r = arr.filter(function (element, index, self) { console.log(element); // 依次打印'A', 'B', 'C' console.log(index); // 依次打印0, 1, 2 console.log(self); // self就是变量arr return true;});</code></pre><p>利用<code>filter</code>,可以巧妙地去除<code>Array</code>的重复元素:</p><pre><code>'use strict';var r, arr = ['apple', 'strawberry', 'banana', 'pear', 'apple', 'orange', 'orange', 'strawberry']; </code></pre><pre><code>r = arr.filter(function (element, index, self) { return self.indexOf(element) === index;});</code></pre><p>去除重复元素依靠的是<code>indexOf</code>总是返回第一个元素的位置,后续的重复元素位置与<code>indexOf</code>返回的位置不相等,因此被<code>filter</code>滤掉了。</p><h4 id="sort-1"><a href="#sort-1" class="headerlink" title="sort"></a>sort</h4><pre><code>'use strict';var arr = [10, 20, 1, 2];arr.sort(function (x, y) { if (x < y) { return -1; } if (x > y) { return 1; } return 0;});console.log(arr); // [1, 2, 10, 20]</code></pre><pre><code>如果要倒序排序,我们可以把大的数放前面:var arr = [10, 20, 1, 2];arr.sort(function (x, y) { if (x < y) { return 1; } if (x > y) { return -1; } return 0;}); // [20, 10, 2, 1]</code></pre><pre><code>var arr = ['Google', 'apple', 'Microsoft'];arr.sort(function (s1, s2) { x1 = s1.toUpperCase(); x2 = s2.toUpperCase(); if (x1 < x2) { return -1; } if (x1 > x2) { return 1; } return 0;}); // ['apple', 'Google', 'Microsoft']</code></pre><p>忽略大小写来比较两个字符串,实际上就是先把字符串都变成大写(或者都变成小写),再比较。</p><h3 id="Array"><a href="#Array" class="headerlink" title="Array"></a>Array</h3><h4 id="every"><a href="#every" class="headerlink" title="every"></a>every</h4><p><code>every()</code>方法可以判断数组的所有元素是否满足测试条件。</p><p>例如,给定一个包含若干字符串的数组,判断所有字符串是否满足指定的测试条件:</p><pre><code>var arr = ['Apple', 'pear', 'orange'];console.log(arr.every(function (s) { return s.length > 0;})); // true, 因为每个元素都满足s.length>0console.log(arr.every(function (s) { return s.toLowerCase() === s;})); // false, 因为不是每个元素都全部是小写</code></pre><h4 id="find"><a href="#find" class="headerlink" title="find"></a>find</h4><p><code>find()</code>方法用于查找符合条件的第一个元素,如果找到了,返回这个元素,否则,返回<code>undefined</code>:</p><pre><code>var arr = ['Apple', 'pear', 'orange'];console.log(arr.find(function (s) { return s.toLowerCase() === s;})); // 'pear', 因为pear全部是小写console.log(arr.find(function (s) { return s.toUpperCase() === s;})); // undefined, 因为没有全部是大写的元素</code></pre><h4 id="findIndex"><a href="#findIndex" class="headerlink" title="findIndex"></a>findIndex</h4><p><code>findIndex()</code>和<code>find()</code>类似,也是查找符合条件的第一个元素,不同之处在于<code>findIndex()</code>会返回这个元素的索引,如果没有找到,返回<code>-1</code>:</p><pre><code>var arr = ['Apple', 'pear', 'orange'];console.log(arr.findIndex(function (s) { return s.toLowerCase() === s;})); // 1, 因为'pear'的索引是1console.log(arr.findIndex(function (s) { return s.toUpperCase() === s;})); // -1</code></pre><h4 id="orEach"><a href="#orEach" class="headerlink" title="orEach"></a>orEach</h4><p><code>forEach()</code>和<code>map()</code>类似,它也把每个元素依次作用于传入的函数,但不会返回新的数组。<code>forEach()</code>常用于遍历数组,因此,传入的函数不需要返回值:</p><pre><code>var arr = ['Apple', 'pear', 'orange'];arr.forEach(console.log); // 依次打印每个元素</code></pre><h3 id="闭包"><a href="#闭包" class="headerlink" title="闭包"></a>闭包</h3><pre><code>function lazy_sum(arr) { var sum = function () { return arr.reduce(function (x, y) { return x + y; }); } return sum;}</code></pre><p>当我们调用<code>lazy_sum()</code>时,返回的并不是求和结果,而是求和函数:</p><pre><code>var f = lazy_sum([1, 2, 3, 4, 5]); // function sum()</code></pre><p>调用函数<code>f</code>时,才真正计算求和的结果:</p><pre><code>f(); // 15</code></pre><p>在这个例子中,我们在函数<code>lazy_sum</code>中又定义了函数<code>sum</code>,并且,内部函数<code>sum</code>可以引用外部函数<code>lazy_sum</code>的参数和局部变量,当<code>lazy_sum</code>返回函数<code>sum</code>时,相关参数和变量都保存在返回的函数中,这种称为“闭包(Closure)”的程序结构拥有极大的威力。</p><p>请再注意一点,当我们调用<code>lazy_sum()</code>时,每次调用都会返回一个新的函数,即使传入相同的参数:</p><pre><code>var f1 = lazy_sum([1, 2, 3, 4, 5]);var f2 = lazy_sum([1, 2, 3, 4, 5]);f1 === f2; // false</code></pre><p><code>f1()</code>和<code>f2()</code>的调用结果互不影响。</p><p>另一个需要注意的问题是,返回的函数并没有立刻执行,而是直到调用了<code>f()</code>才执行。我们来看一个例子:</p><pre><code>function count() { var arr = []; for (var i=1; i<=3; i++) { arr.push(function () { return i * i; }); } return arr;}var results = count();var f1 = results[0];var f2 = results[1];var f3 = results[2];</code></pre><p>在上面的例子中,每次循环,都创建了一个新的函数,然后,把创建的3个函数都添加到一个<code>Array</code>中返回了。</p><p>你可能认为调用<code>f1()</code>,<code>f2()</code>和<code>f3()</code>结果应该是<code>1</code>,<code>4</code>,<code>9</code>,但实际结果是:</p><pre><code>f1(); // 16f2(); // 16f3(); // 16</code></pre><p>全部都是<code>16</code>!原因就在于返回的函数引用了变量<code>i</code>,但它并非立刻执行。等到3个函数都返回时,它们所引用的变量<code>i</code>已经变成了<code>4</code>,因此最终结果为<code>16</code>。</p><p>如果一定要引用循环变量怎么办?方法是再创建一个函数,用该函数的参数绑定循环变量当前的值,无论该循环变量后续如何更改,已绑定到函数参数的值不变:</p><pre><code>function count() { var arr = []; for (var i=1; i<=3; i++) { arr.push((function (n) { return function () { return n * n; } })(i)); } return arr;}var results = count();var f1 = results[0];var f2 = results[1];var f3 = results[2];f1(); // 1f2(); // 4f3(); // 9</code></pre><p>注意这里用了一个“创建一个匿名函数并立刻执行”的语法:</p><pre><code>(function (x) { return x * x;})(3); // 9</code></pre><p>在没有<code>class</code>机制,只有函数的语言里,借助闭包,同样可以封装一个私有变量。我们用JavaScript创建一个计数器:</p><pre><code>'use strict';function create_counter(initial) { var x = initial || 0; return { inc: function () { x += 1; return x; } }}</code></pre><p>它用起来像这样:</p><pre><code>var c1 = create_counter();c1.inc(); // 1c1.inc(); // 2c1.inc(); // 3var c2 = create_counter(10);c2.inc(); // 11c2.inc(); // 12c2.inc(); // 13</code></pre><p>在返回的对象中,实现了一个闭包,该闭包携带了局部变量<code>x</code>,并且,从外部代码根本无法访问到变量<code>x</code>。换句话说,闭包就是携带状态的函数,并且它的状态可以完全对外隐藏起来。</p><p>闭包还可以把多参数的函数变成单参数的函数。例如,要计算xy可以用<code>Math.pow(x, y)</code>函数,不过考虑到经常计算x2或x3,我们可以利用闭包创建新的函数<code>pow2</code>和<code>pow3</code>:</p><pre><code>'use strict';function make_pow(n) { return function (x) { return Math.pow(x, n); } } // 创建两个新函数:var pow2 = make_pow(2);var pow3 = make_pow(3);console.log(pow2(5)); // 25console.log(pow3(7)); // 343</code></pre><h3 id="箭头函数"><a href="#箭头函数" class="headerlink" title="箭头函数"></a>箭头函数</h3><p>为什么叫Arrow Function?因为它的定义用的就是一个箭头:</p><pre><code>x => x * x</code></pre><p>上面的箭头函数相当于:</p><pre><code>function (x) { return x * x;}</code></pre><p>箭头函数相当于匿名函数,并且简化了函数定义。箭头函数有两种格式,一种像上面的,只包含一个表达式,连<code>{ ... }</code>和<code>return</code>都省略掉了。还有一种可以包含多条语句,这时候就不能省略<code>{ ... }</code>和<code>return</code>:</p><pre><code>x => { if (x > 0) { return x * x; } else { return - x * x; }}</code></pre><p>如果参数不是一个,就需要用括号<code>()</code>括起来:</p><pre><code>// 两个参数:(x, y) => x * x + y * y// 无参数:() => 3.14// 可变参数:(x, y, ...rest) => { var i, sum = x + y; for (i=0; i<rest.length; i++) { sum += rest[i]; } return sum;}</code></pre><p>如果要返回一个对象,就要注意,如果是单表达式,这么写的话会报错:</p><pre><code>// SyntaxError:x => { foo: x }</code></pre><p>因为和函数体的<code>{ ... }</code>有语法冲突,所以要改为:</p><pre><code>// ok:x => ({ foo: x })</code></pre><h4 id="this"><a href="#this" class="headerlink" title="this"></a>this</h4><p>箭头函数看上去是匿名函数的一种简写,但实际上,箭头函数和匿名函数有个明显的区别:箭头函数内部的<code>this</code>是词法作用域,由上下文确定。</p><p>回顾前面的例子,由于JavaScript函数对<code>this</code>绑定的错误处理,下面的例子无法得到预期结果:</p><pre><code>var obj = { birth: 1990, getAge: function () { var b = this.birth; // 1990 var fn = function () { return new Date().getFullYear() - this.birth; // this指向window或undefined }; return fn(); }};</code></pre><p>现在,箭头函数完全修复了<code>this</code>的指向,<code>this</code>总是指向词法作用域,也就是外层调用者<code>obj</code>:</p><pre><code>var obj = { birth: 1990, getAge: function () { var b = this.birth; // 1990 var fn = () => new Date().getFullYear() - this.birth; // this指向obj对象 return fn(); }};obj.getAge(); // 25</code></pre><p>如果使用箭头函数,以前的那种hack写法:</p><pre><code>var that = this;</code></pre><p>就不再需要了。</p><p>由于<code>this</code>在箭头函数中已经按照词法作用域绑定了,所以,用<code>call()</code>或者<code>apply()</code>调用箭头函数时,无法对<code>this</code>进行绑定,即传入的第一个参数被忽略:</p><pre><code>var obj = { birth: 1990, getAge: function (year) { var b = this.birth; // 1990 var fn = (y) => y - this.birth; // this.birth仍是1990 return fn.call({birth:2000}, year); }};obj.getAge(2015); // 25</code></pre><h3 id="generator"><a href="#generator" class="headerlink" title="generator"></a>generator</h3><p>要编写一个产生斐波那契数列的函数,可以这么写:</p><pre><code>function fib(max) { var t, a = 0, b = 1, arr = [0, 1]; while (arr.length < max) { [a, b] = [b, a + b]; arr.push(b); } return arr;}// 测试:fib(5); // [0, 1, 1, 2, 3]fib(10); // [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]</code></pre><p>函数只能返回一次,所以必须返回一个<code>Array</code>。但是,如果换成generator,就可以一次返回一个数,不断返回多次。用generator改写如下:</p><pre><code>function* fib(max) { var t, a = 0, b = 1, n = 0; while (n < max) { yield a; [a, b] = [b, a + b]; n ++; } return;}</code></pre><p>直接调用试试:</p><pre><code>fib(5); // fib {[[GeneratorStatus]]: "suspended", [[GeneratorReceiver]]: Window}</code></pre><p>直接调用一个generator和调用函数不一样,<code>fib(5)</code>仅仅是创建了一个generator对象,还没有去执行它。</p><p>调用generator对象有两个方法,一是不断地调用generator对象的<code>next()</code>方法:</p><pre><code>var f = fib(5);f.next(); // {value: 0, done: false}f.next(); // {value: 1, done: false}f.next(); // {value: 1, done: false}f.next(); // {value: 2, done: false}f.next(); // {value: 3, done: false}f.next(); // {value: undefined, done: true}</code></pre><p><code>next()</code>方法会执行generator的代码,然后,每次遇到<code>yield x;</code>就返回一个对象<code>{value: x, done: true/false}</code>,然后“暂停”。返回的<code>value</code>就是<code>yield</code>的返回值,<code>done</code>表示这个generator是否已经执行结束了。如果<code>done</code>为<code>true</code>,则<code>value</code>就是<code>return</code>的返回值。</p><p>当执行到<code>done</code>为<code>true</code>时,这个generator对象就已经全部执行完毕,不要再继续调用<code>next()</code>了。</p><p>generator还有另一个巨大的好处,就是把异步回调代码变成“同步”代码。这个好处要等到后面学了AJAX以后才能体会到。</p><p>没有generator之前的黑暗时代,用AJAX时需要这么写代码:</p><pre><code>ajax('http://url-1', data1, function (err, result) { if (err) { return handle(err); } ajax('http://url-2', data2, function (err, result) { if (err) { return handle(err); } ajax('http://url-3', data3, function (err, result) { if (err) { return handle(err); } return success(result); }); });});</code></pre><p>回调越多,代码越难看。</p><p>有了generator的美好时代,用AJAX时可以这么写:</p><pre><code>try { r1 = yield ajax('http://url-1', data1); r2 = yield ajax('http://url-2', data2); r3 = yield ajax('http://url-3', data3); success(r3);}catch (err) { handle(err);}</code></pre><p>看上去是同步的代码,实际执行是异步的。</p><h2 id="标准对象"><a href="#标准对象" class="headerlink" title="标准对象"></a>标准对象</h2><h4 id="包装对象"><a href="#包装对象" class="headerlink" title="包装对象"></a>包装对象</h4><p>除了这些类型外,JavaScript还提供了包装对象,熟悉Java的小伙伴肯定很清楚<code>int</code>和<code>Integer</code>这种暧昧关系。</p><p><code>number</code>、<code>boolean</code>和<code>string</code>都有包装对象。没错,在JavaScript中,字符串也区分<code>string</code>类型和它的包装类型。包装对象用<code>new</code>创建:</p><pre><code>var n = new Number(123); // 123,生成了新的包装类型var b = new Boolean(true); // true,生成了新的包装类型var s = new String('str'); // 'str',生成了新的包装类型</code></pre><p>虽然包装对象看上去和原来的值一模一样,显示出来也是一模一样,但他们的类型已经变为<code>object</code>了!所以,包装对象和原始值用<code>===</code>比较会返回<code>false</code>:</p><pre><code>typeof new Number(123); // 'object'new Number(123) === 123; // falsetypeof new Boolean(true); // 'object'new Boolean(true) === true; // falsetypeof new String('str'); // 'object'new String('str') === 'str'; // false</code></pre><p>所以<em>闲的蛋疼也不要使用包装对象</em>!尤其是针对<code>string</code>类型!!!</p><p>如果我们在使用<code>Number</code>、<code>Boolean</code>和<code>String</code>时,没有写<code>new</code>会发生什么情况?</p><p>此时,<code>Number()</code>、<code>Boolean</code>和<code>String()</code>被当做普通函数,把任何类型的数据转换为<code>number</code>、<code>boolean</code>和<code>string</code>类型(注意不是其包装类型):</p><pre><code>var n = Number('123'); // 123,相当于parseInt()或parseFloat()typeof n; // 'number'var b = Boolean('true'); // truetypeof b; // 'boolean'var b2 = Boolean('false'); // true! 'false'字符串转换结果为true!因为它是非空字符串!var b3 = Boolean(''); // falsevar s = String(123.45); // '123.45'typeof s; // 'string'</code></pre><p>总结一下,有这么几条规则需要遵守:</p><ul><li>不要使用<code>new Number()</code>、<code>new Boolean()</code>、<code>new String()</code>创建包装对象;</li><li>用<code>parseInt()</code>或<code>parseFloat()</code>来转换任意类型到<code>number</code>;</li><li>用<code>String()</code>来转换任意类型到<code>string</code>,或者直接调用某个对象的<code>toString()</code>方法;</li><li>通常不必把任意类型转换为<code>boolean</code>再判断,因为可以直接写<code>if (myVar) {...}</code>;</li><li><code>typeof</code>操作符可以判断出<code>number</code>、<code>boolean</code>、<code>string</code>、<code>function</code>和<code>undefined</code>;</li><li>判断<code>Array</code>要使用<code>Array.isArray(arr)</code>;</li><li>判断<code>null</code>请使用<code>myVar === null</code>;</li><li>判断某个全局变量是否存在用<code>typeof window.myVar === 'undefined'</code>;</li><li>函数内部判断某个变量是否存在用<code>typeof myVar === 'undefined'</code>。</li></ul><p>最后有细心的同学指出,任何对象都有<code>toString()</code>方法吗?<code>null</code>和<code>undefined</code>就没有!确实如此,这两个特殊值要除外,虽然<code>null</code>还伪装成了<code>object</code>类型。</p><h3 id="Date"><a href="#Date" class="headerlink" title="Date"></a>Date</h3><p>在JavaScript中,<code>Date</code>对象用来表示日期和时间。</p><p>要获取系统当前时间,用:</p><pre><code>var now = new Date();now; // Wed Jun 24 2015 19:49:22 GMT+0800 (CST)now.getFullYear(); // 2015, 年份now.getMonth(); // 5, 月份,注意月份范围是0~11,5表示六月now.getDate(); // 24, 表示24号now.getDay(); // 3, 表示星期三now.getHours(); // 19, 24小时制now.getMinutes(); // 49, 分钟now.getSeconds(); // 22, 秒now.getMilliseconds(); // 875, 毫秒数now.getTime(); // 1435146562875, 以number形式表示的时间戳</code></pre><p>注意,当前时间是浏览器从本机操作系统获取的时间,所以不一定准确,因为用户可以把当前时间设定为任何值。</p><p>如果要创建一个指定日期和时间的<code>Date</code>对象,可以用:</p><pre><code>var d = new Date(2015, 5, 19, 20, 15, 30, 123);d; // Fri Jun 19 2015 20:15:30 GMT+0800 (CST)</code></pre><p>你可能观察到了一个<em>非常非常坑爹</em>的地方,就是JavaScript的月份范围用整数表示是0~11,<code>0</code>表示一月,<code>1</code>表示二月……,所以要表示6月,我们传入的是<code>5</code>!这绝对是JavaScript的设计者当时脑抽了一下,但是现在要修复已经不可能了。</p><h3 id="RegExp"><a href="#RegExp" class="headerlink" title="RegExp"></a>RegExp</h3><p>avaScript有两种方式创建一个正则表达式:</p><p>第一种方式是直接通过<code>/正则表达式/</code>写出来,第二种方式是通过<code>new RegExp('正则表达式')</code>创建一个RegExp对象。</p><p>两种写法是一样的:</p><pre><code>var re1 = /ABC\-001/;var re2 = new RegExp('ABC\\-001');re1; // /ABC\-001/re2; // /ABC\-001/</code></pre><p>注意,如果使用第二种写法,因为字符串的转义问题,字符串的两个<code>\\</code>实际上是一个<code>\</code>。</p><p>先看看如何判断正则表达式是否匹配:</p><pre><code>var re = /^\d{3}\-\d{3,8}$/;re.test('010-12345'); // truere.test('010-1234x'); // falsere.test('010 12345'); // false</code></pre><p>RegExp对象的<code>test()</code>方法用于测试给定的字符串是否符合条件。</p><h4 id="切分字符串"><a href="#切分字符串" class="headerlink" title="切分字符串"></a>切分字符串</h4><p>用正则表达式切分字符串比用固定的字符更灵活,请看正常的切分代码:</p><pre><code>'a b c'.split(' '); // ['a', 'b', '', '', 'c']</code></pre><p>嗯,无法识别连续的空格,用正则表达式试试:</p><pre><code>'a b c'.split(/\s+/); // ['a', 'b', 'c']</code></pre><p>无论多少个空格都可以正常分割。加入<code>,</code>试试:</p><pre><code>'a,b, c d'.split(/[\s\,]+/); // ['a', 'b', 'c', 'd']</code></pre><p>再加入<code>;</code>试试:</p><pre><code>'a,b;; c d'.split(/[\s\,\;]+/); // ['a', 'b', 'c', 'd']</code></pre><p>如果用户输入了一组标签,下次记得用正则表达式来把不规范的输入转化成正确的数组。</p><h4 id="分组"><a href="#分组" class="headerlink" title="分组"></a>分组</h4><p>除了简单地判断是否匹配之外,正则表达式还有提取子串的强大功能。用<code>()</code>表示的就是要提取的分组(Group)。比如:</p><p><code>^(\d{3})-(\d{3,8})$</code>分别定义了两个组,可以直接从匹配的字符串中提取出区号和本地号码:</p><pre><code>var re = /^(\d{3})-(\d{3,8})$/;re.exec('010-12345'); // ['010-12345', '010', '12345']re.exec('010 12345'); // null</code></pre><p>如果正则表达式中定义了组,就可以在<code>RegExp</code>对象上用<code>exec()</code>方法提取出子串来。</p><p><code>exec()</code>方法在匹配成功后,会返回一个<code>Array</code>,第一个元素是正则表达式匹配到的整个字符串,后面的字符串表示匹配成功的子串。</p><p><code>exec()</code>方法在匹配失败时返回<code>null</code>。</p><h3 id="JSON"><a href="#JSON" class="headerlink" title="JSON"></a>JSON</h3><h3 id="序列化"><a href="#序列化" class="headerlink" title="序列化"></a>序列化</h3><p>让我们先把小明这个对象序列化成JSON格式的字符串:</p><pre><code>'use strict';var xiaoming = { name: '小明', age: 14, gender: true, height: 1.65, grade: null, 'middle-school': '\"W3C\" Middle School', skills: ['JavaScript', 'Java', 'Python', 'Lisp']};var s = JSON.stringify(xiaoming);console.log(s);</code></pre><p>要输出得好看一些,可以加上参数,按缩进输出:</p><pre><code>JSON.stringify(xiaoming, null, ' ');</code></pre><p>第二个参数用于控制如何筛选对象的键值,如果我们只想输出指定的属性,可以传入<code>Array</code>:</p><pre><code>JSON.stringify(xiaoming, ['name', 'skills'], ' ');</code></pre><p>结果:</p><pre><code>{ "name": "小明", "skills": [ "JavaScript", "Java", "Python", "Lisp" ]}</code></pre><p>还可以传入一个函数,这样对象的每个键值对都会被函数先处理:</p><pre><code>function convert(key, value) { if (typeof value === 'string') { return value.toUpperCase(); } return value;}JSON.stringify(xiaoming, convert, ' ');</code></pre><p>上面的代码把所有属性值都变成大写:</p><pre><code>{ "name": "小明", "age": 14, "gender": true, "height": 1.65, "grade": null, "middle-school": "\"W3C\" MIDDLE SCHOOL", "skills": [ "JAVASCRIPT", "JAVA", "PYTHON", "LISP" ]}</code></pre><p>如果我们还想要精确控制如何序列化小明,可以给<code>xiaoming</code>定义一个<code>toJSON()</code>的方法,直接返回JSON应该序列化的数据:</p><pre><code>var xiaoming = { name: '小明', age: 14, gender: true, height: 1.65, grade: null, 'middle-school': '\"W3C\" Middle School', skills: ['JavaScript', 'Java', 'Python', 'Lisp'], toJSON: function () { return { // 只输出name和age,并且改变了key: 'Name': this.name, 'Age': this.age }; }};JSON.stringify(xiaoming); // '{"Name":"小明","Age":14}'</code></pre><h3 id="反序列化"><a href="#反序列化" class="headerlink" title="反序列化"></a>反序列化</h3><p>拿到一个JSON格式的字符串,我们直接用<code>JSON.parse()</code>把它变成一个JavaScript对象:</p><pre><code>JSON.parse('[1,2,3,true]'); // [1, 2, 3, true]JSON.parse('{"name":"小明","age":14}'); // Object {name: '小明', age: 14}JSON.parse('true'); // trueJSON.parse('123.45'); // 123.45</code></pre><p><code>JSON.parse()</code>还可以接收一个函数,用来转换解析出的属性:</p><pre><code>var obj = JSON.parse('{"name":"小明","age":14}', function (key, value) { if (key === 'name') { return value + '同学'; } return value;});console.log(JSON.stringify(obj)); // {name: '小明同学', age: 14}</code></pre><h2 id="面向对象编程"><a href="#面向对象编程" class="headerlink" title="面向对象编程"></a>面向对象编程</h2><p>avaScript的原型链和Java的Class区别就在,它没有“Class”的概念,所有对象都是实例,所谓继承关系不过是把一个对象的原型指向另一个对象而已。</p><p>如果你把<code>xiaoming</code>的原型指向其他对象:</p><pre><code>var Bird = { fly: function () { console.log(this.name + ' is flying...'); }};xiaoming.__proto__ = Bird;</code></pre><p>现在<code>xiaoming</code>已经无法<code>run()</code>了,他已经变成了一只鸟:</p><pre><code>xiaoming.fly(); // 小明 is flying...</code></pre><p><em>请注意</em>,上述代码仅用于演示目的。在编写JavaScript代码时,不要直接用<code>obj.__proto__</code>去改变一个对象的原型,并且,低版本的IE也无法使用<code>__proto__</code>。<code>Object.create()</code>方法可以传入一个原型对象,并创建一个基于该原型的新对象,但是新对象什么属性都没有,因此,我们可以编写一个函数来创建<code>xiaoming</code>:</p><pre><code>// 原型对象:var Student = { name: 'Robot', height: 1.2, run: function () { console.log(this.name + ' is running...'); }};function createStudent(name) { // 基于Student原型创建一个新对象: var s = Object.create(Student); // 初始化新对象: s.name = name; return s;}var xiaoming = createStudent('小明');xiaoming.run(); // 小明 is running...xiaoming.__proto__ === Student; // true</code></pre><h3 id="创建对象"><a href="#创建对象" class="headerlink" title="创建对象"></a>创建对象</h3><p>JavaScript对每个创建的对象都会设置一个原型,指向它的原型对象。</p><p>当我们用<code>obj.xxx</code>访问一个对象的属性时,JavaScript引擎先在当前对象上查找该属性,如果没有找到,就到其原型对象上找,如果还没有找到,就一直上溯到<code>Object.prototype</code>对象,最后,如果还没有找到,就只能返回<code>undefined</code>。</p><p>函数也是一个对象,它的原型链是:</p><pre><code>foo ----> Function.prototype ----> Object.prototype ----> null</code></pre><p>由于<code>Function.prototype</code>定义了<code>apply()</code>等方法,因此,所有函数都可以调用<code>apply()</code>方法。</p><p>很容易想到,如果原型链很长,那么访问一个对象的属性就会因为花更多的时间查找而变得更慢,因此要注意不要把原型链搞得太长。</p><h3 id="构造函数"><a href="#构造函数" class="headerlink" title="构造函数"></a>构造函数</h3><p>除了直接用<code>{ ... }</code>创建一个对象外,JavaScript还可以用一种构造函数的方法来创建对象。它的用法是,先定义一个构造函数:</p><pre><code>function Student(name) { this.name = name; this.hello = function () { alert('Hello, ' + this.name + '!'); }}</code></pre><p>你会问,咦,这不是一个普通函数吗?</p><p>这确实是一个普通函数,但是在JavaScript中,可以用关键字<code>new</code>来调用这个函数,并返回一个对象:</p><pre><code>var xiaoming = new Student('小明');xiaoming.name; // '小明'xiaoming.hello(); // Hello, 小明!</code></pre><p><em>注意</em>,如果不写<code>new</code>,这就是一个普通函数,它返回<code>undefined</code>。但是,如果写了<code>new</code>,它就变成了一个构造函数,它绑定的<code>this</code>指向新创建的对象,并默认返回<code>this</code>,也就是说,不需要在最后写<code>return this;</code>。</p><h4 id="忘记写new怎么办"><a href="#忘记写new怎么办" class="headerlink" title="忘记写new怎么办"></a>忘记写new怎么办</h4><p>如果一个函数被定义为用于创建对象的构造函数,但是调用时忘记了写<code>new</code>怎么办?</p><p>在strict模式下,<code>this.name = name</code>将报错,因为<code>this</code>绑定为<code>undefined</code>,在非strict模式下,<code>this.name = name</code>不报错,因为<code>this</code>绑定为<code>window</code>,于是无意间创建了全局变量<code>name</code>,并且返回<code>undefined</code>,这个结果更糟糕。</p><p>所以,调用构造函数千万不要忘记写<code>new</code>。为了区分普通函数和构造函数,按照约定,构造函数首字母应当大写,而普通函数首字母应当小写,这样,一些语法检查工具如<a href="http://www.jslint.com/" target="_blank" rel="noopener">jslint</a>将可以帮你检测到漏写的<code>new</code>。</p><p>最后,我们还可以编写一个<code>createStudent()</code>函数,在内部封装所有的<code>new</code>操作。一个常用的编程模式像这样:</p><pre><code>function Student(props) { this.name = props.name || '匿名'; // 默认值为'匿名' this.grade = props.grade || 1; // 默认值为1}Student.prototype.hello = function () { alert('Hello, ' + this.name + '!');};function createStudent(props) { return new Student(props || {})}</code></pre><p>这个<code>createStudent()</code>函数有几个巨大的优点:一是不需要<code>new</code>来调用,二是参数非常灵活,可以不传,也可以这么传:</p><pre><code>var xiaoming = createStudent({ name: '小明'});xiaoming.grade; // 1</code></pre><h3 id="原型继承"><a href="#原型继承" class="headerlink" title="原型继承"></a>原型继承</h3><p>在传统的基于Class的语言如Java、C++中,继承的本质是扩展一个已有的Class,并生成新的Subclass。</p><p>由于这类语言严格区分类和实例,继承实际上是类型的扩展。但是,JavaScript由于采用原型继承,我们无法直接扩展一个Class,因为根本不存在Class这种类型。</p><p>但是办法还是有的。我们先回顾<code>Student</code>构造函数:</p><pre><code>function Student(props) { this.name = props.name || 'Unnamed';}Student.prototype.hello = function () { alert('Hello, ' + this.name + '!');}</code></pre><p>以及<code>Student</code>的原型链:</p><p><img src="" data-original="https://www.liaoxuefeng.com/files/attachments/1034288810160288/l" alt="js-proto"></p><p>现在,我们要基于<code>Student</code>扩展出<code>PrimaryStudent</code>,可以先定义出<code>PrimaryStudent</code>:</p><pre><code>function PrimaryStudent(props) { // 调用Student构造函数,绑定this变量: Student.call(this, props); this.grade = props.grade || 1;}</code></pre><p>但是,调用了<code>Student</code>构造函数不等于继承了<code>Student</code>,<code>PrimaryStudent</code>创建的对象的原型是:</p><pre><code>new PrimaryStudent() ----> PrimaryStudent.prototype ----> Object.prototype ----> null</code></pre><p>必须想办法把原型链修改为:</p><pre><code>new PrimaryStudent() ----> PrimaryStudent.prototype ----> Student.prototype ----> Object.prototype ----> null</code></pre><p>这样,原型链对了,继承关系就对了。新的基于<code>PrimaryStudent</code>创建的对象不但能调用<code>PrimaryStudent.prototype</code>定义的方法,也可以调用<code>Student.prototype</code>定义的方法。</p><p>如果你想用最简单粗暴的方法这么干:</p><pre><code>PrimaryStudent.prototype = Student.prototype;</code></pre><p>是不行的!如果这样的话,<code>PrimaryStudent</code>和<code>Student</code>共享一个原型对象,那还要定义<code>PrimaryStudent</code>干啥?</p><p>我们必须借助一个中间对象来实现正确的原型链,这个中间对象的原型要指向<code>Student.prototype</code>。为了实现这一点,参考道爷(就是发明JSON的那个道格拉斯)的代码,中间对象可以用一个空函数<code>F</code>来实现:</p><pre><code>// PrimaryStudent构造函数:function PrimaryStudent(props) { Student.call(this, props); this.grade = props.grade || 1;}// 空函数F:function F() {}// 把F的原型指向Student.prototype:F.prototype = Student.prototype;// 把PrimaryStudent的原型指向一个新的F对象,F对象的原型正好指向Student.prototype:PrimaryStudent.prototype = new F();// 把PrimaryStudent原型的构造函数修复为PrimaryStudent:PrimaryStudent.prototype.constructor = PrimaryStudent;// 继续在PrimaryStudent原型(就是new F()对象)上定义方法:PrimaryStudent.prototype.getGrade = function () { return this.grade;};// 创建xiaoming:var xiaoming = new PrimaryStudent({ name: '小明', grade: 2});xiaoming.name; // '小明'xiaoming.grade; // 2// 验证原型:xiaoming.__proto__ === PrimaryStudent.prototype; // truexiaoming.__proto__.__proto__ === Student.prototype; // true// 验证继承关系:xiaoming instanceof PrimaryStudent; // truexiaoming instanceof Student; // true</code></pre><p>用一张图来表示新的原型链:</p><p><img src="" data-original="https://www.liaoxuefeng.com/files/attachments/1034288859918112/l" alt="js-proto-extend"></p><p>注意,函数<code>F</code>仅用于桥接,我们仅创建了一个<code>new F()</code>实例,而且,没有改变原有的<code>Student</code>定义的原型链。</p><p>如果把继承这个动作用一个<code>inherits()</code>函数封装起来,还可以隐藏<code>F</code>的定义,并简化代码:</p><pre><code>function inherits(Child, Parent) { var F = function () {}; F.prototype = Parent.prototype; Child.prototype = new F(); Child.prototype.constructor = Child;}</code></pre><p>这个<code>inherits()</code>函数可以复用:</p><pre><code>function Student(props) { this.name = props.name || 'Unnamed';}Student.prototype.hello = function () { alert('Hello, ' + this.name + '!');}function PrimaryStudent(props) { Student.call(this, props); this.grade = props.grade || 1;}// 实现原型继承链:inherits(PrimaryStudent, Student);// 绑定其他方法到PrimaryStudent原型:PrimaryStudent.prototype.getGrade = function () { return this.grade;};</code></pre><h4 id="小结"><a href="#小结" class="headerlink" title="小结"></a>小结</h4><ol><li>定义新的构造函数,并在内部用<code>call()</code>调用希望“继承”的构造函数,并绑定<code>this</code>;</li><li>借助中间函数<code>F</code>实现原型链继承,最好通过封装的<code>inherits</code>函数完成;</li><li>继续在新的构造函数的原型上定义新方法。</li></ol><h3 id="class继承"><a href="#class继承" class="headerlink" title="class继承"></a>class继承</h3><p>在上面的章节中我们看到了JavaScript的对象模型是基于原型实现的,特点是简单,缺点是理解起来比传统的类-实例模型要困难,最大的缺点是继承的实现需要编写大量代码,并且需要正确实现原型链。</p><p>有没有更简单的写法?有!</p><p>新的关键字<code>class</code>从ES6开始正式被引入到JavaScript中。<code>class</code>的目的就是让定义类更简单。</p><p>我们先回顾用函数实现<code>Student</code>的方法:</p><pre><code>function Student(name) { this.name = name;}Student.prototype.hello = function () { alert('Hello, ' + this.name + '!');</code></pre><p>如果用新的<code>class</code>关键字来编写<code>Student</code>,可以这样写:</p><pre><code>class Student { constructor(name) { this.name = name; } hello() { alert('Hello, ' + this.name + '!'); }}</code></pre><p>比较一下就可以发现,<code>class</code>的定义包含了构造函数<code>constructor</code>和定义在原型对象上的函数<code>hello()</code>(注意没有<code>function</code>关键字),这样就避免了<code>Student.prototype.hello = function () {...}</code>这样分散的代码。</p><p>最后,创建一个<code>Student</code>对象代码和前面章节完全一样:</p><pre><code>var xiaoming = new Student('小明');xiaoming.hello();</code></pre><h3 id="lass继承"><a href="#lass继承" class="headerlink" title="lass继承"></a>lass继承</h3><p>用<code>class</code>定义对象的另一个巨大的好处是继承更方便了。想一想我们从<code>Student</code>派生一个<code>PrimaryStudent</code>需要编写的代码量。现在,原型继承的中间对象,原型对象的构造函数等等都不需要考虑了,直接通过<code>extends</code>来实现:</p><pre><code>class PrimaryStudent extends Student { constructor(name, grade) { super(name); // 记得用super调用父类的构造方法! this.grade = grade; } myGrade() { alert('I am at grade ' + this.grade); }}</code></pre><p>注意<code>PrimaryStudent</code>的定义也是class关键字实现的,而<code>extends</code>则表示原型链对象来自<code>Student</code>。子类的构造函数可能会与父类不太相同,例如,<code>PrimaryStudent</code>需要<code>name</code>和<code>grade</code>两个参数,并且需要通过<code>super(name)</code>来调用父类的构造函数,否则父类的<code>name</code>属性无法正常初始化。</p><p><code>PrimaryStudent</code>已经自动获得了父类<code>Student</code>的<code>hello</code>方法,我们又在子类中定义了新的<code>myGrade</code>方法。</p><p>ES6引入的<code>class</code>和原有的JavaScript原型继承有什么区别呢?实际上它们没有任何区别,<code>class</code>的作用就是让JavaScript引擎去实现原来需要我们自己编写的原型链代码。简而言之,用<code>class</code>的好处就是极大地简化了原型链代码。</p><p>你一定会问,<code>class</code>这么好用,能不能现在就用上?</p><p>现在用还早了点,因为不是所有的主流浏览器都支持ES6的class。如果一定要现在就用上,就需要一个工具把<code>class</code>代码转换为传统的<code>prototype</code>代码,可以试试<a href="https://babeljs.io/" target="_blank" rel="noopener">Babel</a>这个工具。</p><h2 id="浏览器"><a href="#浏览器" class="headerlink" title="浏览器"></a>浏览器</h2><p>由于JavaScript的出现就是为了能在浏览器中运行,所以,浏览器自然是JavaScript开发者必须要关注的。</p><p>目前主流的浏览器分这么几种:</p><ul><li>IE 6~11:国内用得最多的IE浏览器,历来对W3C标准支持差。从IE10开始支持ES6标准;</li><li>Chrome:Google出品的基于Webkit内核浏览器,内置了非常强悍的JavaScript引擎——V8。由于Chrome一经安装就时刻保持自升级,所以不用管它的版本,最新版早就支持ES6了;</li><li>Safari:Apple的Mac系统自带的基于Webkit内核的浏览器,从OS X 10.7 Lion自带的6.1版本开始支持ES6,目前最新的OS X 10.11 El Capitan自带的Safari版本是9.x,早已支持ES6;</li><li>Firefox:Mozilla自己研制的Gecko内核和JavaScript引擎OdinMonkey。早期的Firefox按版本发布,后来终于聪明地学习Chrome的做法进行自升级,时刻保持最新;</li><li>移动设备上目前iOS和Android两大阵营分别主要使用Apple的Safari和Google的Chrome,由于两者都是Webkit核心,结果HTML5首先在手机上全面普及(桌面绝对是Microsoft拖了后腿),对JavaScript的标准支持也很好,最新版本均支持ES6。</li></ul><p>其他浏览器如Opera等由于市场份额太小就被自动忽略了。</p><p>另外还要注意识别各种国产浏览器,如某某安全浏览器,某某旋风浏览器,它们只是做了一个壳,其核心调用的是IE,也有号称同时支持IE和Webkit的“双核”浏览器。</p><p>不同的浏览器对JavaScript支持的差异主要是,有些API的接口不一样,比如AJAX,File接口。对于ES6标准,不同的浏览器对各个特性支持也不一样。</p><p>在编写JavaScript的时候,就要充分考虑到浏览器的差异,尽量让同一份JavaScript代码能运行在不同的浏览器中。</p><h3 id="浏览器对象"><a href="#浏览器对象" class="headerlink" title="浏览器对象"></a>浏览器对象</h3><h4 id="window"><a href="#window" class="headerlink" title="window"></a>window</h4><p><code>window</code>对象不但充当全局作用域,而且表示浏览器窗口。</p><p><code>window</code>对象有<code>innerWidth</code>和<code>innerHeight</code>属性,可以获取浏览器窗口的内部宽度和高度。内部宽高是指除去菜单栏、工具栏、边框等占位元素后,用于显示网页的净宽高。</p><p>兼容性:IE<=8不支持。</p><p>对应的,还有一个<code>outerWidth</code>和<code>outerHeight</code>属性,可以获取浏览器窗口的整个宽高。</p><h4 id="navigator"><a href="#navigator" class="headerlink" title="navigator"></a>navigator</h4><p><code>navigator</code>对象表示浏览器的信息,最常用的属性包括:</p><ul><li>navigator.appName:浏览器名称;</li><li>navigator.appVersion:浏览器版本;</li><li>navigator.language:浏览器设置的语言;</li><li>navigator.platform:操作系统类型;</li><li>navigator.userAgent:浏览器设定的<code>User-Agent</code>字符串。</li></ul><p><em>请注意</em>,<code>navigator</code>的信息可以很容易地被用户修改,所以JavaScript读取的值不一定是正确的。很多初学者为了针对不同浏览器编写不同的代码,喜欢用<code>if</code>判断浏览器版本,例如:</p><pre><code>var width;if (getIEVersion(navigator.userAgent) < 9) { width = document.body.clientWidth;} else { width = window.innerWidth;}</code></pre><p>但这样既可能判断不准确,也很难维护代码。正确的方法是充分利用JavaScript对不存在属性返回<code>undefined</code>的特性,直接用短路运算符<code>||</code>计算:</p><pre><code>var width = window.innerWidth || document.body.clientWidth;</code></pre><h4 id="screen"><a href="#screen" class="headerlink" title="screen"></a>screen</h4><p><code>screen</code>对象表示屏幕的信息,常用的属性有:</p><ul><li>screen.width:屏幕宽度,以像素为单位;</li><li>screen.height:屏幕高度,以像素为单位;</li><li>screen.colorDepth:返回颜色位数,如8、16、24。</li></ul><h4 id="location"><a href="#location" class="headerlink" title="location"></a>location</h4><p><code>location</code>对象表示当前页面的URL信息。例如,一个完整的URL:</p><pre><code>http://www.example.com:8080/path/index.html?a=1&b=2#TOP</code></pre><p>可以用<code>location.href</code>获取。要获得URL各个部分的值,可以这么写:</p><pre><code>location.protocol; // 'http'location.host; // 'www.example.com'location.port; // '8080'location.pathname; // '/path/index.html'location.search; // '?a=1&b=2'location.hash; // 'TOP'</code></pre><p>要加载一个新页面,可以调用<code>location.assign()</code>。如果要重新加载当前页面,调用<code>location.reload()</code>方法非常方便。</p><h4 id="document"><a href="#document" class="headerlink" title="document"></a>document</h4><p><code>document</code>对象表示当前页面。由于HTML在浏览器中以DOM形式表示为树形结构,<code>document</code>对象就是整个DOM树的根节点。</p><p><code>document</code>的<code>title</code>属性是从HTML文档中的<code><title>xxx</title></code>读取的,但是可以动态改变:</p><p>请观察浏览器窗口标题的变化。</p><p>要查找DOM树的某个节点,需要从<code>document</code>对象开始查找。最常用的查找是根据ID和Tag Name。</p><p>用<code>document</code>对象提供的<code>getElementById()</code>和<code>getElementsByTagName()</code>可以按ID获得一个DOM节点和按Tag名称获得一组DOM节点:</p><pre><code>var menu = document.getElementById('drink-menu');var drinks = document.getElementsByTagName('dt');var i, s;s = '提供的饮料有:';for (i=0; i<drinks.length; i++) { s = s + drinks[i].innerHTML + ',';}console.log(s);</code></pre><p><code>document</code>对象还有一个<code>cookie</code>属性,可以获取当前页面的Cookie。</p><p>Cookie是由服务器发送的key-value标示符。因为HTTP协议是无状态的,但是服务器要区分到底是哪个用户发过来的请求,就可以用Cookie来区分。当一个用户成功登录后,服务器发送一个Cookie给浏览器,例如<code>user=ABC123XYZ(加密的字符串)...</code>,此后,浏览器访问该网站时,会在请求头附上这个Cookie,服务器根据Cookie即可区分出用户。</p><p>Cookie还可以存储网站的一些设置,例如,页面显示的语言等等。</p><p>JavaScript可以通过<code>document.cookie</code>读取到当前页面的Cookie:</p><pre><code>document.cookie; // 'v=123; remember=true; prefer=zh'</code></pre><p>为了解决这个问题,服务器在设置Cookie时可以使用<code>httpOnly</code>,设定了<code>httpOnly</code>的Cookie将不能被JavaScript读取。这个行为由浏览器实现,主流浏览器均支持<code>httpOnly</code>选项,IE从IE6 SP1开始支持。</p><p>为了确保安全,服务器端在设置Cookie时,应该始终坚持使用<code>httpOnly</code>。</p><h4 id="history"><a href="#history" class="headerlink" title="history"></a>history</h4><p><code>history</code>对象保存了浏览器的历史记录,JavaScript可以调用<code>history</code>对象的<code>back()</code>或<code>forward ()</code>,相当于用户点击了浏览器的“后退”或“前进”按钮。</p><p>这个对象属于历史遗留对象,对于现代Web页面来说,由于大量使用AJAX和页面交互,简单粗暴地调用<code>history.back()</code>可能会让用户感到非常愤怒。</p><p>新手开始设计Web页面时喜欢在登录页登录成功时调用<code>history.back()</code>,试图回到登录前的页面。这是一种错误的方法。</p><p>任何情况,你都不应该使用<code>history</code>这个对象了。</p><h3 id="操作DOM"><a href="#操作DOM" class="headerlink" title="操作DOM"></a>操作DOM</h3><p>由于HTML文档被浏览器解析后就是一棵DOM树,要改变HTML的结构,就需要通过JavaScript来操作DOM。</p><p>始终记住DOM是一个树形结构。操作一个DOM节点实际上就是这么几个操作:</p><ul><li>更新:更新该DOM节点的内容,相当于更新了该DOM节点表示的HTML的内容;</li><li>遍历:遍历该DOM节点下的子节点,以便进行进一步操作;</li><li>添加:在该DOM节点下新增一个子节点,相当于动态增加了一个HTML节点;</li><li>删除:将该节点从HTML中删除,相当于删掉了该DOM节点的内容以及它包含的所有子节点。</li></ul><p>在操作一个DOM节点前,我们需要通过各种方式先拿到这个DOM节点。最常用的方法是<code>document.getElementById()</code>和<code>document.getElementsByTagName()</code>,以及CSS选择器<code>document.getElementsByClassName()</code>。</p><p>由于ID在HTML文档中是唯一的,所以<code>document.getElementById()</code>可以直接定位唯一的一个DOM节点。<code>document.getElementsByTagName()</code>和<code>document.getElementsByClassName()</code>总是返回一组DOM节点。要精确地选择DOM,可以先定位父节点,再从父节点开始选择,以缩小范围。</p><p>第二种方法是使用<code>querySelector()</code>和<code>querySelectorAll()</code>,需要了解selector语法,然后使用条件来获取节点,更加方便:</p><pre><code>// 通过querySelector获取ID为q1的节点:var q1 = document.querySelector('#q1');// 通过querySelectorAll获取q1节点内的符合条件的所有节点:var ps = q1.querySelectorAll('div.highlighted > p');</code></pre><h4 id="更新DOM"><a href="#更新DOM" class="headerlink" title="更新DOM"></a>更新DOM</h4><p>拿到一个DOM节点后,我们可以对它进行更新。</p><p>可以直接修改节点的文本,方法有两种:</p><p>一种是修改<code>innerHTML</code>属性,这个方式非常强大,不但可以修改一个DOM节点的文本内容,还可以直接通过HTML片段修改DOM节点内部的子树:</p><pre><code>// 获取<p id="p-id">...</p>var p = document.getElementById('p-id');// 设置文本为abc:p.innerHTML = 'ABC'; // <p id="p-id">ABC</p>// 设置HTML:p.innerHTML = 'ABC <span style="color:red">RED</span> XYZ';// <p>...</p>的内部结构已修改</code></pre><p>用<code>innerHTML</code>时要注意,是否需要写入HTML。如果写入的字符串是通过网络拿到了,要注意对字符编码来避免XSS攻击。</p><p>第二种是修改<code>innerText</code>或<code>textContent</code>属性,这样可以自动对字符串进行HTML编码,保证无法设置任何HTML标签:</p><pre><code>// 获取<p id="p-id">...</p>var p = document.getElementById('p-id');// 设置文本:p.innerText = '<script>alert("Hi")</script>';// HTML被自动编码,无法设置一个<script>节点:// <p id="p-id">&lt;script&gt;alert("Hi")&lt;/script&gt;</p></code></pre><p>两者的区别在于读取属性时,<code>innerText</code>不返回隐藏元素的文本,而<code>textContent</code>返回所有文本。另外注意IE<9不支持<code>textContent</code>。</p><p>修改CSS也是经常需要的操作。DOM节点的<code>style</code>属性对应所有的CSS,可以直接获取或设置。因为CSS允许<code>font-size</code>这样的名称,但它并非JavaScript有效的属性名,所以需要在JavaScript中改写为驼峰式命名<code>fontSize</code>:</p><pre><code>// 获取<p id="p-id">...</p>var p = document.getElementById('p-id');// 设置CSS:p.style.color = '#ff0000';p.style.fontSize = '20px';p.style.paddingTop = '2em';</code></pre><h4 id="插入DOM"><a href="#插入DOM" class="headerlink" title="插入DOM"></a>插入DOM</h4><p>我们获得了某个DOM节点,想在这个DOM节点内插入新的DOM,应该如何做?</p><p>如果这个DOM节点是空的,例如,<code><div></div></code>,那么,直接使用<code>innerHTML = '<span>child</span>'</code>就可以修改DOM节点的内容,相当于“插入”了新的DOM节点。</p><p>如果这个DOM节点不是空的,那就不能这么做,因为<code>innerHTML</code>会直接替换掉原来的所有子节点。</p><p>有两个办法可以插入新的节点。一个是使用<code>appendChild</code>,把一个子节点添加到父节点的最后一个子节点。例如:</p><pre><code><!-- HTML结构 --><p id="js">JavaScript</p><div id="list"> <p id="java">Java</p> <p id="python">Python</p> <p id="scheme">Scheme</p></div></code></pre><p>把<code><p id="js">JavaScript</p></code>添加到<code><div id="list"></code>的最后一项:</p><pre><code>var js = document.getElementById('js'), list = document.getElementById('list');list.appendChild(js);</code></pre><p>现在,HTML结构变成了这样:</p><pre><code><!-- HTML结构 --><div id="list"> <p id="java">Java</p> <p id="python">Python</p> <p id="scheme">Scheme</p> <p id="js">JavaScript</p></div></code></pre><p>因为我们插入的<code>js</code>节点已经存在于当前的文档树,因此这个节点首先会从原先的位置删除,再插入到新的位置。</p><p>更多的时候我们会从零创建一个新的节点,然后插入到指定位置:</p><pre><code>var list = document.getElementById('list'), haskell = document.createElement('p');haskell.id = 'haskell';haskell.innerText = 'Haskell';list.appendChild(haskell);</code></pre><p>这样我们就动态添加了一个新的节点:</p><pre><code><!-- HTML结构 --><div id="list"> <p id="java">Java</p> <p id="python">Python</p> <p id="scheme">Scheme</p> <p id="haskell">Haskell</p></div></code></pre><h4 id="insertBefore"><a href="#insertBefore" class="headerlink" title="insertBefore"></a>insertBefore</h4><p>如果我们要把子节点插入到指定的位置怎么办?可以使用<code>parentElement.insertBefore(newElement, referenceElement);</code>,子节点会插入到<code>referenceElement</code>之前。</p><p>还是以上面的HTML为例,假定我们要把<code>Haskell</code>插入到<code>Python</code>之前:</p><pre><code><!-- HTML结构 --><div id="list"> <p id="java">Java</p> <p id="python">Python</p> <p id="scheme">Scheme</p></div></code></pre><p>可以这么写:</p><pre><code>var list = document.getElementById('list'), ref = document.getElementById('python'), haskell = document.createElement('p');haskell.id = 'haskell';haskell.innerText = 'Haskell';list.insertBefore(haskell, ref);</code></pre><p>新的HTML结构如下:</p><pre><code><!-- HTML结构 --><div id="list"> <p id="java">Java</p> <p id="haskell">Haskell</p> <p id="python">Python</p> <p id="scheme">Scheme</p></div></code></pre><p>可见,使用<code>insertBefore</code>重点是要拿到一个“参考子节点”的引用。很多时候,需要循环一个父节点的所有子节点,可以通过迭代<code>children</code>属性实现:</p><pre><code>var i, c, list = document.getElementById('list');for (i = 0; i < list.children.length; i++) { c = list.children[i]; // 拿到第i个子节点}</code></pre><h4 id="删除DOM"><a href="#删除DOM" class="headerlink" title="删除DOM"></a>删除DOM</h4><p>删除一个DOM节点就比插入要容易得多。</p><p>要删除一个节点,首先要获得该节点本身以及它的父节点,然后,调用父节点的<code>removeChild</code>把自己删掉:</p><pre><code>// 拿到待删除节点:var self = document.getElementById('to-be-removed');// 拿到父节点:var parent = self.parentElement;// 删除:var removed = parent.removeChild(self);removed === self; // true</code></pre><p>注意到删除后的节点虽然不在文档树中了,但其实它还在内存中,可以随时再次被添加到别的位置。</p><p>当你遍历一个父节点的子节点并进行删除操作时,要注意,<code>children</code>属性是一个只读属性,并且它在子节点变化时会实时更新。</p><p>例如,对于如下HTML结构:</p><pre><code><div id="parent"> <p>First</p> <p>Second</p></div></code></pre><p>当我们用如下代码删除子节点时:</p><pre><code>var parent = document.getElementById('parent');parent.removeChild(parent.children[0]);parent.removeChild(parent.children[1]); // <-- 浏览器报错</code></pre><p>浏览器报错:<code>parent.children[1]</code>不是一个有效的节点。原因就在于,当<code><p>First</p></code>节点被删除后,<code>parent.children</code>的节点数量已经从2变为了1,索引<code>[1]</code>已经不存在了。</p><p>因此,删除多个节点时,要注意<code>children</code>属性时刻都在变化。</p><h3 id="操作表单"><a href="#操作表单" class="headerlink" title="操作表单"></a>操作表单</h3><p>用JavaScript操作表单和操作DOM是类似的,因为表单本身也是DOM树。</p><p>不过表单的输入框、下拉框等可以接收用户输入,所以用JavaScript来操作表单,可以获得用户输入的内容,或者对一个输入框设置新的内容。</p><p>HTML表单的输入控件主要有以下几种:</p><ul><li>文本框,对应的<code><input type="text"></code>,用于输入文本;</li><li>口令框,对应的<code><input type="password"></code>,用于输入口令;</li><li>单选框,对应的<code><input type="radio"></code>,用于选择一项;</li><li>复选框,对应的<code><input type="checkbox"></code>,用于选择多项;</li><li>下拉框,对应的<code><select></code>,用于选择一项;</li><li>隐藏文本,对应的<code><input type="hidden"></code>,用户不可见,但表单提交时会把隐藏文本发送到服务器。</li></ul><h4 id="获取值"><a href="#获取值" class="headerlink" title="获取值"></a>获取值</h4><p>如果我们获得了一个<code><input></code>节点的引用,就可以直接调用<code>value</code>获得对应的用户输入值:</p><pre><code>// <input type="text" id="email">var input = document.getElementById('email');input.value; // '用户输入的值'</code></pre><p>这种方式可以应用于<code>text</code>、<code>password</code>、<code>hidden</code>以及<code>select</code>。但是,对于单选框和复选框,<code>value</code>属性返回的永远是HTML预设的值,而我们需要获得的实际是用户是否“勾上了”选项,所以应该用<code>checked</code>判断:</p><pre><code>// <label><input type="radio" name="weekday" id="monday" value="1"> Monday</label>// <label><input type="radio" name="weekday" id="tuesday" value="2"> Tuesday</label>var mon = document.getElementById('monday');var tue = document.getElementById('tuesday');mon.value; // '1'tue.value; // '2'mon.checked; // true或者falsetue.checked; // true或者false</code></pre><h4 id="设置值"><a href="#设置值" class="headerlink" title="设置值"></a>设置值</h4><p>设置值和获取值类似,对于<code>text</code>、<code>password</code>、<code>hidden</code>以及<code>select</code>,直接设置<code>value</code>就可以:</p><pre><code>// <input type="text" id="email">var input = document.getElementById('email');input.value = '[email protected]'; // 文本框的内容已更新</code></pre><p>对于单选框和复选框,设置<code>checked</code>为<code>true</code>或<code>false</code>即可。</p><h4 id="提交表单"><a href="#提交表单" class="headerlink" title="提交表单"></a>提交表单</h4><p>最后,JavaScript可以以两种方式来处理表单的提交(AJAX方式在后面章节介绍)。</p><p>方式一是通过<code><form></code>元素的<code>submit()</code>方法提交一个表单,例如,响应一个<code><button></code>的<code>click</code>事件,在JavaScript代码中提交表单:</p><pre><code><!-- HTML --><form id="test-form"> <input type="text" name="test"> <button type="button" onclick="doSubmitForm()">Submit</button></form><script>function doSubmitForm() { var form = document.getElementById('test-form'); // 可以在此修改form的input... // 提交form: form.submit();}</script></code></pre><p>这种方式的缺点是扰乱了浏览器对form的正常提交。浏览器默认点击<code><button type="submit"></code>时提交表单,或者用户在最后一个输入框按回车键。因此,第二种方式是响应<code><form></code>本身的<code>onsubmit</code>事件,在提交form时作修改:</p><pre><code><!-- HTML --><form id="test-form" onsubmit="return checkForm()"> <input type="text" name="test"> <button type="submit">Submit</button></form><script>function checkForm() { var form = document.getElementById('test-form'); // 可以在此修改form的input... // 继续下一步: return true;}</script></code></pre><p>注意要<code>return true</code>来告诉浏览器继续提交,如果<code>return false</code>,浏览器将不会继续提交form,这种情况通常对应用户输入有误,提示用户错误信息后终止提交form。</p><p>在检查和修改<code><input></code>时,要充分利用<code><input type="hidden"></code>来传递数据。</p><p>例如,很多登录表单希望用户输入用户名和口令,但是,安全考虑,提交表单时不传输明文口令,而是口令的MD5。普通JavaScript开发人员会直接修改<code><input></code>:</p><pre><code><!-- HTML --><form id="login-form" method="post" onsubmit="return checkForm()"> <input type="text" id="username" name="username"> <input type="password" id="password" name="password"> <button type="submit">Submit</button></form><script>function checkForm() { var pwd = document.getElementById('password'); // 把用户输入的明文变为MD5: pwd.value = toMD5(pwd.value); // 继续下一步: return true;}</script></code></pre><p>这个做法看上去没啥问题,但用户输入了口令提交时,口令框的显示会突然从几个<code>*</code>变成32个<code>*</code>(因为MD5有32个字符)。</p><p>要想不改变用户的输入,可以利用<code><input type="hidden"></code>实现:</p><pre><code><!-- HTML --><form id="login-form" method="post" onsubmit="return checkForm()"> <input type="text" id="username" name="username"> <input type="password" id="input-password"> <input type="hidden" id="md5-password" name="password"> <button type="submit">Submit</button></form><script>function checkForm() { var input_pwd = document.getElementById('input-password'); var md5_pwd = document.getElementById('md5-password'); // 把用户输入的明文变为MD5: md5_pwd.value = toMD5(input_pwd.value); // 继续下一步: return true;}</script></code></pre><p>注意到<code>id</code>为<code>md5-password</code>的<code><input></code>标记了<code>name="password"</code>,而用户输入的<code>id</code>为<code>input-password</code>的<code><input></code>没有<code>name</code>属性。没有<code>name</code>属性的<code><input></code>的数据不会被提交。</p><h3 id="操作文件"><a href="#操作文件" class="headerlink" title="操作文件"></a>操作文件</h3><p>在HTML表单中,可以上传文件的唯一控件就是<code><input type="file"></code>。</p><p><em>注意</em>:当一个表单包含<code><input type="file"></code>时,表单的<code>enctype</code>必须指定为<code>multipart/form-data</code>,<code>method</code>必须指定为<code>post</code>,浏览器才能正确编码并以<code>multipart/form-data</code>格式发送表单的数据。</p><h3 id="File-API"><a href="#File-API" class="headerlink" title="File API"></a>File API</h3><p>由于JavaScript对用户上传的文件操作非常有限,尤其是无法读取文件内容,使得很多需要操作文件的网页不得不用Flash这样的第三方插件来实现。</p><p>随着HTML5的普及,新增的File API允许JavaScript读取文件内容,获得更多的文件信息。</p><p>HTML5的File API提供了<code>File</code>和<code>FileReader</code>两个主要对象,可以获得文件信息并读取文件。</p><h3 id="AJAX"><a href="#AJAX" class="headerlink" title="AJAX"></a>AJAX</h3><p>AJAX不是JavaScript的规范,它只是一个哥们“发明”的缩写:Asynchronous JavaScript and XML,意思就是用JavaScript执行异步网络请求。</p><p>如果仔细观察一个Form的提交,你就会发现,一旦用户点击“Submit”按钮,表单开始提交,浏览器就会刷新页面,然后在新页面里告诉你操作是成功了还是失败了。如果不幸由于网络太慢或者其他原因,就会得到一个404页面。</p><p>这就是Web的运作原理:一次HTTP请求对应一个页面。</p><p>如果要让用户留在当前页面中,同时发出新的HTTP请求,就必须用JavaScript发送这个新请求,接收到数据后,再用JavaScript更新页面,这样一来,用户就感觉自己仍然停留在当前页面,但是数据却可以不断地更新。</p><p>最早大规模使用AJAX的就是Gmail,Gmail的页面在首次加载后,剩下的所有数据都依赖于AJAX来更新。</p><p>用JavaScript写一个完整的AJAX代码并不复杂,但是需要注意:AJAX请求是异步执行的,也就是说,要通过回调函数获得响应。</p><h3 id="CORS"><a href="#CORS" class="headerlink" title="CORS"></a>CORS</h3><p>如果浏览器支持HTML5,那么就可以一劳永逸地使用新的跨域策略:CORS了。</p><p>CORS全称Cross-Origin Resource Sharing,是HTML5规范定义的如何跨域访问资源。</p><p>了解CORS前,我们先搞明白概念:</p><p>Origin表示本域,也就是浏览器当前页面的域。当JavaScript向外域(如sina.com)发起请求后,浏览器收到响应后,首先检查<code>Access-Control-Allow-Origin</code>是否包含本域,如果是,则此次跨域请求成功,如果不是,则请求失败,JavaScript将无法获取到响应的任何数据。</p><p>用一个图来表示就是:</p><p><img src="" data-original="https://www.liaoxuefeng.com/files/attachments/1027024093709472/l" alt="js-cors"></p><p>假设本域是<code>my.com</code>,外域是<code>sina.com</code>,只要响应头<code>Access-Control-Allow-Origin</code>为<code>http://my.com</code>,或者是<code>*</code>,本次请求就可以成功。</p><p>可见,跨域能否成功,取决于对方服务器是否愿意给你设置一个正确的<code>Access-Control-Allow-Origin</code>,决定权始终在对方手中。</p><p>上面这种跨域请求,称之为“简单请求”。简单请求包括GET、HEAD和POST(POST的Content-Type类型 仅限<code>application/x-www-form-urlencoded</code>、<code>multipart/form-data</code>和<code>text/plain</code>),并且不能出现任何自定义头(例如,<code>X-Custom: 12345</code>),通常能满足90%的需求。</p><p>无论你是否需要用JavaScript通过CORS跨域请求资源,你都要了解CORS的原理。最新的浏览器全面支持HTML5。在引用外域资源时,除了JavaScript和CSS外,都要验证CORS。例如,当你引用了某个第三方CDN上的字体文件时:</p><pre><code>/* CSS */@font-face { font-family: 'FontAwesome'; src: url('http://cdn.com/fonts/fontawesome.ttf') format('truetype');}</code></pre><p>如果该CDN服务商未正确设置<code>Access-Control-Allow-Origin</code>,那么浏览器无法加载字体资源。</p><p>对于PUT、DELETE以及其他类型如<code>application/json</code>的POST请求,在发送AJAX请求之前,浏览器会先发送一个<code>OPTIONS</code>请求(称为preflighted请求)到这个URL上,询问目标服务器是否接受:</p><pre><code>OPTIONS /path/to/resource HTTP/1.1Host: bar.comOrigin: http://my.comAccess-Control-Request-Method: POST</code></pre><p>服务器必须响应并明确指出允许的Method:</p><pre><code>HTTP/1.1 200 OKAccess-Control-Allow-Origin: http://my.comAccess-Control-Allow-Methods: POST, GET, PUT, OPTIONSAccess-Control-Max-Age: 86400</code></pre><p>浏览器确认服务器响应的<code>Access-Control-Allow-Methods</code>头确实包含将要发送的AJAX请求的Method,才会继续发送AJAX,否则,抛出一个错误。</p><p>由于以<code>POST</code>、<code>PUT</code>方式传送JSON格式的数据在REST中很常见,所以要跨域正确处理<code>POST</code>和<code>PUT</code>请求,服务器端必须正确响应<code>OPTIONS</code>请求。</p><p>需要深入了解CORS的童鞋请移步<a href="http://www.w3.org/TR/cors/" target="_blank" rel="noopener">W3C文档</a>。</p><h3 id="Promise"><a href="#Promise" class="headerlink" title="Promise"></a>Promise</h3><p>在JavaScript的世界中,所有代码都是单线程执行的。</p><p>由于这个“缺陷”,导致JavaScript的所有网络操作,浏览器事件,都必须是异步执行。异步执行可以用回调函数实现:</p><pre><code>function callback() { console.log('Done');}console.log('before setTimeout()');setTimeout(callback, 1000); // 1秒钟后调用callback函数console.log('after setTimeout()');</code></pre><p>观察上述代码执行,在Chrome的控制台输出可以看到:</p><pre><code>before setTimeout()after setTimeout()(等待1秒后)Done</code></pre><p>可见,异步操作会在将来的某个时间点触发一个函数调用。</p><p>AJAX就是典型的异步操作。以上一节的代码为例:</p><pre><code>request.onreadystatechange = function () { if (request.readyState === 4) { if (request.status === 200) { return success(request.responseText); } else { return fail(request.status); } }}</code></pre><p>把回调函数<code>success(request.responseText)</code>和<code>fail(request.status)</code>写到一个AJAX操作里很正常,但是不好看,而且不利于代码复用。</p><p>有没有更好的写法?比如写成这样:</p><pre><code>var ajax = ajaxGet('http://...');ajax.ifSuccess(success) .ifFail(fail);</code></pre><p>这种链式写法的好处在于,先统一执行AJAX逻辑,不关心如何处理结果,然后,根据结果是成功还是失败,在将来的某个时候调用<code>success</code>函数或<code>fail</code>函数。</p><p>古人云:“君子一诺千金”,这种“承诺将来会执行”的对象在JavaScript中称为Promise对象。</p><pre><code>我们先看一个最简单的Promise例子:生成一个0-2之间的随机数,如果小于1,则等待一段时间后返回成功,否则返回失败:function test(resolve, reject) { var timeOut = Math.random() * 2; log('set timeout to: ' + timeOut + ' seconds.'); setTimeout(function () { if (timeOut < 1) { log('call resolve()...'); resolve('200 OK'); } else { log('call reject()...'); reject('timeout in ' + timeOut + ' seconds.'); } }, timeOut * 1000);}这个test()函数有两个参数,这两个参数都是函数,如果执行成功,我们将调用resolve('200 OK'),如果执行失败,我们将调用reject('timeout in ' + timeOut + ' seconds.')。可以看出,test()函数只关心自身的逻辑,并不关心具体的resolve和reject将如何处理结果。有了执行函数,我们就可以用一个Promise对象来执行它,并在将来某个时刻获得成功或失败的结果:var p1 = new Promise(test);var p2 = p1.then(function (result) { console.log('成功:' + result);});var p3 = p2.catch(function (reason) { console.log('失败:' + reason);});变量p1是一个Promise对象,它负责执行test函数。由于test函数在内部是异步执行的,当test函数执行成功时,我们告诉Promise对象:// 如果成功,执行这个函数:p1.then(function (result) { console.log('成功:' + result);});当test函数执行失败时,我们告诉Promise对象:p2.catch(function (reason) { console.log('失败:' + reason);});Promise对象可以串联起来,所以上述代码可以简化为:new Promise(test).then(function (result) { console.log('成功:' + result);}).catch(function (reason) { console.log('失败:' + reason);});</code></pre><p>可见Promise最大的好处是在异步执行的流程中,把执行代码和处理结果的代码清晰地分离了:</p><p><img src="" data-original="https://www.liaoxuefeng.com/files/attachments/1027242914217888/l" alt="promise"></p><p>Promise还可以做更多的事情,比如,有若干个异步任务,需要先做任务1,如果成功后再做任务2,任何任务失败则不再继续并执行错误处理函数。</p><p>要串行执行这样的异步任务,不用Promise需要写一层一层的嵌套代码。有了Promise,我们只需要简单地写:</p><pre><code>job1.then(job2).then(job3).catch(handleError);</code></pre><p>其中,<code>job1</code>、<code>job2</code>和<code>job3</code>都是Promise对象。</p><p>除了串行执行若干异步任务外,Promise还可以并行执行异步任务。</p><p>试想一个页面聊天系统,我们需要从两个不同的URL分别获得用户的个人信息和好友列表,这两个任务是可以并行执行的,用<code>Promise.all()</code>实现如下:</p><pre><code>var p1 = new Promise(function (resolve, reject) { setTimeout(resolve, 500, 'P1');});var p2 = new Promise(function (resolve, reject) { setTimeout(resolve, 600, 'P2');});// 同时执行p1和p2,并在它们都完成后执行then:Promise.all([p1, p2]).then(function (results) { console.log(results); // 获得一个Array: ['P1', 'P2']});</code></pre><p>有些时候,多个异步任务是为了容错。比如,同时向两个URL读取用户的个人信息,只需要获得先返回的结果即可。这种情况下,用<code>Promise.race()</code>实现:</p><pre><code>var p1 = new Promise(function (resolve, reject) { setTimeout(resolve, 500, 'P1');});var p2 = new Promise(function (resolve, reject) { setTimeout(resolve, 600, 'P2');});Promise.race([p1, p2]).then(function (result) { console.log(result); // 'P1'});</code></pre><p>由于<code>p1</code>执行较快,Promise的<code>then()</code>将获得结果<code>'P1'</code>。<code>p2</code>仍在继续执行,但执行结果将被丢弃。</p><p>如果我们组合使用Promise,就可以把很多异步任务以并行和串行的方式组合起来执行。</p><h3 id="Canvas"><a href="#Canvas" class="headerlink" title="Canvas"></a>Canvas</h3><p>Canvas是HTML5新增的组件,它就像一块幕布,可以用JavaScript在上面绘制各种图表、动画等。</p><p>没有Canvas的年代,绘图只能借助Flash插件实现,页面不得不用JavaScript和Flash进行交互。有了Canvas,我们就再也不需要Flash了,直接使用JavaScript完成绘制。</p><p>一个Canvas定义了一个指定尺寸的矩形框,在这个范围内我们可以随意绘制:</p><pre><code><canvas id="test-canvas" width="300" height="200"></canvas></code></pre><p>由于浏览器对HTML5标准支持不一致,所以,通常在<code><canvas></code>内部添加一些说明性HTML代码,如果浏览器支持Canvas,它将忽略<code><canvas></code>内部的HTML,如果浏览器不支持Canvas,它将显示<code><canvas></code>内部的HTML:</p><pre><code><canvas id="test-stock" width="300" height="200"> <p>Current Price: 25.51</p></canvas></code></pre><p>在使用Canvas前,用<code>canvas.getContext</code>来测试浏览器是否支持Canvas:</p><pre><code><!-- HTML代码 --><canvas id="test-canvas" width="200" heigth="100"> <p>你的浏览器不支持Canvas</p></canvas></code></pre><p><code>getContext('2d')</code>方法让我们拿到一个<code>CanvasRenderingContext2D</code>对象,所有的绘图操作都需要通过这个对象完成。</p><pre><code>var ctx = canvas.getContext('2d');</code></pre><p>如果需要绘制3D怎么办?HTML5还有一个WebGL规范,允许在Canvas中绘制3D图形:</p><pre><code>gl = canvas.getContext("webgl");</code></pre><p>本节我们只专注于绘制2D图形。</p><h3 id="绘制形状"><a href="#绘制形状" class="headerlink" title="绘制形状"></a>绘制形状</h3><p>我们可以在Canvas上绘制各种形状。在绘制前,我们需要先了解一下Canvas的坐标系统:</p><p><img src="" data-original="https://www.liaoxuefeng.com/files/attachments/1028111602807456/l" alt="canvas-xy"></p><p>Canvas的坐标以左上角为原点,水平向右为X轴,垂直向下为Y轴,以像素为单位,所以每个点都是非负整数。</p><p><code>CanvasRenderingContext2D</code>对象有若干方法来绘制图形:</p><pre><code>'use strict'; var canvas = document.getElementById('test-shape-canvas'), ctx = canvas.getContext('2d'); </code></pre><pre><code>ctx.clearRect(0, 0, 200, 200); // 擦除(0,0)位置大小为200x200的矩形,擦除的意思是把该区域变为透明ctx.fillStyle = '#dddddd'; // 设置颜色ctx.fillRect(10, 10, 130, 130); // 把(10,10)位置大小为130x130的矩形涂色// 利用Path绘制复杂路径:var path=new Path2D();path.arc(75, 75, 50, 0, Math.PI*2, true);path.moveTo(110,75);path.arc(75, 75, 35, 0, Math.PI, false);path.moveTo(65, 65);path.arc(60, 65, 5, 0, Math.PI*2, true);path.moveTo(95, 65);path.arc(90, 65, 5, 0, Math.PI*2, true);ctx.strokeStyle = '#0000ff';ctx.stroke(path);</code></pre><h3 id="绘制文本"><a href="#绘制文本" class="headerlink" title="绘制文本"></a>绘制文本</h3><p>绘制文本就是在指定的位置输出文本,可以设置文本的字体、样式、阴影等,与CSS完全一致:</p><pre><code>'use strict';var canvas = document.getElementById('test-text-canvas'), ctx = canvas.getContext('2d');</code></pre><pre><code>ctx.clearRect(0, 0, canvas.width, canvas.height);ctx.shadowOffsetX = 2;ctx.shadowOffsetY = 2;ctx.shadowBlur = 2;ctx.shadowColor = '#666666';ctx.font = '24px Arial';ctx.fillStyle = '#333333';ctx.fillText('带阴影的文字', 20, 40);</code></pre><p>Canvas除了能绘制基本的形状和文本,还可以实现动画、缩放、各种滤镜和像素转换等高级操作。如果要实现非常复杂的操作,考虑以下优化方案:</p><ul><li>通过创建一个不可见的Canvas来绘图,然后将最终绘制结果复制到页面的可见Canvas中;</li><li>尽量使用整数坐标而不是浮点数;</li><li>可以创建多个重叠的Canvas绘制不同的层,而不是在一个Canvas中绘制非常复杂的图;</li><li>背景图片如果不变可以直接用<code><img></code>标签并放到最底层。</li></ul><h2 id="jQuery"><a href="#jQuery" class="headerlink" title="jQuery"></a>jQuery</h2><p>你可能听说过jQuery,它名字起得很土,但却是JavaScript世界中使用最广泛的一个库。</p><p>江湖传言,全世界大约有80~90%的网站直接或间接地使用了jQuery。鉴于它如此流行,又如此好用,所以每一个入门JavaScript的前端工程师都应该了解和学习它。</p><p>jQuery这么流行,肯定是因为它解决了一些很重要的问题。实际上,jQuery能帮我们干这些事情:</p><ul><li>消除浏览器差异:你不需要自己写冗长的代码来针对不同的浏览器来绑定事件,编写AJAX等代码;</li><li>简洁的操作DOM的方法:写<code>$('#test')</code>肯定比<code>document.getElementById('test')</code>来得简洁;</li><li>轻松实现动画、修改CSS等各种操作。</li></ul><p>jQuery的理念“Write Less, Do More“,让你写更少的代码,完成更多的工作!</p><h4 id="使用jQuery"><a href="#使用jQuery" class="headerlink" title="使用jQuery"></a>使用jQuery</h4><p>使用jQuery只需要在页面的<code><head></code>引入jQuery文件即可:</p><pre><code><html><head> <script src="//code.jquery.com/jquery-1.11.3.min.js"></script> ...</head><body> ...</body></html></code></pre><h4 id="符号"><a href="#符号" class="headerlink" title="$符号"></a>$符号</h4><p><code>$</code>是著名的jQuery符号。实际上,jQuery把所有功能全部封装在一个全局变量<code>jQuery</code>中,而<code>$</code>也是一个合法的变量名,它是变量<code>jQuery</code>的别名:</p><pre><code>window.jQuery; // jQuery(selector, context)window.$; // jQuery(selector, context)$ === jQuery; // truetypeof($); // 'function'</code></pre><p><code>$</code>本质上就是一个函数,但是函数也是对象,于是<code>$</code>除了可以直接调用外,也可以有很多其他属性。</p><p><em>注意</em>,你看到的<code>$</code>函数名可能不是<code>jQuery(selector, context)</code>,因为很多JavaScript压缩工具可以对函数名和参数改名,所以压缩过的jQuery源码<code>$</code>函数可能变成<code>a(b, c)</code>。</p><p>绝大多数时候,我们都直接用<code>$</code>(因为写起来更简单嘛)。但是,如果<code>$</code>这个变量不幸地被占用了,而且还不能改,那我们就只能让<code>jQuery</code>把<code>$</code>变量交出来,然后就只能使用<code>jQuery</code>这个变量:</p><pre><code>$; // jQuery(selector, context)jQuery.noConflict();$; // undefinedjQuery; // jQuery(selector, context)</code></pre><p>这种黑魔法的原理是jQuery在占用<code>$</code>之前,先在内部保存了原来的<code>$</code>,调用<code>jQuery.noConflict()</code>时会把原来保存的变量还原。</p><h3 id="选择器"><a href="#选择器" class="headerlink" title="选择器"></a>选择器</h3><h3 id="按ID查找"><a href="#按ID查找" class="headerlink" title="按ID查找"></a>按ID查找</h3><p>如果某个DOM节点有<code>id</code>属性,利用jQuery查找如下:</p><pre><code>// 查找<div id="abc">:var div = $('#abc');</code></pre><p><em>注意</em>,<code>#abc</code>以<code>#</code>开头。返回的对象是jQuery对象。</p><p>什么是jQuery对象?jQuery对象类似数组,它的每个元素都是一个引用了DOM节点的对象。</p><p>以上面的查找为例,如果<code>id</code>为<code>abc</code>的<code><div></code>存在,返回的jQuery对象如下:</p><pre><code>[<div id="abc">...</div>]</code></pre><p>如果<code>id</code>为<code>abc</code>的<code><div></code>不存在,返回的jQuery对象如下:</p><pre><code>[]</code></pre><p>总之jQuery的选择器不会返回<code>undefined</code>或者<code>null</code>,这样的好处是你不必在下一行判断<code>if (div === undefined)</code>。</p><p>jQuery对象和DOM对象之间可以互相转化:</p><pre><code>var div = $('#abc'); // jQuery对象var divDom = div.get(0); // 假设存在div,获取第1个DOM元素var another = $(divDom); // 重新把DOM包装为jQuery对象</code></pre><p>通常情况下你不需要获取DOM对象,直接使用jQuery对象更加方便。如果你拿到了一个DOM对象,那可以简单地调用<code>$(aDomObject)</code>把它变成jQuery对象,这样就可以方便地使用jQuery的API了。</p><h3 id="按tag查找"><a href="#按tag查找" class="headerlink" title="按tag查找"></a>按tag查找</h3><p>按tag查找只需要写上tag名称就可以了:</p><pre><code>var ps = $('p'); // 返回所有<p>节点ps.length; // 数一数页面有多少个<p>节点</code></pre><h3 id="按class查找"><a href="#按class查找" class="headerlink" title="按class查找"></a>按class查找</h3><p>按class查找注意在class名称前加一个<code>.</code>:</p><pre><code>var a = $('.red'); // 所有节点包含`class="red"`都将返回// 例如:// <div class="red">...</div>// <p class="green red">...</p></code></pre><p>通常很多节点有多个class,我们可以查找同时包含<code>red</code>和<code>green</code>的节点:</p><pre><code>var a = $('.red.green'); // 注意没有空格!// 符合条件的节点:// <div class="red green">...</div>// <div class="blue green red">...</div></code></pre><h4 id="按属性查找"><a href="#按属性查找" class="headerlink" title="按属性查找"></a>按属性查找</h4><p>一个DOM节点除了<code>id</code>和<code>class</code>外还可以有很多属性,很多时候按属性查找会非常方便,比如在一个表单中按属性来查找:</p><pre><code>var email = $('[name=email]'); // 找出<??? name="email">var passwordInput = $('[type=password]'); // 找出<??? type="password">var a = $('[items="A B"]'); // 找出<??? items="A B"></code></pre><p>当属性的值包含空格等特殊字符时,需要用双引号括起来。</p><p>按属性查找还可以使用前缀查找或者后缀查找:</p><pre><code>var icons = $('[name^=icon]'); // 找出所有name属性值以icon开头的DOM// 例如: name="icon-1", name="icon-2"var names = $('[name$=with]'); // 找出所有name属性值以with结尾的DOM// 例如: name="startswith", name="endswith"</code></pre><p>这个方法尤其适合通过class属性查找,且不受class包含多个名称的影响:</p><pre><code>var icons = $('[class^="icon-"]'); // 找出所有class包含至少一个以`icon-`开头的DOM// 例如: class="icon-clock", class="abc icon-home"</code></pre><h4 id="组合查找"><a href="#组合查找" class="headerlink" title="组合查找"></a>组合查找</h4><p>组合查找就是把上述简单选择器组合起来使用。如果我们查找<code>$('[name=email]')</code>,很可能把表单外的<code><div name="email"></code>也找出来,但我们只希望查找<code><input></code>,就可以这么写:</p><pre><code>var emailInput = $('input[name=email]'); // 不会找出<div name="email"></code></pre><p>同样的,根据tag和class来组合查找也很常见:</p><pre><code>var tr = $('tr.red'); // 找出<tr class="red ...">...</tr></code></pre><h4 id="多项选择器"><a href="#多项选择器" class="headerlink" title="多项选择器"></a>多项选择器</h4><p>多项选择器就是把多个选择器用<code>,</code>组合起来一块选:</p><pre><code>$('p,div'); // 把<p>和<div>都选出来$('p.red,p.green'); // 把<p class="red">和<p class="green">都选出来</code></pre><p>要注意的是,选出来的元素是按照它们在HTML中出现的顺序排列的,而且不会有重复元素。例如,<code><p class="red green"></code>不会被上面的<code>$('p.red,p.green')</code>选择两次。</p><h4 id="层级选择器"><a href="#层级选择器" class="headerlink" title="层级选择器"></a>层级选择器</h4><p>除了基本的选择器外,jQuery的层级选择器更加灵活,也更强大。</p><p>因为DOM的结构就是层级结构,所以我们经常要根据层级关系进行选择。</p><h3 id="层级选择器(Descendant-Selector)"><a href="#层级选择器(Descendant-Selector)" class="headerlink" title="层级选择器(Descendant Selector)"></a>层级选择器(Descendant Selector)</h3><p>如果两个DOM元素具有层级关系,就可以用<code>$('ancestor descendant')</code>来选择,层级之间用空格隔开。例如:</p><pre><code><!-- HTML结构 --><div class="testing"> <ul class="lang"> <li class="lang-javascript">JavaScript</li> <li class="lang-python">Python</li> <li class="lang-lua">Lua</li> </ul></div></code></pre><p>要选出JavaScript,可以用层级选择器:</p><pre><code>$('ul.lang li.lang-javascript'); // [<li class="lang-javascript">JavaScript</li>]$('div.testing li.lang-javascript'); // [<li class="lang-javascript">JavaScript</li>]</code></pre><p>因为<code><div></code>和<code><ul></code>都是<code><li></code>的祖先节点,所以上面两种方式都可以选出相应的<code><li></code>节点。</p><p>要选择所有的<code><li></code>节点,用:</p><pre><code>$('ul.lang li');</code></pre><p>这种层级选择器相比单个的选择器好处在于,它缩小了选择范围,因为首先要定位父节点,才能选择相应的子节点,这样避免了页面其他不相关的元素</p><p>例如:</p><pre><code>$('form[name=upload] input');</code></pre><p>就把选择范围限定在<code>name</code>属性为<code>upload</code>的表单里。如果页面有很多表单,其他表单的<code><input></code>不会被选择。</p><p>多层选择也是允许的:</p><pre><code>$('form.test p input'); // 在form表单选择被<p>包含的<input></code></pre><h4 id="子选择器(Child-Selector)"><a href="#子选择器(Child-Selector)" class="headerlink" title="子选择器(Child Selector)"></a>子选择器(Child Selector)</h4><p>子选择器<code>$('parent>child')</code>类似层级选择器,但是限定了层级关系必须是父子关系,就是<code><child></code>节点必须是<code><parent></code>节点的直属子节点。还是以上面的例子:</p><pre><code>$('ul.lang>li.lang-javascript'); // 可以选出[<li class="lang-javascript">JavaScript</li>]$('div.testing>li.lang-javascript'); // [], 无法选出,因为<div>和<li>不构成父子关系</code></pre><h4 id="过滤器(Filter)"><a href="#过滤器(Filter)" class="headerlink" title="过滤器(Filter)"></a>过滤器(Filter)</h4><p>过滤器一般不单独使用,它通常附加在选择器上,帮助我们更精确地定位元素。观察过滤器的效果:</p><pre><code>$('ul.lang li'); // 选出JavaScript、Python和Lua 3个节点$('ul.lang li:first-child'); // 仅选出JavaScript$('ul.lang li:last-child'); // 仅选出Lua$('ul.lang li:nth-child(2)'); // 选出第N个元素,N从1开始$('ul.lang li:nth-child(even)'); // 选出序号为偶数的元素$('ul.lang li:nth-child(odd)'); // 选出序号为奇数的元素</code></pre><h4 id="表单相关"><a href="#表单相关" class="headerlink" title="表单相关"></a>表单相关</h4><p>针对表单元素,jQuery还有一组特殊的选择器:</p><ul><li><code>:input</code>:可以选择<code><input></code>,<code><textarea></code>,<code><select></code>和<code><button></code>;</li><li><code>:file</code>:可以选择<code><input type="file"></code>,和<code>input[type=file]</code>一样;</li><li><code>:checkbox</code>:可以选择复选框,和<code>input[type=checkbox]</code>一样;</li><li><code>:radio</code>:可以选择单选框,和<code>input[type=radio]</code>一样;</li><li><code>:focus</code>:可以选择当前输入焦点的元素,例如把光标放到一个<code><input></code>上,用<code>$('input:focus')</code>就可以选出;</li><li><code>:checked</code>:选择当前勾上的单选框和复选框,用这个选择器可以立刻获得用户选择的项目,如<code>$('input[type=radio]:checked')</code>;</li><li><code>:enabled</code>:可以选择可以正常输入的<code><input></code>、<code><select></code> 等,也就是没有灰掉的输入;</li><li><code>:disabled</code>:和<code>:enabled</code>正好相反,选择那些不能输入的。</li></ul><p>此外,jQuery还有很多有用的选择器,例如,选出可见的或隐藏的元素:</p><pre><code>$('div:visible'); // 所有可见的div$('div:hidden'); // 所有隐藏的div</code></pre><h4 id="查找和过滤"><a href="#查找和过滤" class="headerlink" title="查找和过滤"></a>查找和过滤</h4><p>通常情况下选择器可以直接定位到我们想要的元素,但是,当我们拿到一个jQuery对象后,还可以以这个对象为基准,进行查找和过滤。</p><p>最常见的查找是在某个节点的所有子节点中查找,使用<code>find()</code>方法,它本身又接收一个任意的选择器。例如如下的HTML结构:</p><ul><li>JavaScript</li><li>Python</li><li>Swift</li><li>Scheme</li><li>Haskell</li></ul><pre><code><!-- HTML结构 --><ul class="lang"> <li class="js dy">JavaScript</li> <li class="dy">Python</li> <li id="swift">Swift</li> <li class="dy">Scheme</li> <li name="haskell">Haskell</li></ul></code></pre><p>用<code>find()</code>查找:</p><pre><code>var ul = $('ul.lang'); // 获得<ul>var dy = ul.find('.dy'); // 获得JavaScript, Python, Schemevar swf = ul.find('#swift'); // 获得Swiftvar hsk = ul.find('[name=haskell]'); // 获得Haskell</code></pre><p>如果要从当前节点开始向上查找,使用<code>parent()</code>方法:</p><pre><code>var swf = $('#swift'); // 获得Swiftvar parent = swf.parent(); // 获得Swift的上层节点<ul>var a = swf.parent('.red'); // 获得Swift的上层节点<ul>,同时传入过滤条件。如果ul不符合条件,返回空jQuery对象</code></pre><p>对于位于同一层级的节点,可以通过<code>next()</code>和<code>prev()</code>方法,例如:</p><p>当我们已经拿到<code>Swift</code>节点后:</p><pre><code>var swift = $('#swift');swift.next(); // Schemeswift.next('[name=haskell]'); // 空的jQuery对象,因为Swift的下一个元素Scheme不符合条件[name=haskell]swift.prev(); // Pythonswift.prev('.dy'); // Python,因为Python同时符合过滤器条件.dy</code></pre><h4 id="过滤"><a href="#过滤" class="headerlink" title="过滤"></a>过滤</h4><p>和函数式编程的map、filter类似,jQuery对象也有类似的方法。</p><p><code>filter()</code>方法可以过滤掉不符合选择器条件的节点:</p><pre><code>var langs = $('ul.lang li'); // 拿到JavaScript, Python, Swift, Scheme和Haskellvar a = langs.filter('.dy'); // 拿到JavaScript, Python, Scheme</code></pre><p>或者传入一个函数,要特别注意函数内部的<code>this</code>被绑定为DOM对象,不是jQuery对象:</p><pre><code>var langs = $('ul.lang li'); // 拿到JavaScript, Python, Swift, Scheme和Haskelllangs.filter(function () { return this.innerHTML.indexOf('S') === 0; // 返回S开头的节点}); // 拿到Swift, Scheme</code></pre><p><code>map()</code>方法把一个jQuery对象包含的若干DOM节点转化为其他对象:</p><pre><code>var langs = $('ul.lang li'); // 拿到JavaScript, Python, Swift, Scheme和Haskellvar arr = langs.map(function () { return this.innerHTML;}).get(); // 用get()拿到包含string的Array:['JavaScript', 'Python', 'Swift', 'Scheme', 'Haskell']</code></pre><p>此外,一个jQuery对象如果包含了不止一个DOM节点,<code>first()</code>、<code>last()</code>和<code>slice()</code>方法可以返回一个新的jQuery对象,把不需要的DOM节点去掉:</p><pre><code>var langs = $('ul.lang li'); // 拿到JavaScript, Python, Swift, Scheme和Haskellvar js = langs.first(); // JavaScript,相当于$('ul.lang li:first-child')var haskell = langs.last(); // Haskell, 相当于$('ul.lang li:last-child')var sub = langs.slice(2, 4); // Swift, Scheme, 参数和数组的slice()方法一致</code></pre><h3 id="操作DOM-1"><a href="#操作DOM-1" class="headerlink" title="操作DOM"></a>操作DOM</h3><h4 id="修改Text和HTML"><a href="#修改Text和HTML" class="headerlink" title="修改Text和HTML"></a>修改Text和HTML</h4><p>jQuery对象的<code>text()</code>和<code>html()</code>方法分别获取节点的文本和原始HTML文本,例如,如下的HTML结构:</p><pre><code><!-- HTML结构 --><ul id="test-ul"> <li class="js">JavaScript</li> <li name="book">Java &amp; JavaScript</li></ul></code></pre><p>分别获取文本和HTML:</p><pre><code>$('#test-ul li[name=book]').text(); // 'Java & JavaScript'$('#test-ul li[name=book]').html(); // 'Java &amp; JavaScript'</code></pre><p>如何设置文本或HTML?jQuery的API设计非常巧妙:无参数调用<code>text()</code>是获取文本,传入参数就变成设置文本,HTML也是类似操作,自己动手试试:</p><pre><code>'use strict'; var j1 = $('#test-ul li.js');var j2 = $('#test-ul li[name=book]');</code></pre><pre><code>j1.html('<span style="color: red">JavaScript</span>');j2.text('JavaScript & ECMAScript');</code></pre><p>一个jQuery对象可以包含0个或任意个DOM对象,它的方法实际上会作用在对应的每个DOM节点上。在上面的例子中试试:</p><pre><code>$('#test-ul li').text('JS'); // 是不是两个节点都变成了JS?</code></pre><p>所以jQuery对象的另一个好处是我们可以执行一个操作,作用在对应的一组DOM节点上。即使选择器没有返回任何DOM节点,调用jQuery对象的方法仍然不会报错:</p><pre><code>// 如果不存在id为not-exist的节点:$('#not-exist').text('Hello'); // 代码不报错,没有节点被设置为'Hello'</code></pre><p>这意味着jQuery帮你免去了许多<code>if</code>语句。</p><p><em>注意</em>,jQuery对象的所有方法都返回一个jQuery对象(可能是新的也可能是自身),这样我们可以进行链式调用,非常方便。</p><p>jQuery对象的<code>css()</code>方法可以这么用:</p><pre><code>var div = $('#test-div');div.css('color'); // '#000033', 获取CSS属性div.css('color', '#336699'); // 设置CSS属性div.css('color', ''); // 清除CSS属性</code></pre><p>为了和JavaScript保持一致,CSS属性可以用<code>'background-color'</code>和<code>'backgroundColor'</code>两种格式。</p><p><code>css()</code>方法将作用于DOM节点的<code>style</code>属性,具有最高优先级。如果要修改<code>class</code>属性,可以用jQuery提供的下列方法:</p><pre><code>var div = $('#test-div');div.hasClass('highlight'); // false, class是否包含highlightdiv.addClass('highlight'); // 添加highlight这个classdiv.removeClass('highlight'); // 删除highlight这个class</code></pre><h4 id="显示和隐藏DOM"><a href="#显示和隐藏DOM" class="headerlink" title="显示和隐藏DOM"></a>显示和隐藏DOM</h4><p>要隐藏一个DOM,我们可以设置CSS的<code>display</code>属性为<code>none</code>,利用<code>css()</code>方法就可以实现。不过,要显示这个DOM就需要恢复原有的<code>display</code>属性,这就得先记下来原有的<code>display</code>属性到底是<code>block</code>还是<code>inline</code>还是别的值。</p><p>考虑到显示和隐藏DOM元素使用非常普遍,jQuery直接提供<code>show()</code>和<code>hide()</code>方法,我们不用关心它是如何修改<code>display</code>属性的,总之它能正常工作:</p><pre><code>var a = $('a[target=_blank]');a.hide(); // 隐藏a.show(); // 显示</code></pre><p><em>注意</em>,隐藏DOM节点并未改变DOM树的结构,它只影响DOM节点的显示。这和删除DOM节点是不同的。</p><h4 id="获取DOM信息"><a href="#获取DOM信息" class="headerlink" title="获取DOM信息"></a>获取DOM信息</h4><p>利用jQuery对象的若干方法,我们直接可以获取DOM的高宽等信息,而无需针对不同浏览器编写特定代码:</p><pre><code>// 浏览器可视窗口大小:$(window).width(); // 800$(window).height(); // 600// HTML文档大小:$(document).width(); // 800$(document).height(); // 3500// 某个div的大小:var div = $('#test-div');div.width(); // 600div.height(); // 300div.width(400); // 设置CSS属性 width: 400px,是否生效要看CSS是否有效div.height('200px'); // 设置CSS属性 height: 200px,是否生效要看CSS是否有效</code></pre><p><code>attr()</code>和<code>removeAttr()</code>方法用于操作DOM节点的属性:</p><pre><code>// <div id="test-div" name="Test" start="1">...</div>var div = $('#test-div');div.attr('data'); // undefined, 属性不存在div.attr('name'); // 'Test'div.attr('name', 'Hello'); // div的name属性变为'Hello'div.removeAttr('name'); // 删除name属性div.attr('name'); // undefined</code></pre><p><code>prop()</code>方法和<code>attr()</code>类似,但是HTML5规定有一种属性在DOM节点中可以没有值,只有出现与不出现两种,例如:</p><pre><code><input id="test-radio" type="radio" name="test" checked value="1"></code></pre><p>等价于:</p><pre><code><input id="test-radio" type="radio" name="test" checked="checked" value="1"></code></pre><p><code>attr()</code>和<code>prop()</code>对于属性<code>checked</code>处理有所不同:</p><pre><code>var radio = $('#test-radio');radio.attr('checked'); // 'checked'radio.prop('checked'); // true</code></pre><p><code>prop()</code>返回值更合理一些。不过,用<code>is()</code>方法判断更好:</p><pre><code>var radio = $('#test-radio');radio.is(':checked'); // true</code></pre><p>类似的属性还有<code>selected</code>,处理时最好用<code>is(':selected')</code>。</p><h4 id="操作表单-1"><a href="#操作表单-1" class="headerlink" title="操作表单"></a>操作表单</h4><p>对于表单元素,jQuery对象统一提供<code>val()</code>方法获取和设置对应的<code>value</code>属性:</p><pre><code>/* <input id="test-input" name="email" value=""> <select id="test-select" name="city"> <option value="BJ" selected>Beijing</option> <option value="SH">Shanghai</option> <option value="SZ">Shenzhen</option> </select> <textarea id="test-textarea">Hello</textarea>*/var input = $('#test-input'), select = $('#test-select'), textarea = $('#test-textarea');input.val(); // 'test'input.val('[email protected]'); // 文本框的内容已变为[email protected](); // 'BJ'select.val('SH'); // 选择框已变为Shanghaitextarea.val(); // 'Hello'textarea.val('Hi'); // 文本区域已更新为'Hi'</code></pre><p>可见,一个<code>val()</code>就统一了各种输入框的取值和赋值的问题。</p><h4 id="修改DOM结构"><a href="#修改DOM结构" class="headerlink" title="修改DOM结构"></a>修改DOM结构</h4><h5 id="添加DOM"><a href="#添加DOM" class="headerlink" title="添加DOM"></a>添加DOM</h5><p>要添加新的DOM节点,除了通过jQuery的<code>html()</code>这种暴力方法外,还可以用<code>append()</code>方法,例如:</p><pre><code><div id="test-div"> <ul> <li><span>JavaScript</span></li> <li><span>Python</span></li> <li><span>Swift</span></li> </ul></div></code></pre><p>如何向列表新增一个语言?首先要拿到<code><ul></code>节点:</p><pre><code>var ul = $('#test-div>ul');</code></pre><p>然后,调用<code>append()</code>传入HTML片段:</p><pre><code>ul.append('<li><span>Haskell</span></li>');</code></pre><p>除了接受字符串,<code>append()</code>还可以传入原始的DOM对象,jQuery对象和函数对象:</p><pre><code>// 创建DOM对象:var ps = document.createElement('li');ps.innerHTML = '<span>Pascal</span>';// 添加DOM对象:ul.append(ps);// 添加jQuery对象:ul.append($('#scheme'));// 添加函数对象:ul.append(function (index, html) { return '<li><span>Language - ' + index + '</span></li>';});</code></pre><p>传入函数时,要求返回一个字符串、DOM对象或者jQuery对象。因为jQuery的<code>append()</code>可能作用于一组DOM节点,只有传入函数才能针对每个DOM生成不同的子节点。</p><p><code>append()</code>把DOM添加到最后,<code>prepend()</code>则把DOM添加到最前。</p><p>另外注意,如果要添加的DOM节点已经存在于HTML文档中,它会首先从文档移除,然后再添加,也就是说,用<code>append()</code>,你可以移动一个DOM节点。</p><p>如果要把新节点插入到指定位置,例如,JavaScript和Python之间,那么,可以先定位到JavaScript,然后用<code>after()</code>方法:</p><pre><code>var js = $('#test-div>ul>li:first-child');js.after('<li><span>Lua</span></li>');</code></pre><p>也就是说,同级节点可以用<code>after()</code>或者<code>before()</code>方法。</p><h5 id="删除节点"><a href="#删除节点" class="headerlink" title="删除节点"></a>删除节点</h5><p>要删除DOM节点,拿到jQuery对象后直接调用<code>remove()</code>方法就可以了。如果jQuery对象包含若干DOM节点,实际上可以一次删除多个DOM节点:</p><pre><code>var li = $('#test-div>ul>li');li.remove(); // 所有<li>全被删除</code></pre><h3 id="事件"><a href="#事件" class="headerlink" title="事件"></a>事件</h3><p>因为JavaScript在浏览器中以单线程模式运行,页面加载后,一旦页面上所有的JavaScript代码被执行完后,就只能依赖触发事件来执行JavaScript代码。</p><p>浏览器在接收到用户的鼠标或键盘输入后,会自动在对应的DOM节点上触发相应的事件。如果该节点已经绑定了对应的JavaScript处理函数,该函数就会自动调用。</p><p>由于不同的浏览器绑定事件的代码都不太一样,所以用jQuery来写代码,就屏蔽了不同浏览器的差异,我们总是编写相同的代码。</p><p>举个例子,假设要在用户点击了超链接时弹出提示框,我们用jQuery这样绑定一个<code>click</code>事件:</p><pre><code>/* HTML: * * <a id="test-link" href="#0">点我试试</a> * */// 获取超链接的jQuery对象:var a = $('#test-link');a.on('click', function () { alert('Hello!');});</code></pre><p><code>on</code>方法用来绑定一个事件,我们需要传入事件名称和对应的处理函数。</p><p>另一种更简化的写法是直接调用<code>click()</code>方法:</p><pre><code>a.click(function () { alert('Hello!');});</code></pre><p>jQuery能够绑定的事件主要包括:</p><h3 id="鼠标事件"><a href="#鼠标事件" class="headerlink" title="鼠标事件"></a>鼠标事件</h3><ul><li>click: 鼠标单击时触发;</li><li>dblclick:鼠标双击时触发;</li><li>mouseenter:鼠标进入时触发;</li><li>mouseleave:鼠标移出时触发;</li><li>mousemove:鼠标在DOM内部移动时触发;</li><li>hover:鼠标进入和退出时触发两个函数,相当于mouseenter加上mouseleave。</li></ul><h3 id="键盘事件"><a href="#键盘事件" class="headerlink" title="键盘事件"></a>键盘事件</h3><p>键盘事件仅作用在当前焦点的DOM上,通常是<code><input></code>和<code><textarea></code>。</p><ul><li>keydown:键盘按下时触发;</li><li>keyup:键盘松开时触发;</li><li>keypress:按一次键后触发。</li></ul><h3 id="其他事件"><a href="#其他事件" class="headerlink" title="其他事件"></a>其他事件</h3><ul><li>focus:当DOM获得焦点时触发;</li><li>blur:当DOM失去焦点时触发;</li><li>change:当<code><input></code>、<code><select></code>或<code><textarea></code>的内容改变时触发;</li><li>submit:当<code><form></code>提交时触发;</li><li>ready:当页面被载入并且DOM树完成初始化后触发。</li></ul><p>其中,<code>ready</code>仅作用于<code>document</code>对象。由于<code>ready</code>事件在DOM完成初始化后触发,且只触发一次,所以非常适合用来写其他的初始化代码。假设我们想给一个<code><form></code>表单绑定<code>submit</code>事件,下面的代码没有预期的效果:</p><pre><code><html><head> <script> // 代码有误: $('#testForm).on('submit', function () { alert('submit!'); }); </script></head><body> <form id="testForm"> ... </form></body></code></pre><p>因为JavaScript在此执行的时候,<code><form></code>尚未载入浏览器,所以<code>$('#testForm)</code>返回<code>[]</code>,并没有绑定事件到任何DOM上。</p><p>所以我们自己的初始化代码必须放到<code>document</code>对象的<code>ready</code>事件中,保证DOM已完成初始化:</p><pre><code><html><head> <script> $(document).on('ready', function () { $('#testForm).on('submit', function () { alert('submit!'); }); }); </script></head><body> <form id="testForm"> ... </form></body></code></pre><p>这样写就没有问题了。因为相关代码会在DOM树初始化后再执行。</p><p>由于<code>ready</code>事件使用非常普遍,所以可以这样简化:</p><pre><code>$(document).ready(function () { // on('submit', function)也可以简化: $('#testForm).submit(function () { alert('submit!'); });});</code></pre><p>甚至还可以再简化为:</p><pre><code>$(function () { // init...});</code></pre><p>上面的这种写法最为常见。如果你遇到<code>$(function () {...})</code>的形式,牢记这是<code>document</code>对象的<code>ready</code>事件处理函数。</p><p>完全可以反复绑定事件处理函数,它们会依次执行:</p><pre><code>$(function () { console.log('init A...');});$(function () { console.log('init B...');});$(function () { console.log('init C...');});</code></pre><h3 id="取消绑定"><a href="#取消绑定" class="headerlink" title="取消绑定"></a>取消绑定</h3><p>一个已被绑定的事件可以解除绑定,通过<code>off('click', function)</code>实现:</p><pre><code>function hello() { alert('hello!');}a.click(hello); // 绑定事件// 10秒钟后解除绑定:setTimeout(function () { a.off('click', hello);}, 10000);</code></pre><p>需要特别注意的是,下面这种写法是无效的:</p><pre><code>// 绑定事件:a.click(function () { alert('hello!');});// 解除绑定:a.off('click', function () { alert('hello!');});</code></pre><p>这是因为两个匿名函数虽然长得一模一样,但是它们是两个<em>不同</em>的函数对象,<code>off('click', function () {...})</code>无法移除已绑定的第一个匿名函数。</p><p>为了实现移除效果,可以使用<code>off('click')</code>一次性移除已绑定的<code>click</code>事件的所有处理函数。</p><p>同理,无参数调用<code>off()</code>一次性移除已绑定的所有类型的事件处理函数。</p><h5 id="事件触发条件"><a href="#事件触发条件" class="headerlink" title="事件触发条件"></a>事件触发条件</h5><p>一个需要注意的问题是,事件的触发总是由用户操作引发的。例如,我们监控文本框的内容改动:</p><pre><code>var input = $('#test-input');input.change(function () { console.log('changed...');});</code></pre><p>当用户在文本框中输入时,就会触发<code>change</code>事件。但是,如果用JavaScript代码去改动文本框的值,将<em>不会</em>触发<code>change</code>事件:</p><pre><code>var input = $('#test-input');input.val('change it!'); // 无法触发change事件</code></pre><p>有些时候,我们希望用代码触发<code>change</code>事件,可以直接调用无参数的<code>change()</code>方法来触发该事件:</p><pre><code>var input = $('#test-input');input.val('change it!');input.change(); // 触发change事件</code></pre><p><code>input.change()</code>相当于<code>input.trigger('change')</code>,它是<code>trigger()</code>方法的简写。</p><p>为什么我们希望手动触发一个事件呢?如果不这么做,很多时候,我们就得写两份一模一样的代码。</p><h5 id="浏览器安全限制"><a href="#浏览器安全限制" class="headerlink" title="浏览器安全限制"></a>浏览器安全限制</h5><p>在浏览器中,有些JavaScript代码只有在用户触发下才能执行,例如,<code>window.open()</code>函数:</p><pre><code>// 无法弹出新窗口,将被浏览器屏蔽:$(function () { window.open('/');});</code></pre><p>这些“敏感代码”只能由用户操作来触发:</p><pre><code>var button1 = $('#testPopupButton1');var button2 = $('#testPopupButton2');function popupTestWindow() { window.open('/');}button1.click(function () { popupTestWindow();});button2.click(function () { // 不立刻执行popupTestWindow(),3秒后执行: setTimeout(popupTestWindow, 3000);});</code></pre><p>当用户点击<code>button1</code>时,<code>click</code>事件被触发,由于<code>popupTestWindow()</code>在<code>click</code>事件处理函数内执行,这是浏览器允许的,而<code>button2</code>的<code>click</code>事件并未立刻执行<code>popupTestWindow()</code>,延迟执行的<code>popupTestWindow()</code>将被浏览器拦截。</p><h3 id="动画"><a href="#动画" class="headerlink" title="动画"></a>动画</h3><p>用JavaScript实现动画,原理非常简单:我们只需要以固定的时间间隔(例如,0.1秒),每次把DOM元素的CSS样式修改一点(例如,高宽各增加10%),看起来就像动画了。</p><p>但是要用JavaScript手动实现动画效果,需要编写非常复杂的代码。如果想要把动画效果用函数封装起来便于复用,那考虑的事情就更多了。</p><p>使用jQuery实现动画,代码已经简单得不能再简化了:只需要一行代码!</p><p>让我们先来看看jQuery内置的几种动画样式:</p><h3 id="show-hide"><a href="#show-hide" class="headerlink" title="show / hide"></a>show / hide</h3><p>直接以无参数形式调用<code>show()</code>和<code>hide()</code>,会显示和隐藏DOM元素。但是,只要传递一个时间参数进去,就变成了动画:</p><pre><code>var div = $('#test-show-hide');div.hide(3000); // 在3秒钟内逐渐消失</code></pre><p>时间以毫秒为单位,但也可以是<code>'slow'</code>,<code>'fast'</code>这些字符串:</p><pre><code>var div = $('#test-show-hide');div.show('slow'); // 在0.6秒钟内逐渐显示</code></pre><p><code>toggle()</code>方法则根据当前状态决定是<code>show()</code>还是<code>hide()</code>。</p><p>你可能已经看出来了,<code>show()</code>和<code>hide()</code>是从左上角逐渐展开或收缩的,而<code>slideUp()</code>和<code>slideDown()</code>则是在垂直方向逐渐展开或收缩的。</p><p><code>slideUp()</code>把一个可见的DOM元素收起来,效果跟拉上窗帘似的,<code>slideDown()</code>相反,而<code>slideToggle()</code>则根据元素是否可见来决定下一步动作:</p><pre><code>var div = $('#test-slide');div.slideUp(3000); // 在3秒钟内逐渐向上消失</code></pre><h3 id="fadeIn-fadeOut"><a href="#fadeIn-fadeOut" class="headerlink" title="fadeIn / fadeOut"></a>fadeIn / fadeOut</h3><p><code>fadeIn()</code>和<code>fadeOut()</code>的动画效果是淡入淡出,也就是通过不断设置DOM元素的<code>opacity</code>属性来实现,而<code>fadeToggle()</code>则根据元素是否可见来决定下一步动作:</p><pre><code>var div = $('#test-fade');div.fadeOut('slow'); // 在0.6秒内淡出</code></pre><h3 id="自定义动画"><a href="#自定义动画" class="headerlink" title="自定义动画"></a>自定义动画</h3><p>如果上述动画效果还不能满足你的要求,那就祭出最后大招:<code>animate()</code>,它可以实现任意动画效果,我们需要传入的参数就是DOM元素最终的CSS状态和时间,jQuery在时间段内不断调整CSS直到达到我们设定的值:</p><pre><code>var div = $('#test-animate');div.animate({ opacity: 0.25, width: '256px', height: '256px'}, 3000); // 在3秒钟内CSS过渡到设定值</code></pre><p><code>animate()</code>还可以再传入一个函数,当动画结束时,该函数将被调用:</p><pre><code>var div = $('#test-animate');div.animate({ opacity: 0.25, width: '256px', height: '256px'}, 3000, function () { console.log('动画已结束'); // 恢复至初始状态: $(this).css('opacity', '1.0').css('width', '128px').css('height', '128px');});</code></pre><p>实际上这个回调函数参数对于基本动画也是适用的。</p><h4 id="串行动画"><a href="#串行动画" class="headerlink" title="串行动画"></a>串行动画</h4><p>jQuery的动画效果还可以串行执行,通过<code>delay()</code>方法还可以实现暂停,这样,我们可以实现更复杂的动画效果,而代码却相当简单:</p><pre><code>var div = $('#test-animates');// 动画效果:slideDown - 暂停 - 放大 - 暂停 - 缩小div.slideDown(2000) .delay(1000) .animate({ width: '256px', height: '256px' }, 2000) .delay(1000) .animate({ width: '128px', height: '128px' }, 2000);}</script></code></pre><p>因为动画需要执行一段时间,所以jQuery必须不断返回新的Promise对象才能后续执行操作。简单地把动画封装在函数中是不够的。</p><h4 id="为什么有的动画没有效果"><a href="#为什么有的动画没有效果" class="headerlink" title="为什么有的动画没有效果"></a>为什么有的动画没有效果</h4><p>你可能会遇到,有的动画如<code>slideUp()</code>根本没有效果。这是因为jQuery动画的原理是逐渐改变CSS的值,如<code>height</code>从<code>100px</code>逐渐变为<code>0</code>。但是很多不是block性质的DOM元素,对它们设置<code>height</code>根本就不起作用,所以动画也就没有效果。</p><p>此外,jQuery也没有实现对<code>background-color</code>的动画效果,用<code>animate()</code>设置<code>background-color</code>也没有效果。这种情况下可以使用CSS3的<code>transition</code>实现动画效果。</p><h3 id="AJAX-1"><a href="#AJAX-1" class="headerlink" title="AJAX"></a>AJAX</h3><p>jQuery在全局对象<code>jQuery</code>(也就是<code>$</code>)绑定了<code>ajax()</code>函数,可以处理AJAX请求。<code>ajax(url, settings)</code>函数需要接收一个URL和一个可选的<code>settings</code>对象,常用的选项如下:</p><ul><li>async:是否异步执行AJAX请求,默认为<code>true</code>,千万不要指定为<code>false</code>;</li><li>method:发送的Method,缺省为<code>'GET'</code>,可指定为<code>'POST'</code>、<code>'PUT'</code>等;</li><li>contentType:发送POST请求的格式,默认值为<code>'application/x-www-form-urlencoded; charset=UTF-8'</code>,也可以指定为<code>text/plain</code>、<code>application/json</code>;</li><li>data:发送的数据,可以是字符串、数组或object。如果是GET请求,data将被转换成query附加到URL上,如果是POST请求,根据contentType把data序列化成合适的格式;</li><li>headers:发送的额外的HTTP头,必须是一个object;</li><li>dataType:接收的数据格式,可以指定为<code>'html'</code>、<code>'xml'</code>、<code>'json'</code>、<code>'text'</code>等,缺省情况下根据响应的<code>Content-Type</code>猜测。</li></ul><p>下面的例子发送一个GET请求,并返回一个JSON格式的数据:</p><pre><code>var jqxhr = $.ajax('/api/categories', { dataType: 'json'});// 请求已经发送了</code></pre><p>不过,如何用回调函数处理返回的数据和出错时的响应呢?</p><p>还记得Promise对象吗?jQuery的jqXHR对象类似一个Promise对象,我们可以用链式写法来处理各种回调:</p><pre><code>'use strict';function ajaxLog(s) { var txt = $('#test-response-text'); txt.val(txt.val() + '\n' + s);}$('#test-response-text').val('');</code></pre><pre><code>var jqxhr = $.ajax('/api/categories', { dataType: 'json'}).done(function (data) { ajaxLog('成功, 收到的数据: ' + JSON.stringify(data));}).fail(function (xhr, status) { ajaxLog('失败: ' + xhr.status + ', 原因: ' + status);}).always(function () { ajaxLog('请求完成: 无论成功或失败都会调用');});</code></pre><h4 id="get"><a href="#get" class="headerlink" title="get"></a>get</h4><p>对常用的AJAX操作,jQuery提供了一些辅助方法。由于GET请求最常见,所以jQuery提供了<code>get()</code>方法,可以这么写:</p><pre><code>var jqxhr = $.get('/path/to/resource', { name: 'Bob Lee', check: 1});</code></pre><p>第二个参数如果是object,jQuery自动把它变成query string然后加到URL后面,实际的URL是:</p><pre><code>/path/to/resource?name=Bob%20Lee&check=1</code></pre><p>这样我们就不用关心如何用URL编码并构造一个query string了。</p><h4 id="post"><a href="#post" class="headerlink" title="post"></a>post</h4><p><code>post()</code>和<code>get()</code>类似,但是传入的第二个参数默认被序列化为<code>application/x-www-form-urlencoded</code>:</p><pre><code>var jqxhr = $.post('/path/to/resource', { name: 'Bob Lee', check: 1});</code></pre><p>实际构造的数据<code>name=Bob%20Lee&check=1</code>作为POST的body被发送。</p><h4 id="getJSON"><a href="#getJSON" class="headerlink" title="getJSON"></a>getJSON</h4><p>由于JSON用得越来越普遍,所以jQuery也提供了<code>getJSON()</code>方法来快速通过GET获取一个JSON对象:</p><pre><code>var jqxhr = $.getJSON('/path/to/resource', { name: 'Bob Lee', check: 1}).done(function (data) { // data已经被解析为JSON对象了});</code></pre><p>jQuery的AJAX完全封装的是JavaScript的AJAX操作,所以它的安全限制和前面讲的用JavaScript写AJAX完全一样。</p><p>如果需要使用JSONP,可以在<code>ajax()</code>中设置<code>jsonp: 'callback'</code>,让jQuery实现JSONP跨域加载数据。</p><p>关于跨域的设置请参考<a href="https://www.liaoxuefeng.com/wiki/1022910821149312/1023022129105888" target="_blank" rel="noopener">浏览器</a> - <a href="https://www.liaoxuefeng.com/wiki/1022910821149312/1023022332902400" target="_blank" rel="noopener">AJAX</a>一节中CORS的设置。</p><h3 id="扩展"><a href="#扩展" class="headerlink" title="扩展"></a>扩展</h3><p>当我们使用jQuery对象的方法时,由于jQuery对象可以操作一组DOM,而且支持链式操作,所以用起来非常方便。</p><p>但是jQuery内置的方法永远不可能满足所有的需求。比如,我们想要高亮显示某些DOM元素,用jQuery可以这么实现:</p><pre><code>$('span.hl').css('backgroundColor', '#fffceb').css('color', '#d85030');$('p a.hl').css('backgroundColor', '#fffceb').css('color', '#d85030');</code></pre><p>总是写重复代码可不好,万一以后还要修改字体就更麻烦了,能不能统一起来,写个<code>highlight()</code>方法?</p><pre><code>$('span.hl').highlight();$('p a.hl').highlight();</code></pre><p>答案是肯定的。我们可以扩展jQuery来实现自定义方法。将来如果要修改高亮的逻辑,只需修改一处扩展代码。这种方式也称为编写jQuery插件。</p><p>最终,我们得出编写一个jQuery插件的原则:</p><ol><li>给<code>$.fn</code>绑定函数,实现插件的代码逻辑;</li><li>插件函数最后要<code>return this;</code>以支持链式调用;</li><li>插件函数要有默认值,绑定在<code>$.fn.<pluginName>.defaults</code>上;</li><li>用户在调用时可传入设定值以便覆盖默认值。</li></ol><h2 id="错误处理"><a href="#错误处理" class="headerlink" title="错误处理"></a>错误处理</h2><h2 id="underscore"><a href="#underscore" class="headerlink" title="underscore"></a>underscore</h2>]]></content>
</entry>
<entry>
<title>Elasticsearch</title>
<link href="/2020/051614457.html"/>
<url>/2020/051614457.html</url>
<content type="html"><![CDATA[<h1 id="ElasticSearch-安装"><a href="#ElasticSearch-安装" class="headerlink" title="ElasticSearch 安装"></a>ElasticSearch 安装</h1><p>声明:jdk1.8最低要求</p><blockquote><p>window安装</p></blockquote><p>1, 解压就可以用了</p><p>2,熟悉目录</p><pre class=" language-txt"><code class="language-txt">bin 启动文件 config 配置文件 log4j2 日志配置文件 jvm.options java 虚拟机相关的配置elasticsearch.yml elasticsearch 的配置文件! 默认 9200 端口! 跨域! lib 相关jar包 logs 日志! modules 功能模块 plugins 插件!</code></pre><p>3 启动,访问9200</p><p>4 访问测试</p><pre class=" language-json"><code class="language-json"><span class="token punctuation">{</span> <span class="token property">"name"</span> <span class="token operator">:</span> <span class="token string">"DESKTOP-PDG11N4"</span><span class="token punctuation">,</span> <span class="token property">"cluster_name"</span> <span class="token operator">:</span> <span class="token string">"elasticsearch"</span><span class="token punctuation">,</span> <span class="token property">"cluster_uuid"</span> <span class="token operator">:</span> <span class="token string">"Uq-cvd0OQDuB4Ung68gM9g"</span><span class="token punctuation">,</span> <span class="token property">"version"</span> <span class="token operator">:</span> <span class="token punctuation">{</span> <span class="token property">"number"</span> <span class="token operator">:</span> <span class="token string">"7.6.1"</span><span class="token punctuation">,</span> <span class="token property">"build_flavor"</span> <span class="token operator">:</span> <span class="token string">"default"</span><span class="token punctuation">,</span> <span class="token property">"build_type"</span> <span class="token operator">:</span> <span class="token string">"zip"</span><span class="token punctuation">,</span> <span class="token property">"build_hash"</span> <span class="token operator">:</span> <span class="token string">"aa751e09be0a5072e8570670309b1f12348f023b"</span><span class="token punctuation">,</span> <span class="token property">"build_date"</span> <span class="token operator">:</span> <span class="token string">"2020-02-29T00:15:25.529771Z"</span><span class="token punctuation">,</span> <span class="token property">"build_snapshot"</span> <span class="token operator">:</span> <span class="token boolean">false</span><span class="token punctuation">,</span> <span class="token property">"lucene_version"</span> <span class="token operator">:</span> <span class="token string">"8.4.0"</span><span class="token punctuation">,</span> <span class="token property">"minimum_wire_compatibility_version"</span> <span class="token operator">:</span> <span class="token string">"6.8.0"</span><span class="token punctuation">,</span> <span class="token property">"minimum_index_compatibility_version"</span> <span class="token operator">:</span> <span class="token string">"6.0.0-beta1"</span> <span class="token punctuation">}</span><span class="token punctuation">,</span> <span class="token property">"tagline"</span> <span class="token operator">:</span> <span class="token string">"You Know, for Search"</span><span class="token punctuation">}</span></code></pre><blockquote><p>安装可视化es head插件</p></blockquote><p>先要安装node.js</p><p>1.启动</p><pre><code>npm installnpm run start</code></pre><p>2.连接测试发现,存在跨域问题:配置es</p><pre><code>http.cors.enabled: true http.cors.allow-origin: "*"</code></pre><p>3.重启es服务器,然后再次连接</p><blockquote><p>安装Kibana</p></blockquote><p>1 测试,打开</p><p>2 开发工具</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587816489346.png" alt="1587816296927"></p><p>3 汉化</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587816446098.png" alt="1587816446098"></p><h1 id="ES核心概念"><a href="#ES核心概念" class="headerlink" title="ES核心概念"></a>ES核心概念</h1><hr><p><strong>物理设计:</strong> </p><p>elasticsearch 在后台把每个<strong>索引划分成多个分片</strong>,每分分片可以在集群中的不同服务器间迁移 </p><p>一个人就是一个集群!默认的集群名称就是 elaticsearh</p><p><strong>逻辑设计:</strong> </p><p>一个索引类型中,包含多个文档,比如说文档1,文档2。 当我们索引一篇文档时,可以通过这样的一各 </p><p>顺序找到 它: 索引 ▷ 类型 ▷ 文档ID ,通过这个组合我们就能索引到某个具体的文档。 注意:ID不必是整 </p><p>数,实际上它是个字 符串。</p><blockquote><p>文档</p></blockquote><p>就是一条条数据</p><ul><li><p>自我包含,一篇文档同时包含字段和对应的值,也就是同时包含 key:value! </p></li><li><p>可以是层次型的,一个文档中包含自文档,复杂的逻辑实体就是这么来的! {就是一个json对象! </p><p>fastjson进行自动转换!} </p></li><li><p>灵活的结构,文档不依赖预先定义的模式,我们知道关系型数据库中,要提前定义字段才能使用, </p><p>在elasticsearch中,对于字段是非常灵活的,有时候,我们可以忽略该字段,或者动态的添加一个 </p><p>新的字段。 </p></li></ul><blockquote><p>类型</p></blockquote><p>类型是文档的逻辑容器,就像关系型数据库一样,表格是行的容器。 类型中对于字段的定义称为映射, </p><p>比如 name 映 射为字符串类型。 我们说文档是无模式的,它们不需要拥有映射中所定义的所有字段, </p><p>比如新增一个字段,那么elasticsearch是怎么做的呢?elasticsearch会自动的将新字段加入映射,但是这 </p><p>个字段的不确定它是什么类型,elasticsearch就开始猜,如果这个值是18,那么elasticsearch会认为它 </p><p>是整形。 但是elasticsearch也可能猜不对, 所以最安全的方式就是提前定义好所需要的映射,这点跟关 </p><p>系型数据库殊途同归了,先定义好字段,然后再使用,别 整什么幺蛾子。</p><blockquote><p>索引</p></blockquote><p>索引就是数据库</p><p>elasticsearch的索引和Lucene的索引对比 </p><p>在elasticsearch中, 索引 (库)这个词被频繁使用,这就是术语的使用。 在elasticsearch中,索引被 </p><p>分为多个分片,每份 分片是一个Lucene的索引。所以一个elasticsearch索引是由多个Lucene索引组成 </p><p>的。别问为什么,谁让elasticsearch使用Lucene作为底层呢! 如无特指,说起索引都是指elasticsearch </p><p>的索引。 </p><p>接下来的一切操作都在kibana中Dev Tools下的Console里完成。基础操作! </p><h1 id="IK分词器"><a href="#IK分词器" class="headerlink" title="IK分词器"></a>IK分词器</h1><hr><blockquote><p>什么是IK分词器</p></blockquote><p>分词:即把一段中文或者别的划分成一个个的关键字,我们在搜索时候会把自己的信息进行分词,会把 </p><p>数据库中或者索引库中的数据进行分词,然后进行一个匹配操作,默认的中文分词是将每个字看成一个 </p><p>词,比如 “我爱狂神” 会被分为”我”,”爱”,”狂”,”神”,这显然是不符合要求的,所以我们需要安装中文分词 </p><p>器ik来解决这个问题。 </p><p>如果要使用中文,建议使用ik分词器!</p><p>解压</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587818192683.png" alt="1587818192683"></p><blockquote><p>查看不同的分词效果</p></blockquote><p>ik_smart</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587818927560.png" alt="1587818927560"></p><p>ik_max_word</p><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587818969679.png" alt="1587818969679"></p><blockquote><p>自己配置分词字典</p></blockquote><p><img src="" data-original="C:%5CUsers%5CCR553%5CAppData%5CRoaming%5CTypora%5Ctypora-user-images%5C1587819421696.png" alt="1587819421696"></p><p>字典记得改成utf-8</p><h1 id="ES-API"><a href="#ES-API" class="headerlink" title="ES API"></a>ES API</h1><hr><blockquote><p>具体API</p></blockquote><pre class=" language-java"><code class="language-java"><span class="token annotation punctuation">@Autowired</span> <span class="token keyword">private</span> RestHighLevelClient restHighLevelClient<span class="token punctuation">;</span> <span class="token comment" spellcheck="true">/** * 创建索引 * * @throws IOException */</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">create</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> IOException <span class="token punctuation">{</span> CreateIndexRequest request <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">CreateIndexRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> CreateIndexResponse response <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">indices</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">create</span><span class="token punctuation">(</span>request<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>response<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">/** * 查看索引 * * @throws IOException */</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">get</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> IOException <span class="token punctuation">{</span> GetIndexRequest request <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">GetIndexRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">boolean</span> exists <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">indices</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">exists</span><span class="token punctuation">(</span>request<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>exists<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">/** * 删除索引 * * @throws IOException */</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">delete</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> IOException <span class="token punctuation">{</span> DeleteIndexRequest request <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">DeleteIndexRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> AcknowledgedResponse delete <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">indices</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">delete</span><span class="token punctuation">(</span>request<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>delete<span class="token punctuation">.</span><span class="token function">isAcknowledged</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">//获取文档</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">addDocument</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> IOException <span class="token punctuation">{</span> User user <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">User</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user<span class="token punctuation">.</span><span class="token function">setAge</span><span class="token punctuation">(</span><span class="token number">12</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user<span class="token punctuation">.</span><span class="token function">setName</span><span class="token punctuation">(</span><span class="token string">"梅海迪"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> IndexRequest request <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">IndexRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> request<span class="token punctuation">.</span><span class="token function">timeout</span><span class="token punctuation">(</span><span class="token string">"1s"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> request<span class="token punctuation">.</span><span class="token function">id</span><span class="token punctuation">(</span><span class="token string">"1"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> request<span class="token punctuation">.</span><span class="token function">source</span><span class="token punctuation">(</span>JSON<span class="token punctuation">.</span><span class="token function">toJSONString</span><span class="token punctuation">(</span>user<span class="token punctuation">)</span><span class="token punctuation">,</span> XContentType<span class="token punctuation">.</span>JSON<span class="token punctuation">)</span><span class="token punctuation">;</span> IndexResponse response <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">index</span><span class="token punctuation">(</span>request<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>response<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">//获取文档,是否存在</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">testExists</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> IOException <span class="token punctuation">{</span> GetRequest request <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">GetRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">,</span> <span class="token string">"1"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">boolean</span> exists <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">exists</span><span class="token punctuation">(</span>request<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>exists<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">//获取文档信息</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">testGetDoc</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> IOException <span class="token punctuation">{</span> GetRequest request <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">GetRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">,</span> <span class="token string">"1"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> GetResponse response <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">get</span><span class="token punctuation">(</span>request<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>response<span class="token punctuation">.</span><span class="token function">getSourceAsString</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>response<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">//更新文档</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">testUpdate</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> IOException <span class="token punctuation">{</span> UpdateRequest updateRequest <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">UpdateRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">,</span> <span class="token string">"1"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> User user <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">User</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user<span class="token punctuation">.</span><span class="token function">setName</span><span class="token punctuation">(</span><span class="token string">"梅海迪真帅"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user<span class="token punctuation">.</span><span class="token function">setAge</span><span class="token punctuation">(</span><span class="token number">14</span><span class="token punctuation">)</span><span class="token punctuation">;</span> updateRequest<span class="token punctuation">.</span><span class="token function">doc</span><span class="token punctuation">(</span>JSON<span class="token punctuation">.</span><span class="token function">toJSONString</span><span class="token punctuation">(</span>user<span class="token punctuation">)</span><span class="token punctuation">,</span>XContentType<span class="token punctuation">.</span>JSON<span class="token punctuation">)</span><span class="token punctuation">;</span> UpdateResponse update <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">update</span><span class="token punctuation">(</span>updateRequest<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>update<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">//删除文档</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">testDelete</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> Exception <span class="token punctuation">{</span> DeleteRequest request <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">DeleteRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">,</span> <span class="token string">"1"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> request<span class="token punctuation">.</span><span class="token function">timeout</span><span class="token punctuation">(</span><span class="token string">"1s"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> DeleteResponse delete <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">delete</span><span class="token punctuation">(</span>request<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>delete<span class="token punctuation">.</span><span class="token function">status</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">//批量插入</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">testBulk</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> Exception <span class="token punctuation">{</span> BulkRequest bulkRequest <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">BulkRequest</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> List<span class="token operator"><</span>User<span class="token operator">></span> list<span class="token operator">=</span><span class="token keyword">new</span> <span class="token class-name">ArrayList</span><span class="token operator"><</span>User<span class="token operator">></span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> User user <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">User</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> User user2 <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">User</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> User user3 <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">User</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user<span class="token punctuation">.</span><span class="token function">setAge</span><span class="token punctuation">(</span><span class="token number">12</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user<span class="token punctuation">.</span><span class="token function">setName</span><span class="token punctuation">(</span><span class="token string">"梅海迪"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> list<span class="token punctuation">.</span><span class="token function">add</span><span class="token punctuation">(</span>user<span class="token punctuation">)</span><span class="token punctuation">;</span> user2<span class="token punctuation">.</span><span class="token function">setAge</span><span class="token punctuation">(</span><span class="token number">13</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user2<span class="token punctuation">.</span><span class="token function">setName</span><span class="token punctuation">(</span><span class="token string">"张三"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> list<span class="token punctuation">.</span><span class="token function">add</span><span class="token punctuation">(</span>user2<span class="token punctuation">)</span><span class="token punctuation">;</span> user3<span class="token punctuation">.</span><span class="token function">setAge</span><span class="token punctuation">(</span><span class="token number">44</span><span class="token punctuation">)</span><span class="token punctuation">;</span> user3<span class="token punctuation">.</span><span class="token function">setName</span><span class="token punctuation">(</span><span class="token string">"历史"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> list<span class="token punctuation">.</span><span class="token function">add</span><span class="token punctuation">(</span>user3<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span> <span class="token punctuation">(</span><span class="token keyword">int</span> i <span class="token operator">=</span> <span class="token number">0</span><span class="token punctuation">;</span> i <span class="token operator"><</span> list<span class="token punctuation">.</span><span class="token function">size</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> i<span class="token operator">++</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> bulkRequest<span class="token punctuation">.</span><span class="token function">add</span><span class="token punctuation">(</span> <span class="token keyword">new</span> <span class="token class-name">IndexRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">)</span> <span class="token punctuation">.</span><span class="token function">id</span><span class="token punctuation">(</span><span class="token string">""</span><span class="token operator">+</span><span class="token punctuation">(</span>i<span class="token operator">+</span><span class="token number">1</span><span class="token punctuation">)</span><span class="token punctuation">)</span> <span class="token punctuation">.</span><span class="token function">source</span><span class="token punctuation">(</span>JSON<span class="token punctuation">.</span><span class="token function">toJSONString</span><span class="token punctuation">(</span>list<span class="token punctuation">.</span><span class="token function">get</span><span class="token punctuation">(</span>i<span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">,</span>XContentType<span class="token punctuation">.</span>JSON<span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> BulkResponse bulk <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">bulk</span><span class="token punctuation">(</span>bulkRequest<span class="token punctuation">,</span> RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>bulk<span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token comment" spellcheck="true">//查询</span> <span class="token annotation punctuation">@Test</span> <span class="token keyword">void</span> <span class="token function">TestSearch</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">throws</span> Exception <span class="token punctuation">{</span> SearchRequest searchRequest <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">SearchRequest</span><span class="token punctuation">(</span><span class="token string">"mhd_index"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> SearchSourceBuilder searchSourceBuilder <span class="token operator">=</span> <span class="token keyword">new</span> <span class="token class-name">SearchSourceBuilder</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span> TermQueryBuilder termQueryBuilder <span class="token operator">=</span> QueryBuilders<span class="token punctuation">.</span><span class="token function">termQuery</span><span class="token punctuation">(</span><span class="token string">"age"</span><span class="token punctuation">,</span> <span class="token number">13</span><span class="token punctuation">)</span><span class="token punctuation">;</span> searchSourceBuilder<span class="token punctuation">.</span><span class="token function">query</span><span class="token punctuation">(</span>termQueryBuilder<span class="token punctuation">)</span><span class="token punctuation">;</span> searchSourceBuilder<span class="token punctuation">.</span><span class="token function">timeout</span><span class="token punctuation">(</span><span class="token keyword">new</span> <span class="token class-name">TimeValue</span><span class="token punctuation">(</span><span class="token number">60</span><span class="token punctuation">,</span> TimeUnit<span class="token punctuation">.</span>SECONDS<span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> searchRequest<span class="token punctuation">.</span><span class="token function">source</span><span class="token punctuation">(</span>searchSourceBuilder<span class="token punctuation">)</span><span class="token punctuation">;</span> SearchResponse searchResponse <span class="token operator">=</span> restHighLevelClient<span class="token punctuation">.</span><span class="token function">search</span><span class="token punctuation">(</span>searchRequest<span class="token punctuation">,</span>RequestOptions<span class="token punctuation">.</span>DEFAULT<span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>JSON<span class="token punctuation">.</span><span class="token function">toJSONString</span><span class="token punctuation">(</span>searchResponse<span class="token punctuation">.</span><span class="token function">getHits</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span><span class="token string">"######################"</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token keyword">for</span> <span class="token punctuation">(</span>SearchHit documentFields <span class="token operator">:</span> searchResponse<span class="token punctuation">.</span><span class="token function">getHits</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">getHits</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> System<span class="token punctuation">.</span>out<span class="token punctuation">.</span><span class="token function">println</span><span class="token punctuation">(</span>documentFields<span class="token punctuation">.</span><span class="token function">getSourceAsMap</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token punctuation">}</span> <span class="token punctuation">}</span></code></pre><h1 id="案例"><a href="#案例" class="headerlink" title="案例"></a>案例</h1><hr><p>爬虫</p><p>前后端分离//待完成</p><p>高亮//待完成</p>]]></content>
</entry>
<entry>
<title>Spring</title>
<link href="/2020/051618155.html"/>
<url>/2020/051618155.html</url>
<content type="html"><</p><p>Spring 框架是一个分层架构,由 7 个定义良好的模块组成。Spring 模块构建在核心容器之上,核心容器定义了创建、配置和管理 bean 的方式 .</p><p></p><p>组成 Spring 框架的每个模块(或组件)都可以单独存在,或者与其他一个或多个模块联合实现。每个模块的功能如下:</p><ul><li><strong>核心容器</strong>:核心容器提供 Spring 框架的基本功能。核心容器的主要组件是BeanFactory,它是工厂模式的实现。BeanFactory 使用<em>控制反转</em>(IOC) 模式将应用程序的配置和依赖性规范与实际的应用程序代码分开。</li><li><strong>Spring 上下文</strong>:Spring 上下文是一个配置文件,向 Spring 框架提供上下文信息。Spring 上下文包括企业服务,例如 JNDI、EJB、电子邮件、国际化、校验和调度功能。</li><li><strong>Spring AOP</strong>:通过配置管理特性,Spring AOP 模块直接将面向切面的编程功能 , 集成到了 Spring 框架中。所以,可以很容易地使 Spring 框架管理任何支持 AOP的对象。Spring AOP 模块为基于 Spring 的应用程序中的对象提供了事务管理服务。通过使用 Spring AOP,不用依赖组件,就可以将声明性事务管理集成到应用程序中。</li><li><strong>Spring DAO</strong>:JDBC DAO 抽象层提供了有意义的异常层次结构,可用该结构来管理异常处理和不同数据库供应商抛出的错误消息。异常层次结构简化了错误处理,并且极大地降低了需要编写的异常代码数量(例如打开和关闭连接)。Spring DAO 的面向 JDBC 的异常遵从通用的 DAO 异常层次结构。</li><li><strong>Spring ORM</strong>:Spring 框架插入了若干个 ORM 框架,从而提供了 ORM 的对象关系工具,其中包括 JDO、Hibernate 和 iBatis SQL Map。所有这些都遵从 Spring 的通用事务和 DAO 异常层次结构。</li><li><strong>Spring Web 模块</strong>:Web 上下文模块建立在应用程序上下文模块之上,为基于 Web 的应用程序提供了上下文。所以,Spring 框架支持与 Jakarta Struts 的集成。Web 模块还简化了处理多部分请求以及将请求参数绑定到域对象的工作。</li><li><strong>Spring MVC 框架</strong>:MVC 框架是一个全功能的构建 Web 应用程序的 MVC 实现。通过策略接口,MVC 框架变成为高度可配置的,MVC 容纳了大量视图技术,其中包括 JSP、Velocity、Tiles、iText 和 POI。</li></ul><h3 id="拓展"><a href="#拓展" class="headerlink" title="拓展"></a>拓展</h3><p><strong>Spring Boot与Spring Cloud</strong></p><ul><li>Spring Boot 是 Spring 的一套快速配置脚手架,可以基于Spring Boot 快速开发单个微服务;</li><li>Spring Cloud是基于Spring Boot实现的;</li><li>Spring Boot专注于快速、方便集成的单个微服务个体,Spring Cloud关注全局的服务治理框架;</li><li>Spring Boot使用了约束优于配置的理念,很多集成方案已经帮你选择好了,能不配置就不配置 , Spring Cloud很大的一部分是基于Spring Boot来实现,Spring Boot可以离开Spring Cloud独立使用开发项目,但是Spring Cloud离不开Spring Boot,属于依赖的关系。</li><li>SpringBoot在SpringClound中起到了承上启下的作用,如果你要学习SpringCloud必须要学习SpringBoot。</li></ul><p></p><h1 id="IOC"><a href="#IOC" class="headerlink" title="IOC"></a>IOC</h1><h3 id="IOC本质"><a href="#IOC本质" class="headerlink" title="IOC本质"></a>IOC本质</h3><p><strong>控制反转IoC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IoC的一种方法</strong>,也有人认为DI只是IoC的另一种说法。没有IoC的程序中 , 我们使用面向对象编程 , 对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方,个人认为所谓控制反转就是:获得依赖对象的方式反转了。</p><p></p><p><strong>IoC是Spring框架的核心内容</strong>,使用多种方式完美的实现了IoC,可以使用XML配置,也可以使用注解,新版本的Spring也可以零配置实现IoC。</p><p>Spring容器在初始化时先读取配置文件,根据配置文件或元数据创建与组织对象存入容器中,程序使用时再从Ioc容器中取出需要的对象。</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7KtDiaOqFy5ourlJ8FTVV2FF67dfeA6cRT7EiafNcibWyf57SGpkZ01JnpiaaicNB1ibBjGaicAvayKEWJ0A/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p>采用XML方式配置Bean的时候,Bean的定义信息是和实现分离的,而采用注解的方式可以把两者合为一体,Bean的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。</p><p><strong>控制反转是一种通过描述(XML或注解)并通过第三方去生产或获取特定对象的方式。在Spring中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependency Injection,DI)。</strong></p><h3 id="HelloSpring"><a href="#HelloSpring" class="headerlink" title="HelloSpring"></a>HelloSpring</h3><blockquote><p>导入Jar包</p></blockquote><p>注 : spring 需要导入commons-logging进行日志记录 . 我们利用maven , 他会自动下载对应的依赖项 .</p><pre><code><dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>5.1.10.RELEASE</version></dependency></code></pre><blockquote><p>编写代码</p></blockquote><p>1、编写一个Hello实体类</p><pre><code>public class Hello { private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } public void show(){ System.out.println("Hello,"+ name ); }}</code></pre><p>2、编写我们的spring文件 , 这里我们命名为beans.xml</p><pre><code><?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!--bean就是java对象 , 由Spring创建和管理--> <bean id="hello" class="com.kuang.pojo.Hello"> <property name="name" value="Spring"/> </bean></beans></code></pre><p>3、我们可以去进行测试了 .</p><pre><code>@Testpublic void test(){ //解析beans.xml文件 , 生成管理相应的Bean对象 ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); //getBean : 参数即为spring配置文件中bean的id . Hello hello = (Hello) context.getBean("hello"); hello.show();}</code></pre><blockquote><p>思考</p></blockquote><ul><li>Hello 对象是谁创建的 ? 【hello 对象是由Spring创建的</li><li>Hello 对象的属性是怎么设置的 ? hello 对象的属性是由Spring容器设置的</li></ul><p>这个过程就叫控制反转 :</p><ul><li>控制 : 谁来控制对象的创建 , 传统应用程序的对象是由程序本身控制创建的 , 使用Spring后 , 对象是由Spring来创建的</li><li>反转 : 程序本身不创建对象 , 而变成被动的接收对象 .</li></ul><p>依赖注入 : 就是利用set方法来进行注入的.</p><p> IOC是一种编程思想,由主动的编程变成被动的接收</p><p>可以通过newClassPathXmlApplicationContext去浏览一下底层源码 .</p><blockquote><p>修改案例一</p></blockquote><p>我们在案例一中, 新增一个Spring配置文件beans.xml</p><pre><code><?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean id="MysqlImpl" class="com.kuang.dao.impl.UserDaoMySqlImpl"/> <bean id="OracleImpl" class="com.kuang.dao.impl.UserDaoOracleImpl"/> <bean id="ServiceImpl" class="com.kuang.service.impl.UserServiceImpl"> <!--注意: 这里的name并不是属性 , 而是set方法后面的那部分 , 首字母小写--> <!--引用另外一个bean , 不是用value 而是用 ref--> <property name="userDao" ref="OracleImpl"/> </bean></beans></code></pre><p>测试!</p><pre><code>@Testpublic void test2(){ ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); UserServiceImpl serviceImpl = (UserServiceImpl) context.getBean("ServiceImpl"); serviceImpl.getUser();}</code></pre><p>OK , 到了现在 , 我们彻底不用再程序中去改动了 , 要实现不同的操作 , 只需要在xml配置文件中进行修改 , 所谓的IoC,一句话搞定 : 对象由Spring 来创建 , 管理 , 装配 ! </p><h3 id="IOC创建对象方式"><a href="#IOC创建对象方式" class="headerlink" title="IOC创建对象方式"></a>IOC创建对象方式</h3><blockquote><p>通过无参构造方法来创建</p></blockquote><p>1、User.java</p><pre><code>public class User { private String name; public User() { System.out.println("user无参构造方法"); } public void setName(String name) { this.name = name; } public void show(){ System.out.println("name="+ name ); }}</code></pre><p>2、beans.xml</p><pre><code><?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean id="user" class="com.kuang.pojo.User"> <property name="name" value="kuangshen"/> </bean></beans></code></pre><p>3、测试类</p><pre><code>@Testpublic void test(){ ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); //在执行getBean的时候, user已经创建好了 , 通过无参构造 User user = (User) context.getBean("user"); //调用对象的方法 . user.show();}</code></pre><p>结果可以发现,在调用show方法之前,User对象已经通过无参构造初始化了!</p><blockquote><p>通过有参构造方法来创建</p></blockquote><p>1、UserT . java</p><pre><code>public class UserT { private String name; public UserT(String name) { this.name = name; } public void setName(String name) { this.name = name; } public void show(){ System.out.println("name="+ name ); }}</code></pre><p>2、beans.xml 有三种方式编写</p><pre><code><!-- 第一种根据index参数下标设置 --><bean id="userT" class="com.kuang.pojo.UserT"> <!-- index指构造方法 , 下标从0开始 --> <constructor-arg index="0" value="kuangshen2"/></bean><!-- 第二种根据参数名字设置 --><bean id="userT" class="com.kuang.pojo.UserT"> <!-- name指参数名 --> <constructor-arg name="name" value="kuangshen2"/></bean><!-- 第三种根据参数类型设置 --><bean id="userT" class="com.kuang.pojo.UserT"> <constructor-arg type="java.lang.String" value="kuangshen2"/></bean></code></pre><p>3、测试</p><pre><code>@Testpublic void testT(){ ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); UserT user = (UserT) context.getBean("userT"); user.show();}</code></pre><p>结论:在配置文件加载的时候。其中管理的对象都已经初始化了!</p><h1 id="Spring配置"><a href="#Spring配置" class="headerlink" title="Spring配置"></a>Spring配置</h1><h3 id="别名"><a href="#别名" class="headerlink" title="别名"></a>别名</h3><p>alias 设置别名 , 为bean设置别名 , 可以设置多个别名</p><pre><code><!--设置别名:在获取Bean的时候可以使用别名获取--><alias name="userT" alias="userNew"/></code></pre><h3 id="bean配置"><a href="#bean配置" class="headerlink" title="bean配置"></a>bean配置</h3><pre><code><!--bean就是java对象,由Spring创建和管理--><!-- id 是bean的标识符,要唯一,如果没有配置id,name就是默认标识符 如果配置id,又配置了name,那么name是别名 name可以设置多个别名,可以用逗号,分号,空格隔开 如果不配置id和name,可以根据applicationContext.getBean(.class)获取对象;class是bean的全限定名=包名+类名--><bean id="hello" name="hello2 h2,h3;h4" class="com.kuang.pojo.Hello"> <property name="name" value="Spring"/></bean></code></pre><h3 id="import"><a href="#import" class="headerlink" title="import"></a>import</h3><p>团队的合作通过import来实现 .</p><pre><code><import resource="{path}/beans.xml"/></code></pre><h1 id="依赖注入"><a href="#依赖注入" class="headerlink" title="依赖注入"></a>依赖注入</h1><h3 id="概念"><a href="#概念" class="headerlink" title="概念"></a>概念</h3><ul><li>依赖注入(Dependency Injection,DI)。</li><li>依赖 : 指Bean对象的创建依赖于容器 . </li><li>注入 : 指Bean对象所依赖的资源 , 由容器来设置和装配 .</li></ul><h3 id="构造器注入"><a href="#构造器注入" class="headerlink" title="构造器注入"></a>构造器注入</h3><p>我们在之前的案例已经讲过了</p><h3 id="Set-注入"><a href="#Set-注入" class="headerlink" title="Set 注入"></a>Set 注入</h3><p>要求被注入的属性 , 必须有set方法 , set方法的方法名由set + 属性首字母大写 , 如果属性是boolean类型 , 没有set方法 , 是 is .</p><p>测试pojo类 :</p><p>Address.java</p><pre><code> public class Address { private String address; public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } }</code></pre><p>Student.java</p><pre><code> package com.kuang.pojo; import java.util.List; import java.util.Map; import java.util.Properties; import java.util.Set; public class Student { private String name; private Address address; private String[] books; private List<String> hobbys; private Map<String,String> card; private Set<String> games; private String wife; private Properties info; public void setName(String name) { this.name = name; } public void setAddress(Address address) { this.address = address; } public void setBooks(String[] books) { this.books = books; } public void setHobbys(List<String> hobbys) { this.hobbys = hobbys; } public void setCard(Map<String, String> card) { this.card = card; } public void setGames(Set<String> games) { this.games = games; } public void setWife(String wife) { this.wife = wife; } public void setInfo(Properties info) { this.info = info; } public void show(){ System.out.println("name="+ name + ",address="+ address.getAddress() + ",books=" ); for (String book:books){ System.out.print("<<"+book+">>\t"); } System.out.println("\n爱好:"+hobbys); System.out.println("card:"+card); System.out.println("games:"+games); System.out.println("wife:"+wife); System.out.println("info:"+info); } }</code></pre><p>1、<strong>常量注入</strong></p><pre><code> <bean id="student" class="com.kuang.pojo.Student"> <property name="name" value="小明"/> </bean></code></pre><p>测试:</p><pre><code> @Test public void test01(){ ApplicationContext context = newClassPathXmlApplicationContext("applicationContext.xml"); Student student = (Student) context.getBean("student"); System.out.println(student.getName()); }</code></pre><p>2、<strong>Bean注入</strong> </p><p>注意点:这里的值是一个引用,ref</p><pre><code> <bean id="addr" class="com.kuang.pojo.Address"> <property name="address" value="重庆"/> </bean> <bean id="student" class="com.kuang.pojo.Student"> <property name="name" value="小明"/> <property name="address" ref="addr"/> </bean></code></pre><p>3、<strong>数组注入</strong></p><pre><code> <bean id="student" class="com.kuang.pojo.Student"> <property name="name" value="小明"/> <property name="address" ref="addr"/> <property name="books"> <array> <value>西游记</value> <value>红楼梦</value> <value>水浒传</value> </array> </property> </bean></code></pre><p>4、<strong>List注入</strong></p><pre><code> <property name="hobbys"> <list> <value>听歌</value> <value>看电影</value> <value>爬山</value> </list> </property></code></pre><p>5、<strong>Map注入</strong></p><pre><code> <property name="card"> <map> <entry key="中国邮政" value="456456456465456"/> <entry key="建设" value="1456682255511"/> </map> </property></code></pre><p>6、<strong>set注入</strong></p><pre><code> <property name="games"> <set> <value>LOL</value> <value>BOB</value> <value>COC</value> </set> </property></code></pre><p>7、<strong>Null注入</strong></p><pre><code> <property name="wife"><null/></property></code></pre><p>8、<strong>Properties注入</strong></p><pre><code> <property name="info"> <props> <prop key="学号">20190604</prop> <prop key="性别">男</prop> <prop key="姓名">小明</prop> </props> </property></code></pre><p>测试结果:</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7K5cyS8ZRTpajtSInicNHbMYGHEFnrCA8Jyr6ian5MWrUHtnKBpYYTTtysbp5UPYKQxSiaUHJibPKlicuQ/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p><strong>p命名和c命名注入</strong></p><p>User.java :【注意:这里没有有参构造器!】</p><pre><code> public class User { private String name; private int age; public void setName(String name) { this.name = name; } public void setAge(int age) { this.age = age; } @Override public String toString() { return "User{" + "name='" + name + '\'' + ", age=" + age + '}'; } }</code></pre><p>1、P命名空间注入 : 需要在头文件中加入约束文件</p><pre><code> 导入约束 : xmlns:p="http://www.springframework.org/schema/p" <!--P(属性: properties)命名空间 , 属性依然要设置set方法--> <bean id="user" class="com.kuang.pojo.User" p:name="狂神" p:age="18"/></code></pre><p>2、c 命名空间注入 : 需要在头文件中加入约束文件</p><pre><code> 导入约束 : xmlns:c="http://www.springframework.org/schema/c" <!--C(构造: Constructor)命名空间 , 属性依然要设置set方法--> <bean id="user" class="com.kuang.pojo.User" c:name="狂神" c:age="18"/></code></pre><p>发现问题:爆红了,刚才我们没有写有参构造!</p><p>解决:把有参构造器加上,这里也能知道,c 就是所谓的构造器注入!</p><p>测试代码:</p><pre><code> @Test public void test02(){ ApplicationContext context = newClassPathXmlApplicationContext("applicationContext.xml"); User user = (User) context.getBean("user"); System.out.println(user); }</code></pre><h1 id="Bean的作用域"><a href="#Bean的作用域" class="headerlink" title="Bean的作用域"></a>Bean的作用域</h1><p>在Spring中,那些组成应用程序的主体及由Spring IoC容器所管理的对象,被称之为bean。简单地讲,bean就是由IoC容器初始化、装配及管理的对象 .</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7K5cyS8ZRTpajtSInicNHbMYfmmAQF8hrnicY49FRXEkR5xkxD5A4H5pVUia3mFhrDdh4gBt183EiaFaQ/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p>几种作用域中,request、session作用域仅在基于web的应用中使用(不必关心你所采用的是什么web应用框架),只能用在基于web的Spring ApplicationContext环境。</p><h4 id="Singleton"><a href="#Singleton" class="headerlink" title="Singleton"></a>Singleton</h4><p>当一个bean的作用域为Singleton,那么Spring IoC容器中只会存在一个共享的bean实例,并且所有对bean的请求,只要id与该bean定义相匹配,则只会返回bean的同一实例。Singleton是单例类型,就是在创建起容器时就同时自动创建了一个bean的对象,不管你是否使用,他都存在了,每次获取到的对象都是同一个对象。注意,Singleton作用域是Spring中的缺省作用域。要在XML中将bean定义成singleton,可以这样配置:</p><pre><code> <bean id="ServiceImpl" class="cn.csdn.service.ServiceImpl" scope="singleton"></code></pre><p>测试:</p><pre><code> @Test public void test03(){ ApplicationContext context = newClassPathXmlApplicationContext("applicationContext.xml"); User user = (User) context.getBean("user"); User user2 = (User) context.getBean("user"); System.out.println(user==user2); }</code></pre><h4 id="Prototype"><a href="#Prototype" class="headerlink" title="Prototype"></a>Prototype</h4><p>当一个bean的作用域为Prototype,表示一个bean定义对应多个对象实例。Prototype作用域的bean会导致在每次对该bean请求(将其注入到另一个bean中,或者以程序的方式调用容器的getBean()方法)时都会创建一个新的bean实例。Prototype是原型类型,它在我们创建容器的时候并没有实例化,而是当我们获取bean的时候才会去创建一个对象,而且我们每次获取到的对象都不是同一个对象。根据经验,对有状态的bean应该使用prototype作用域,而对无状态的bean则应该使用singleton作用域。在XML中将bean定义成prototype,可以这样配置:</p><pre><code> <bean id="account" class="com.foo.DefaultAccount" scope="prototype"/> 或者 <bean id="account" class="com.foo.DefaultAccount" singleton="false"/></code></pre><h4 id="Request"><a href="#Request" class="headerlink" title="Request"></a>Request</h4><p>当一个bean的作用域为Request,表示在一次HTTP请求中,一个bean定义对应一个实例;即每个HTTP请求都会有各自的bean实例,它们依据某个bean定义创建而成。该作用域仅在基于web的Spring ApplicationContext情形下有效。考虑下面bean定义:</p><pre><code> <bean id="loginAction" class=cn.csdn.LoginAction" scope="request"/></code></pre><p>针对每次HTTP请求,Spring容器会根据loginAction bean的定义创建一个全新的LoginAction bean实例,且该loginAction bean实例仅在当前HTTP request内有效,因此可以根据需要放心的更改所建实例的内部状态,而其他请求中根据loginAction bean定义创建的实例,将不会看到这些特定于某个请求的状态变化。当处理请求结束,request作用域的bean实例将被销毁。</p><h4 id="Session"><a href="#Session" class="headerlink" title="Session"></a>Session</h4><p>当一个bean的作用域为Session,表示在一个HTTP Session中,一个bean定义对应一个实例。该作用域仅在基于web的Spring ApplicationContext情形下有效。考虑下面bean定义:</p><pre><code> <bean id="userPreferences" class="com.foo.UserPreferences" scope="session"/></code></pre><p>针对某个HTTP Session,Spring容器会根据userPreferences bean定义创建一个全新的userPreferences bean实例,且该userPreferences bean仅在当前HTTP Session内有效。与request作用域一样,可以根据需要放心的更改所创建实例的内部状态,而别的HTTP Session中根据userPreferences创建的实例,将不会看到这些特定于某个HTTP Session的状态变化。当HTTP Session最终被废弃的时候,在该HTTP Session作用域内的bean也会被废弃掉。</p><h1 id="自动装配"><a href="#自动装配" class="headerlink" title="自动装配"></a>自动装配</h1><h3 id="自动装配说明"><a href="#自动装配说明" class="headerlink" title="自动装配说明"></a>自动装配说明</h3><ul><li>自动装配是使用spring满足bean依赖的一种方法</li><li>spring会在应用上下文中为某个bean寻找其依赖的bean。</li></ul><p>Spring中bean有三种装配机制,分别是:</p><ol><li>在xml中显式配置;</li><li>在java中显式配置;</li><li>隐式的bean发现机制和自动装配。</li></ol><p>这里我们主要讲第三种:自动化的装配bean。</p><p>Spring的自动装配需要从两个角度来实现,或者说是两个操作:</p><ol><li>组件扫描(component scanning):spring会自动发现应用上下文中所创建的bean;</li><li>自动装配(autowiring):spring自动满足bean之间的依赖,也就是我们说的IoC/DI;</li></ol><p>组件扫描和自动装配组合发挥巨大威力,使得显示的配置降低到最少。</p><p><strong>推荐不使用自动装配xml配置 , 而使用注解 .</strong></p><blockquote><p>测试环境搭建</p></blockquote><p>1、新建一个项目</p><p>2、新建两个实体类,Cat Dog 都有一个叫的方法</p><pre><code>public class Cat { public void shout() { System.out.println("miao~"); }}public class Dog { public void shout() { System.out.println("wang~"); }}</code></pre><p>3、新建一个用户类 User</p><pre><code>public class User { private Cat cat; private Dog dog; private String str;}</code></pre><p>4、编写Spring配置文件</p><pre><code><?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean id="dog" class="com.kuang.pojo.Dog"/> <bean id="cat" class="com.kuang.pojo.Cat"/> <bean id="user" class="com.kuang.pojo.User"> <property name="cat" ref="cat"/> <property name="dog" ref="dog"/> <property name="str" value="qinjiang"/> </bean></beans></code></pre><p>5、测试</p><pre><code>public class MyTest { @Test public void testMethodAutowire() { ApplicationContext context = newClassPathXmlApplicationContext("beans.xml"); User user = (User) context.getBean("user"); user.getCat().shout(); user.getDog().shout(); }}</code></pre><p>结果正常输出,环境OK</p><h3 id="byName"><a href="#byName" class="headerlink" title="byName"></a>byName</h3><p><strong>autowire byName (按名称自动装配)</strong></p><p>由于在手动配置xml过程中,常常发生字母缺漏和大小写等错误,而无法对其进行检查,使得开发效率降低。</p><p> 采用自动装配将避免这些错误,并且使配置简单化。</p><p>测试:</p><p>1、修改bean配置,增加一个属性 autowire=”byName”</p><pre><code><bean id="user" class="com.kuang.pojo.User" autowire="byName"> <property name="str" value="qinjiang"/></bean></code></pre><p>2、再次测试,结果依旧成功输出!</p><p>3、我们将 cat 的bean id修改为 catXXX</p><p>4、再次测试, 执行时报空指针java.lang.NullPointerException。因为按byName规则找不对应set方法,真正的setCat就没执行,对象就没有初始化,所以调用时就会报空指针错误。</p><p><strong>小结:</strong></p><p>当一个bean节点带有 autowire byName的属性时。</p><ol><li><p>将查找其类中所有的set方法名,例如setCat,获得将set去掉并且首字母小写的字符串,即cat。</p></li><li><p>去spring容器中寻找是否有此字符串名称id的对象。</p></li><li><p>如果有,就取出注入;如果没有,就报空指针异常。</p></li></ol><h3 id="byType"><a href="#byType" class="headerlink" title="byType"></a>byType</h3><p><strong>autowire byType (按类型自动装配)</strong></p><p>使用autowire byType首先需要保证:同一类型的对象,在spring容器中唯一。如果不唯一,会报不唯一的异常。</p><pre><code>NoUniqueBeanDefinitionException</code></pre><p>测试:</p><p>1、将user的bean配置修改一下 : autowire=”byType”</p><p>2、测试,正常输出</p><p>3、在注册一个cat 的bean对象!</p><pre><code><bean id="dog" class="com.kuang.pojo.Dog"/><bean id="cat" class="com.kuang.pojo.Cat"/><bean id="cat2" class="com.kuang.pojo.Cat"/><bean id="user" class="com.kuang.pojo.User" autowire="byType"> <property name="str" value="qinjiang"/></bean></code></pre><p>4、测试,报错:NoUniqueBeanDefinitionException</p><p>5、删掉cat2,将cat的bean名称改掉!测试!因为是按类型装配,所以并不会报异常,也不影响最后的结果。甚至将id属性去掉,也不影响结果。</p><p>这就是按照类型自动装配!</p><p>使用注解</p><h3 id="使用注解"><a href="#使用注解" class="headerlink" title="使用注解"></a>使用注解</h3><p>jdk1.5开始支持注解,spring2.5开始全面支持注解。</p><p>准备工作:利用注解的方式注入属性。</p><p>1、在spring配置文件中引入context文件头</p><pre><code>xmlns:context="http://www.springframework.org/schema/context"http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd</code></pre><p>2、开启属性注解支持!</p><pre><code><context:annotation-config/></code></pre><h4 id=""><a href="#" class="headerlink" title=""></a></h4><h4 id="Autowired"><a href="#Autowired" class="headerlink" title="@Autowired"></a>@Autowired</h4><ul><li>@Autowired是按类型自动转配的,不支持id匹配。</li><li>需要导入 spring-aop的包!</li></ul><p>测试:</p><p>1、将User类中的set方法去掉,使用@Autowired注解</p><pre><code>public class User { @Autowired private Cat cat; @Autowired private Dog dog; private String str; public Cat getCat() { return cat; } public Dog getDog() { return dog; } public String getStr() { return str; }}</code></pre><p>2、此时配置文件内容</p><pre><code><context:annotation-config/><bean id="dog" class="com.kuang.pojo.Dog"/><bean id="cat" class="com.kuang.pojo.Cat"/><bean id="user" class="com.kuang.pojo.User"/></code></pre><p>3、测试,成功输出结果!</p><p>【小狂神科普时间】</p><p>@Autowired(required=false) 说明:false,对象可以为null;true,对象必须存对象,不能为null。</p><pre><code>//如果允许对象为null,设置required = false,默认为true@Autowired(required = false)private Cat cat;</code></pre><h4 id="-1"><a href="#-1" class="headerlink" title=""></a></h4><h4 id="Qualifier"><a href="#Qualifier" class="headerlink" title="@Qualifier"></a>@Qualifier</h4><ul><li>@Autowired是根据类型自动装配的,加上@Qualifier则可以根据byName的方式自动装配</li><li>@Qualifier不能单独使用。</li></ul><p>测试实验步骤:</p><p>1、配置文件修改内容,保证类型存在对象。且名字不为类的默认名字!</p><pre><code><bean id="dog1" class="com.kuang.pojo.Dog"/><bean id="dog2" class="com.kuang.pojo.Dog"/><bean id="cat1" class="com.kuang.pojo.Cat"/><bean id="cat2" class="com.kuang.pojo.Cat"/></code></pre><p>2、没有加Qualifier测试,直接报错</p><p>3、在属性上添加Qualifier注解</p><pre><code>@Autowired@Qualifier(value = "cat2")private Cat cat;@Autowired@Qualifier(value = "dog2")private Dog dog;</code></pre><p>测试,成功输出!</p><h4 id="-2"><a href="#-2" class="headerlink" title=""></a></h4><h4 id="Resource"><a href="#Resource" class="headerlink" title="@Resource"></a>@Resource</h4><ul><li>@Resource如有指定的name属性,先按该属性进行byName方式查找装配;</li><li>其次再进行默认的byName方式进行装配;</li><li>如果以上都不成功,则按byType的方式自动装配。</li><li>都不成功,则报异常。</li></ul><p>实体类:</p><pre><code>public class User { //如果允许对象为null,设置required = false,默认为true @Resource(name = "cat2") private Cat cat; @Resource private Dog dog; private String str;}</code></pre><p>beans.xml</p><pre><code><bean id="dog" class="com.kuang.pojo.Dog"/><bean id="cat1" class="com.kuang.pojo.Cat"/><bean id="cat2" class="com.kuang.pojo.Cat"/><bean id="user" class="com.kuang.pojo.User"/></code></pre><p>测试:结果OK</p><p>配置文件2:beans.xml , 删掉cat2</p><pre><code><bean id="dog" class="com.kuang.pojo.Dog"/><bean id="cat1" class="com.kuang.pojo.Cat"/></code></pre><p>实体类上只保留注解</p><pre><code>@Resourceprivate Cat cat;@Resourceprivate Dog dog;</code></pre><p>结果:OK</p><p>结论:先进行byName查找,失败;再进行byType查找,成功。</p><h3 id="-3"><a href="#-3" class="headerlink" title=""></a></h3><h3 id="小结"><a href="#小结" class="headerlink" title="小结"></a>小结</h3><p>@Autowired与@Resource异同:</p><p>1、@Autowired与@Resource都可以用来装配bean。都可以写在字段上,或写在setter方法上。</p><p>2、@Autowired默认按类型装配(属于spring规范),默认情况下必须要求依赖对象必须存在,如果要允许null 值,可以设置它的required属性为false,如:@Autowired(required=false) ,如果我们想使用名称装配可以结合@Qualifier注解进行使用</p><p>3、@Resource(属于J2EE复返),默认按照名称进行装配,名称可以通过name属性进行指定。如果没有指定name属性,当注解写在字段上时,默认取字段名进行按照名称查找,如果注解写在setter方法上默认取属性名进行装配。当找不到与名称匹配的bean时才按照类型进行装配。但是需要注意的是,如果name属性一旦指定,就只会按照名称进行装配。</p><p>它们的作用相同都是用注解方式注入对象,但执行顺序不同。@Autowired先byType,@Resource先byName。</p><h1 id="注解开发"><a href="#注解开发" class="headerlink" title="注解开发"></a>注解开发</h1><p>在spring4之后,想要使用注解形式,必须得要引入aop的包</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7IzypAUHDfDEujP4ry6cHxWTvAS6qRS0qrmMCic3QqF9icGNcPj8IZwNo3R9VEgpAgWHrStBN1ya6Tg/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p>在配置文件当中,还得要引入一个context约束</p><pre><code><?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"></beans></code></pre><h3 id="Bean的实现"><a href="#Bean的实现" class="headerlink" title="Bean的实现"></a>Bean的实现</h3><p>我们之前都是使用 bean 的标签进行bean注入,但是实际开发中,我们一般都会使用注解!</p><p>1、配置扫描哪些包下的注解</p><pre><code><!--指定注解扫描包--><context:component-scan base-package="com.kuang.pojo"/></code></pre><p>2、在指定包下编写类,增加注解</p><pre><code>@Component("user")// 相当于配置文件中 <bean id="user" class="当前注解的类"/>public class User { public String name = "秦疆";}</code></pre><p>3、测试</p><pre><code>@Testpublic void test(){ ApplicationContext applicationContext = new ClassPathXmlApplicationContext("beans.xml"); User user = (User) applicationContext.getBean("user"); System.out.println(user.name);}</code></pre><h3 id="属性注入"><a href="#属性注入" class="headerlink" title="属性注入"></a>属性注入</h3><p>使用注解注入属性</p><p>1、可以不用提供set方法,直接在直接名上添加@value(“值”)</p><pre><code>@Component("user")// 相当于配置文件中 <bean id="user" class="当前注解的类"/>public class User { @Value("秦疆") // 相当于配置文件中 <property name="name" value="秦疆"/> public String name;}</code></pre><p>2、如果提供了set方法,在set方法上添加@value(“值”);</p><pre><code>@Component("user")public class User { public String name; @Value("秦疆") public void setName(String name) { this.name = name; }}</code></pre><h3 id="衍生注解"><a href="#衍生注解" class="headerlink" title="衍生注解"></a>衍生注解</h3><p>我们这些注解,就是替代了在配置文件当中配置步骤而已!更加的方便快捷!</p><p><strong>@Component三个衍生注解</strong></p><p>为了更好的进行分层,Spring可以使用其它三个注解,功能一样,目前使用哪一个功能都一样。</p><ul><li>@Controller:web层</li><li>@Service:service层</li><li>@Repository:dao层</li></ul><p>写上这些注解,就相当于将这个类交给Spring管理装配了!</p><blockquote><p>自动装配注解</p></blockquote><p>在Bean的自动装配已经讲过了,可以回顾!</p><blockquote><p>作用域</p></blockquote><p>@scope</p><ul><li>singleton:默认的,Spring会采用单例模式创建这个对象。关闭工厂 ,所有的对象都会销毁。</li><li>prototype:多例模式。关闭工厂 ,所有的对象不会销毁。内部的垃圾回收机制会回收</li></ul><pre><code>@Controller("user")@Scope("prototype")public class User { @Value("秦疆") public String name;}</code></pre><h3 id="小结-1"><a href="#小结-1" class="headerlink" title="小结"></a>小结</h3><p><strong>XML与注解比较</strong></p><ul><li>XML可以适用任何场景 ,结构清晰,维护方便</li><li>注解不是自己提供的类使用不了,开发简单方便</li></ul><p><strong>xml与注解整合开发</strong> :推荐最佳实践</p><ul><li>xml管理Bean</li><li>注解完成属性注入</li><li>使用过程中, 可以不用扫描,扫描是为了类上的注解</li></ul><pre><code><context:annotation-config/> </code></pre><p>作用:</p><ul><li><p>进行注解驱动注册,从而使注解生效</p></li><li><p>用于激活那些已经在spring容器里注册过的bean上面的注解,也就是显示的向Spring注册</p></li><li><p>如果不扫描包,就需要手动配置bean</p></li><li><p>如果不加注解驱动,则注入的值为null!</p></li></ul><h3 id="基于Java类进行配置"><a href="#基于Java类进行配置" class="headerlink" title="基于Java类进行配置"></a>基于Java类进行配置</h3><p>JavaConfig 原来是 Spring 的一个子项目,它通过 Java 类的方式提供 Bean 的定义信息,在 Spring4 的版本, JavaConfig 已正式成为 Spring4 的核心功能 。</p><p>测试:</p><p>1、编写一个实体类,Dog</p><pre><code>@Component //将这个类标注为Spring的一个组件,放到容器中!public class Dog { public String name = "dog";}</code></pre><p>2、新建一个config配置包,编写一个MyConfig配置类</p><pre><code>@Configuration //代表这是一个配置类public class MyConfig { @Bean //通过方法注册一个bean,这里的返回值就Bean的类型,方法名就是bean的id! public Dog dog(){ return new Dog(); }}</code></pre><p>3、测试</p><pre><code>@Testpublic void test2(){ ApplicationContext applicationContext = new AnnotationConfigApplicationContext(MyConfig.class); Dog dog = (Dog) applicationContext.getBean("dog"); System.out.println(dog.name);}</code></pre><p>4、成功输出结果!</p><p><strong>导入其他配置如何做呢?</strong></p><p>1、我们再编写一个配置类!</p><pre><code>@Configuration //代表这是一个配置类public class MyConfig2 {}</code></pre><p>2、在之前的配置类中我们来选择导入这个配置类</p><pre><code>@Configuration@Import(MyConfig2.class) //导入合并其他配置类,类似于配置文件中的 inculde 标签public class MyConfig { @Bean public Dog dog(){ return new Dog(); }}</code></pre><p>关于这种Java类的配置方式,我们在之后的SpringBoot 和 SpringCloud中还会大量看到,我们需要知道这些注解的作用即可!</p><h1 id="代理模式"><a href="#代理模式" class="headerlink" title="代理模式"></a>代理模式</h1><hr><p>为什么要学习代理模式,因为AOP的底层机制就是动态代理!</p><p>代理模式:</p><ul><li>静态代理</li><li>动态代理</li></ul><p>学习aop之前 , 我们要先了解一下代理模式!</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7LoeicP1O2nfyA6H0XPa9jMLRnfS7LpO0Iic7fEEPFCgRs0ggNXCwf6IKo1tibjPmNSTEYeII5ro7YLQ/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><h2 id="静态代理"><a href="#静态代理" class="headerlink" title="静态代理"></a>静态代理</h2><p><strong>静态代理角色分析</strong></p><ul><li><p>抽象角色 : 一般使用接口或者抽象类来实现</p></li><li><p>真实角色 : 被代理的角色</p></li><li><p>代理角色 : 代理真实角色 ; 代理真实角色后 , 一般会做一些附属的操作 .</p></li><li><p>客户 : 使用代理角色来进行一些操作 .</p></li></ul><p><strong>代码实现</strong></p><p>Rent . java 即抽象角色</p><pre><code>//抽象角色:租房public interface Rent { public void rent();}</code></pre><p>Host . java 即真实角色</p><pre><code>//真实角色: 房东,房东要出租房子public class Host implements Rent{ public void rent() { System.out.println("房屋出租"); }}</code></pre><p>Proxy . java 即代理角色</p><pre><code>//代理角色:中介public class Proxy implements Rent { private Host host; public Proxy() { } public Proxy(Host host) { this.host = host; } //租房 public void rent(){ seeHouse(); host.rent(); fare(); } //看房 public void seeHouse(){ System.out.println("带房客看房"); } //收中介费 public void fare(){ System.out.println("收中介费"); }}</code></pre><p>Client . java 即客户</p><pre><code>//客户类,一般客户都会去找代理!public class Client { public static void main(String[] args) { //房东要租房 Host host = new Host(); //中介帮助房东 Proxy proxy = new Proxy(host); //你去找中介! proxy.rent(); }}</code></pre><p>分析:在这个过程中,你直接接触的就是中介,就如同现实生活中的样子,你看不到房东,但是你依旧租到了房东的房子通过代理,这就是所谓的代理模式,程序源自于生活,所以学编程的人,一般能够更加抽象的看待生活中发生的事情。</p><p><strong>静态代理的好处:</strong></p><ul><li>可以使得我们的真实角色更加纯粹 . 不再去关注一些公共的事情 .</li><li>公共的业务由代理来完成 . 实现了业务的分工 ,</li><li>公共业务发生扩展时变得更加集中和方便 .</li></ul><p>缺点 :</p><ul><li>类多了 , 多了代理类 , 工作量变大了 . 开发效率降低 .</li></ul><p>我们想要静态代理的好处,又不想要静态代理的缺点,所以 , 就有了动态代理 !</p><h3 id="静态代理再理解"><a href="#静态代理再理解" class="headerlink" title="静态代理再理解"></a>静态代理再理解</h3><p>同学们练习完毕后,我们再来举一个例子,巩固大家的学习!</p><p>练习步骤:</p><p>1、创建一个抽象角色,比如咋们平时做的用户业务,抽象起来就是增删改查!</p><pre><code>//抽象角色:增删改查业务public interface UserService { void add(); void delete(); void update(); void query();}</code></pre><p>2、我们需要一个真实对象来完成这些增删改查操作</p><pre><code>//真实对象,完成增删改查操作的人public class UserServiceImpl implements UserService { public void add() { System.out.println("增加了一个用户"); } public void delete() { System.out.println("删除了一个用户"); } public void update() { System.out.println("更新了一个用户"); } public void query() { System.out.println("查询了一个用户"); }}</code></pre><p>3、需求来了,现在我们需要增加一个日志功能,怎么实现!</p><ul><li>思路1 :在实现类上增加代码 【麻烦!】</li><li>思路2:使用代理来做,能够不改变原来的业务情况下,实现此功能就是最好的了!</li></ul><p>4、设置一个代理类来处理日志!代理角色</p><pre><code>//代理角色,在这里面增加日志的实现public class UserServiceProxy implements UserService { private UserServiceImpl userService; public void setUserService(UserServiceImpl userService) { this.userService = userService; } public void add() { log("add"); userService.add(); } public void delete() { log("delete"); userService.delete(); } public void update() { log("update"); userService.update(); } public void query() { log("query"); userService.query(); } public void log(String msg){ System.out.println("执行了"+msg+"方法"); }}</code></pre><p>5、测试访问类:</p><pre><code>public class Client { public static void main(String[] args) { //真实业务 UserServiceImpl userService = new UserServiceImpl(); //代理类 UserServiceProxy proxy = new UserServiceProxy(); //使用代理类实现日志功能! proxy.setUserService(userService); proxy.add(); }}</code></pre><p>OK,到了现在代理模式大家应该都没有什么问题了,重点大家需要理解其中的思想;</p><p>我们在不改变原来的代码的情况下,实现了对原有功能的增强,这是AOP中最核心的思想</p><p>聊聊AOP:纵向开发,横向开发</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7LoeicP1O2nfyA6H0XPa9jMLJqcgicA5aEKtxYibgLPicNfDwicKIn9NlFl86rriaVRicKnEXlPNiacbHiaLibw/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><h2 id="动态代理"><a href="#动态代理" class="headerlink" title="动态代理"></a>动态代理</h2><ul><li><p>动态代理的角色和静态代理的一样 .</p></li><li><p>动态代理的代理类是动态生成的 . 静态代理的代理类是我们提前写好的</p></li><li><p>动态代理分为两类 : 一类是基于接口动态代理 , 一类是基于类的动态代理</p></li><li><ul><li>基于接口的动态代理—-JDK动态代理</li><li>基于类的动态代理–cglib</li><li>现在用的比较多的是 javasist 来生成动态代理 . 百度一下javasist</li><li>我们这里使用JDK的原生代码来实现,其余的道理都是一样的!、</li></ul></li></ul><p><strong>JDK的动态代理需要了解两个类</strong></p><p>核心 : InvocationHandler 和 Proxy , 打开JDK帮助文档看看</p><p>【InvocationHandler:调用处理程序】</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7LoeicP1O2nfyA6H0XPa9jMLxvyvZMwn9gIEibuxjFwE3enJ4TgKO5PXxM5BPr6Bh7GQwExLvst4AsQ/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><pre><code>Object invoke(Object proxy, 方法 method, Object[] args);//参数//proxy - 调用该方法的代理实例//method -所述方法对应于调用代理实例上的接口方法的实例。方法对象的声明类将是该方法声明的接口,它可以是代理类继承该方法的代理接口的超级接口。//args -包含的方法调用传递代理实例的参数值的对象的阵列,或null如果接口方法没有参数。原始类型的参数包含在适当的原始包装器类的实例中,例如java.lang.Integer或java.lang.Boolean 。</code></pre><p>【Proxy : 代理】</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7LoeicP1O2nfyA6H0XPa9jMLficZiaPU0h9wdeDicTMgBHemVvIdYTsE712DhkDfg0pdRg169oG5FHTmw/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7LoeicP1O2nfyA6H0XPa9jMLCIv9ibKb4c9KjmZNNbsDbZojUy0aB1lS3ibqa1SJaBzkK7KneicEX43Zw/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7LoeicP1O2nfyA6H0XPa9jML394CqGFmCP1nUlaU9mdLk19o1qIzjicTgDiaPz7ibR371jAo3uNNQ8Qgw/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><pre><code>//生成代理类public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(), rent.getClass().getInterfaces(),this);}</code></pre><p><strong>代码实现</strong> </p><p>抽象角色和真实角色和之前的一样!</p><p>Rent . java 即抽象角色</p><pre><code>//抽象角色:租房public interface Rent { public void rent();}</code></pre><p>Host . java 即真实角色</p><pre><code>//真实角色: 房东,房东要出租房子public class Host implements Rent{ public void rent() { System.out.println("房屋出租"); }}</code></pre><p>ProxyInvocationHandler. java 即代理角色</p><pre><code>public class ProxyInvocationHandler implements InvocationHandler { private Rent rent; public void setRent(Rent rent) { this.rent = rent; } //生成代理类,重点是第二个参数,获取要代理的抽象角色!之前都是一个角色,现在可以代理一类角色 public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(), rent.getClass().getInterfaces(),this); } // proxy : 代理类 method : 代理类的调用处理程序的方法对象. // 处理代理实例上的方法调用并返回结果 @Override public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable { seeHouse(); //核心:本质利用反射实现! Object result = method.invoke(rent, args); fare(); return result; } //看房 public void seeHouse(){ System.out.println("带房客看房"); } //收中介费 public void fare(){ System.out.println("收中介费"); }}</code></pre><p>Client . java</p><pre><code>//租客public class Client { public static void main(String[] args) { //真实角色 Host host = new Host(); //代理实例的调用处理程序 ProxyInvocationHandler pih = new ProxyInvocationHandler(); pih.setRent(host); //将真实角色放置进去! Rent proxy = (Rent)pih.getProxy(); //动态生成对应的代理类! proxy.rent(); }}</code></pre><p>核心:<strong>一个动态代理 , 一般代理某一类业务 , 一个动态代理可以代理多个类,代理的是接口!、</strong></p><h3 id="深化理解"><a href="#深化理解" class="headerlink" title="深化理解"></a>深化理解</h3><p>我们来使用动态代理实现代理我们后面写的UserService!</p><p>我们也可以编写一个通用的动态代理实现的类!所有的代理对象设置为Object即可!</p><pre><code>public class ProxyInvocationHandler implements InvocationHandler { private Object target; public void setTarget(Object target) { this.target = target; } //生成代理类 public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(), target.getClass().getInterfaces(),this); } // proxy : 代理类 // method : 代理类的调用处理程序的方法对象. public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable { log(method.getName()); Object result = method.invoke(target, args); return result; } public void log(String methodName){ System.out.println("执行了"+methodName+"方法"); }}</code></pre><p>测试!</p><pre><code>public class Test { public static void main(String[] args) { //真实对象 UserServiceImpl userService = new UserServiceImpl(); //代理对象的调用处理程序 ProxyInvocationHandler pih = new ProxyInvocationHandler(); pih.setTarget(userService); //设置要代理的对象 UserService proxy = (UserService)pih.getProxy(); //动态生成代理类! proxy.delete(); }}</code></pre><p>测试,增删改查,查看结果!</p><blockquote><h5 id="动态代理的好处"><a href="#动态代理的好处" class="headerlink" title="动态代理的好处"></a>动态代理的好处</h5></blockquote><p>静态代理有的它都有,静态代理没有的,它也有!</p><ul><li>可以使得我们的真实角色更加纯粹 . 不再去关注一些公共的事情 .</li><li>公共的业务由代理来完成 . 实现了业务的分工 ,</li><li>公共业务发生扩展时变得更加集中和方便 .</li><li>一个动态代理 , 一般代理某一类业务</li><li>一个动态代理可以代理多个类,代理的是接口!</li></ul><h1 id="AOP"><a href="#AOP" class="headerlink" title="AOP"></a>AOP</h1><h3 id="什么是AOP"><a href="#什么是AOP" class="headerlink" title="什么是AOP"></a>什么是AOP</h3><p>AOP(Aspect Oriented Programming)意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。</p><p></p><h3 id="Aop在Spring中的作用"><a href="#Aop在Spring中的作用" class="headerlink" title="Aop在Spring中的作用"></a>Aop在Spring中的作用</h3><p>提供声明式事务;允许用户自定义切面</p><p>以下名词需要了解下:</p><ul><li>横切关注点:跨越应用程序多个模块的方法或功能。即是,与我们业务逻辑无关的,但是我们需要关注的部分,就是横切关注点。如日志 , 安全 , 缓存 , 事务等等 ….</li><li>切面(ASPECT):横切关注点 被模块化 的特殊对象。即,它是一个类。</li><li>通知(Advice):切面必须要完成的工作。即,它是类中的一个方法。</li><li>目标(Target):被通知对象。</li><li>代理(Proxy):向目标对象应用通知之后创建的对象。</li><li>切入点(PointCut):切面通知 执行的 “地点”的定义。</li><li>连接点(JointPoint):与切入点匹配的执行点。</li></ul><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7JAeTYOaaH6rZ6WmLLgwQLHVOZ1JpRb7ViaprZCRXsUbH0bZpibiaTjqib68LQHOWZicSvuU8Y1dquUVGw/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p>SpringAOP中,通过Advice定义横切逻辑,Spring中支持5种类型的Advice:</p><p><img src="" data-original="https://mmbiz.qpic.cn/mmbiz_png/uJDAUKrGC7JAeTYOaaH6rZ6WmLLgwQLHbAWH8haUQeJ0LVBxxX0icC5TZlBkEBGibibey7jFrCbibPzQcRhkNFcGAA/640?wx_fmt=png&tp=webp&wxfrom=5&wx_lazy=1&wx_co=1" alt="img"></p><p>即 Aop 在 不改变原有代码的情况下 , 去增加新的功能 .</p><h3 id="使用Spring实现Aop"><a href="#使用Spring实现Aop" class="headerlink" title="使用Spring实现Aop"></a>使用Spring实现Aop</h3><p>【重点】使用AOP织入,需要导入一个依赖包!</p><pre><code><!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver --><dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.4</version></dependency></code></pre><h4 id="第一种方式"><a href="#第一种方式" class="headerlink" title="第一种方式"></a>第一种方式</h4><p><strong>通过 Spring API 实现</strong></p><p>首先编写我们的业务接口和实现类</p><pre><code>public interface UserService { public void add(); public void delete(); public void update(); public void search();}public class UserServiceImpl implements UserService{ @Override public void add() { System.out.println("增加用户"); } @Override public void delete() { System.out.println("删除用户"); } @Override public void update() { System.out.println("更新用户"); } @Override public void search() { System.out.println("查询用户"); }}</code></pre><p>然后去写我们的增强类 , 我们编写两个 , 一个前置增强 一个后置增强</p><pre><code>public class Log implements MethodBeforeAdvice { //method : 要执行的目标对象的方法 //objects : 被调用的方法的参数 //Object : 目标对象 @Override public void before(Method method, Object[] objects, Object o) throws Throwable { System.out.println( o.getClass().getName() + "的" + method.getName() + "方法被执行了"); }}public class AfterLog implements AfterReturningAdvice { //returnValue 返回值 //method被调用的方法 //args 被调用的方法的对象的参数 //target 被调用的目标对象 @Override public void afterReturning(Object returnValue, Method method, Object[] args,Object target) throws Throwable { System.out.println("执行了" + target.getClass().getName() +"的"+method.getName()+"方法," +"返回值:"+returnValue); }}</code></pre><p>最后去spring的文件中注册 , 并实现aop切入实现 , 注意导入约束 .</p><pre><code><?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!--注册bean--> <bean id="userService" class="com.kuang.service.UserServiceImpl"/> <bean id="log" class="com.kuang.log.Log"/> <bean id="afterLog" class="com.kuang.log.AfterLog"/> <!--aop的配置--> <aop:config> <!--切入点 expression:表达式匹配要执行的方法--> <aop:pointcut id="pointcut" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/> <!--执行环绕; advice-ref执行方法 . pointcut-ref切入点--> <aop:advisor advice-ref="log" pointcut-ref="pointcut"/> <aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/> </aop:config></beans></code></pre><p>测试</p><pre><code>public class MyTest { @Test public void test(){ ApplicationContext context = newClassPathXmlApplicationContext("beans.xml"); UserService userService = (UserService) context.getBean("userService"); userService.search(); }}</code></pre><p>Aop的重要性 : 很重要 . 一定要理解其中的思路 , 主要是思想的理解这一块 .</p><p>Spring的Aop就是将公共的业务 (日志 , 安全等) 和领域业务结合起来 , 当执行领域业务时 , 将会把公共业务加进来 . 实现公共业务的重复利用 . 领域业务更纯粹 , 程序猿专注领域业务 , 其本质还是动态代理 . </p><h4 id="第二种方式"><a href="#第二种方式" class="headerlink" title="第二种方式"></a>第二种方式</h4><p><strong>自定义类来实现Aop</strong></p><p>目标业务类不变依旧是userServiceImpl</p><p>第一步 : 写我们自己的一个切入类</p><pre><code>public class DiyPointcut { public void before(){ System.out.println("---------方法执行前---------"); } public void after(){ System.out.println("---------方法执行后---------"); }}</code></pre><p>去spring中配置</p><pre><code><!--第二种方式自定义实现--><!--注册bean--><bean id="diy" class="com.kuang.config.DiyPointcut"/><!--aop的配置--><aop:config> <!--第二种方式:使用AOP的标签实现--> <aop:aspect ref="diy"> <aop:pointcut id="diyPonitcut" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/> <aop:before pointcut-ref="diyPonitcut" method="before"/> <aop:after pointcut-ref="diyPonitcut" method="after"/> </aop:aspect></aop:config></code></pre><p>测试:</p><pre><code>public class MyTest { @Test public void test(){ ApplicationContext context = newClassPathXmlApplicationContext("beans.xml"); UserService userService = (UserService) context.getBean("userService"); userService.add(); }}</code></pre><h4 id="第三种方式"><a href="#第三种方式" class="headerlink" title="第三种方式"></a>第三种方式</h4><p><strong>使用注解实现</strong></p><p>第一步:编写一个注解实现的增强类</p><pre><code>package com.kuang.config;import org.aspectj.lang.ProceedingJoinPoint;import org.aspectj.lang.annotation.After;import org.aspectj.lang.annotation.Around;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;@Aspectpublic class AnnotationPointcut { @Before("execution(* com.kuang.service.UserServiceImpl.*(..))") public void before(){ System.out.println("---------方法执行前---------"); } @After("execution(* com.kuang.service.UserServiceImpl.*(..))") public void after(){ System.out.println("---------方法执行后---------"); } @Around("execution(* com.kuang.service.UserServiceImpl.*(..))") public void around(ProceedingJoinPoint jp) throws Throwable { System.out.println("环绕前"); System.out.println("签名:"+jp.getSignature()); //执行目标方法proceed Object proceed = jp.proceed(); System.out.println("环绕后"); System.out.println(proceed); }}</code></pre><p>第二步:在Spring配置文件中,注册bean,并增加支持注解的配置</p><pre><code><!--第三种方式:注解实现--><bean id="annotationPointcut" class="com.kuang.config.AnnotationPointcut"/><aop:aspectj-autoproxy/></code></pre><p>aop:aspectj-autoproxy:说明</p><pre><code>通过aop命名空间的<aop:aspectj-autoproxy />声明自动为spring容器中那些配置@aspectJ切面的bean创建代理,织入切面。当然,spring 在内部依旧采用AnnotationAwareAspectJAutoProxyCreator进行自动代理的创建工作,但具体实现的细节已经被<aop:aspectj-autoproxy />隐藏起来了<aop:aspectj-autoproxy />有一个proxy-target-class属性,默认为false,表示使用jdk动态代理织入增强,当配为<aop:aspectj-autoproxy poxy-target-class="true"/>时,表示使用CGLib动态代理技术织入增强。不过即使proxy-target-class设置为false,如果目标类没有声明接口,则spring将自动使用CGLib动态代理。</code></pre><h1 id="事务"><a href="#事务" class="headerlink" title="事务"></a>事务</h1><h3 id="回顾事务"><a href="#回顾事务" class="headerlink" title="回顾事务"></a>回顾事务</h3><ul><li>事务在项目开发过程非常重要,涉及到数据的一致性的问题,不容马虎!</li><li>事务管理是企业级应用程序开发中必备技术,用来确保数据的完整性和一致性。</li></ul><p>事务就是把一系列的动作当成一个独立的工作单元,这些动作要么全部完成,要么全部不起作用。</p><h4 id="事务四个属性ACID"><a href="#事务四个属性ACID" class="headerlink" title="事务四个属性ACID"></a>事务四个属性ACID</h4><ol><li><p>原子性(atomicity)</p></li><li><ul><li>事务是原子性操作,由一系列动作组成,事务的原子性确保动作要么全部完成,要么完全不起作用</li></ul></li><li><p>一致性(consistency)</p></li><li><ul><li>一旦所有事务动作完成,事务就要被提交。数据和资源处于一种满足业务规则的一致性状态中</li></ul></li><li><p>隔离性(isolation)</p></li><li><ul><li>可能多个事务会同时处理相同的数据,因此每个事务都应该与其他事务隔离开来,防止数据损坏</li></ul></li><li><p>持久性(durability)</p></li></ol><ol start="8"><li><ul><li>事务一旦完成,无论系统发生什么错误,结果都不会受到影响。通常情况下,事务的结果被写到持久化存储器中</li></ul></li></ol><blockquote><h4 id="测试"><a href="#测试" class="headerlink" title="测试"></a>测试</h4></blockquote><p>将上面的代码拷贝到一个新项目中</p><p>在之前的案例中,我们给userDao接口新增两个方法,删除和增加用户;</p><pre><code>//添加一个用户int addUser(User user);//根据id删除用户int deleteUser(int id);</code></pre><p>mapper文件,我们故意把 deletes 写错,测试!</p><pre><code><insert id="addUser" parameterType="com.kuang.pojo.User">insert into user (id,name,pwd) values (#{id},#{name},#{pwd})</insert><delete id="deleteUser" parameterType="int">deletes from user where id = #{id}</delete></code></pre><p>编写接口的实现类,在实现类中,我们去操作一波</p><pre><code>public class UserDaoImpl extends SqlSessionDaoSupport implements UserMapper { //增加一些操作 public List<User> selectUser() { User user = new User(4,"小明","123456"); UserMapper mapper = getSqlSession().getMapper(UserMapper.class); mapper.addUser(user); mapper.deleteUser(4); return mapper.selectUser(); } //新增 public int addUser(User user) { UserMapper mapper = getSqlSession().getMapper(UserMapper.class); return mapper.addUser(user); } //删除 public int deleteUser(int id) { UserMapper mapper = getSqlSession().getMapper(UserMapper.class); return mapper.deleteUser(id); }}</code></pre><p>测试</p><pre><code>@Testpublic void test2(){ ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); UserMapper mapper = (UserMapper) context.getBean("userDao"); List<User> user = mapper.selectUser(); System.out.println(user);}</code></pre><p>报错:sql异常,delete写错了</p><p>结果 :插入成功!</p><p>没有进行事务的管理;我们想让他们都成功才成功,有一个失败,就都失败,我们就应该需要<strong>事务!</strong></p><p>以前我们都需要自己手动管理事务,十分麻烦!</p><p>但是Spring给我们提供了事务管理,我们只需要配置即可;</p><h3 id="Spring中的事务管理"><a href="#Spring中的事务管理" class="headerlink" title="Spring中的事务管理"></a>Spring中的事务管理</h3><p>Spring在不同的事务管理API之上定义了一个抽象层,使得开发人员不必了解底层的事务管理API就可以使用Spring的事务管理机制。Spring支持编程式事务管理和声明式的事务管理。</p><p><strong>编程式事务管理</strong></p><ul><li>将事务管理代码嵌到业务方法中来控制事务的提交和回滚</li><li>缺点:必须在每个事务操作业务逻辑中包含额外的事务管理代码</li></ul><p><strong>声明式事务管理</strong></p><ul><li>一般情况下比编程式事务好用。</li><li>将事务管理代码从业务方法中分离出来,以声明的方式来实现事务管理。</li><li>将事务管理作为横切关注点,通过aop方法模块化。Spring中通过Spring AOP框架支持声明式事务管理。</li></ul><p><strong>使用Spring管理事务,注意头文件的约束导入 : tx</strong></p><pre><code>xmlns:tx="http://www.springframework.org/schema/tx"http://www.springframework.org/schema/txhttp://www.springframework.org/schema/tx/spring-tx.xsd"></code></pre><p><strong>事务管理器</strong></p><ul><li>无论使用Spring的哪种事务管理策略(编程式或者声明式)事务管理器都是必须的。</li><li>就是 Spring的核心事务管理抽象,管理封装了一组独立于技术的方法。</li></ul><p><strong>JDBC事务</strong></p><pre><code><bean id="transactionManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource" /></bean></code></pre><p><strong>配置好事务管理器后我们需要去配置事务的通知</strong></p><pre><code><!--配置事务通知--><tx:advice id="txAdvice" transaction-manager="transactionManager"> <tx:attributes> <!--配置哪些方法使用什么样的事务,配置事务的传播特性--> <tx:method name="add" propagation="REQUIRED"/> <tx:method name="delete" propagation="REQUIRED"/> <tx:method name="update" propagation="REQUIRED"/> <tx:method name="search*" propagation="REQUIRED"/> <tx:method name="get" read-only="true"/> <tx:method name="*" propagation="REQUIRED"/> </tx:attributes></tx:advice></code></pre><h4 id="spring事务传播特性:"><a href="#spring事务传播特性:" class="headerlink" title="spring事务传播特性:"></a>spring事务传播特性:</h4><p>事务传播行为就是多个事务方法相互调用时,事务如何在这些方法间传播。spring支持7种事务传播行为:</p><ul><li>propagation_requierd:如果当前没有事务,就新建一个事务,如果已存在一个事务中,加入到这个事务中,这是最常见的选择。</li><li>propagation_supports:支持当前事务,如果没有当前事务,就以非事务方法执行。</li><li>propagation_mandatory:使用当前事务,如果没有当前事务,就抛出异常。</li><li>propagation_required_new:新建事务,如果当前存在事务,把当前事务挂起。</li><li>propagation_not_supported:以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。</li><li>propagation_never:以非事务方式执行操作,如果当前事务存在则抛出异常。</li><li>propagation_nested:如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与propagation_required类似的操作</li></ul><p>Spring 默认的事务传播行为是 PROPAGATION_REQUIRED,它适合于绝大多数的情况。</p><p>假设 ServiveX#methodX() 都工作在事务环境下(即都被 Spring 事务增强了),假设程序中存在如下的调用链:Service1#method1()->Service2#method2()->Service3#method3(),那么这 3 个服务类的 3 个方法通过 Spring 的事务传播机制都工作在同一个事务中。</p><p>就好比,我们刚才的几个方法存在调用,所以会被放在一组事务当中!</p><h4 id="配置AOP"><a href="#配置AOP" class="headerlink" title="配置AOP"></a>配置AOP</h4><p>导入aop的头文件!</p><pre><code><!--配置aop织入事务--><aop:config> <aop:pointcut id="txPointcut" expression="execution(* com.kuang.dao.*.*(..))"/> <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/></aop:config></code></pre><p><strong>进行测试</strong></p><p>删掉刚才插入的数据,再次测试!</p><pre><code>@Testpublic void test2(){ ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); UserMapper mapper = (UserMapper) context.getBean("userDao"); List<User> user = mapper.selectUser(); System.out.println(user);}</code></pre><blockquote><p>思考问题?</p></blockquote><p>为什么需要配置事务?</p><ul><li>如果不配置,就需要我们手动提交控制事务;</li><li>事务在项目开发过程非常重要,涉及到数据的一致性的问题,不容马虎!</li></ul>]]></content>
</entry>
</search>