Ajax——(笔记)


Ajax

在网页中利用 XMLHttpRequest 对象和服务器进行数据交互的方式

URL统一资源定位符
http://www.cnblogs.com/liulongbinblogs/p/11649393.html
通信协议://服务器名称/资源在服务器的具体位置
请求 ——> 处理 ——> 响应

jQuery中的Ajax

$.get()

获取资源:$.get(url, [data], [callback])

$.get(
	'http://www.liulongbin.top:3006/api/getbooks', // 请求的资源地址
	{ id: 2 }, // 请求资源期间要携带的参数
	function(res){ // 请求成功时的回调函数
		console.log(res);
	}
)

$.post()

提交数据:$.post(url, [data], [callback])

$.post(
	'http://www.liulongbin.top:3006/api/addbook', // 提交数据的地址
	{ bookname: '水浒传', author: '施耐庵', publisher: '上海图书出版社' }, // 要提交的数据
	function(res){ // 数据提交成功时的回调函数
		console.log(res);
	}
)

$.ajax()

$.ajax({
	type: 'GET', // 请求的方式,例如 GET 或 POST
	url: 'http://www.liulongbin.top:3006/api/getbooks',  // 请求的 URL 地址
	data: { id: 2 },// 这次请求要携带的数据
	success: function(res) { // 请求成功之后的回调函数
		console.log(res);
	} 
})

$.ajax({
	type: 'POST', // 请求的方式,例如 GET 或 POST
	url: 'http://www.liulongbin.top:3006/api/addbook',  // 请求的 URL 地址
	data: { bookname: '水浒传', author: '施耐庵', publisher: '上海图书出版社' },// 这次请求要携带的数据
	success: function(res) { // 请求成功之后的回调函数
		console.log(res);
	} 
})

接口

使用 Ajax 请求数据时,被请求的 URL 地址,就叫做数据接口(简称接口)。同时,每个接口必须有请求方式。

接口文档
  1. 接口名称:用来标识各个接口的简单说明,如登录接口,获取图书列表接口等。
  2. 接口URL:接口的调用地址。
  3. 调用方式:接口的调用方式,如 GET 或 POST。
  4. 参数格式:接口需要传递的参数,每个参数必须包含参数名称、参数类型、是否必选、参数说明这4项内容。
  5. 响应格式:接口的返回值的详细描述,一般包含数据名称、数据类型、说明3项内容。
  6. 返回示例(可选):通过对象的形式,例举服务器返回数据的结构。

图书管理案例

function getBookList(){ // 获取 渲染 图书管理
	$.get(' http://www.liulongbin.top:3006/api/getbooks',function(rel){
	if(rel.status !== 200) return alert("获取失败");//不等于200是错的
		var rows = [];//存放数据
		$.each(rel.data,function(i,ele){
			var id = ele.id
			var bookname = ele.bookname;
			var author = ele.bookname;
			var publisher = ele.bookname;
			rows.push("<tr><td>"+id+"</td><td>"+bookname+"</td><td>"+author+"</td><td>"+publisher+"</td><td><a href='javascript:;' data-id="+ele.id+">删除</a></td></tr>");
					//数组追加数据
		});
	$("#tb").empty().append(rows.join(""));//将之前渲染的去掉重新追加数据将数组拆分成字符串
	})
}

form表单

属性描述
actionURL地址向何处发送表单数据,当提交表单后,页面会立即跳转到 action 属性指定的 URL 地址
methodget或post规定以何种方式把表单数据提交到 action URL
enctypeapplication/x-www-form-urlencoded规定在发送表单数据之前如何对其进行编码
target_blank 在新窗口中打开
_self 默认
用来规定在何处打开 action URL

在涉及到文件上传的操作时,必须将 enctype 的值设置为 multipart/form-data

Ajax提交表单

submit:监听事件
阻止默认行为:e.preventDefault()

$('#form1').submit(function(e) {
   // 阻止表单的提交和页面的跳转
   e.preventDefault()
})

$('#form1').on('submit', function(e) {
   // 阻止表单的提交和页面的跳转
   e.preventDefault()
})

jQuery中的serialize():获取表单中的所有数据表单元素必须有name属性

<form id="form1">
    <input type="text" name="username" />
    <input type="password" name="password" />
    <button type="submit">提交</button>
</form>
$('#form1').serialize() // 原生reset():清空表单内容
// 调用的结果:
// username=用户名的值&password=密码的值

评论列表案例

function getCmtList() { // 获取评论列表
    $.get('http://www.liulongbin.top:3006/api/cmtlist', function (res) {  
      if(res.status !== 200) {
        return alert('获取评论列表失败!')
      }
      var rows = []
      $.each(res.data, function (i, item) { // 循环拼接字符串
        rows.push('<li class="list-group-item">'+ item.content +'<span class="badge cmt-date">评论时间:'+ item.time +'</span><span class="badge cmt-person">评论人:'+ item.username +'</span></li>')
      })
      $('#cmt-list').empty().append(rows.join('')) // 渲染列表的UI结构
    })
  }
$('#formAddCmt').submit(function(e) { // 发表评论
    e.preventDefault() // 阻止表单的默认提交行为
    // 快速得到表单中的数据
    var data = $(this).serialize() // 查询字符串格式
    $.post('http://www.liulongbin.top:3006/api/addcmt', data, function(res) {
      if (res.status !== 201) {
        return alert('发表评论失败!')
      }
      // 刷新评论列表
      getCmtList()
      // 快速清空表单内容
      $('#formAddCmt')[0].reset()
    })
})

模板引擎

通过字符串拼接的形式,来渲染UI结构。如果UI结构比较复杂,则拼接字符串的时候需要格外注意引号之前的嵌套。且一旦需求发生变化,修改起来也非常麻烦

模板引擎,顾名思义,它可以根据程序员指定的模板结构和数据,自动生成一个完整的HTML页面

art-template

art-template 是一个简约、超快的模板引擎。中文官网首页为 http://aui.github.io/art-template/zh-cn/index.html
(模板结构)+(数据)=(渲染页面)

标准语法

在 {{ }} 内可以进行变量输出,或循环数组等操作

输出
{{value}} // 变量
{{obj.key}} // 对象属性
{{obj['key']}}
{{a ? b : c}} // 三元表达式
{{a || b}} // 逻辑或
{{a + b}} // 加减乘除
原文输出
{{@ value }} // 原文输出 保证 HTML 标签被正常渲染
条件输出
{{if value}} 按需输出的内容 {{/if}}  
{{if v1}} 按需输出的内容 {{else if v2}} 按需输出的内容 {{/if}}

循环输出

{{each arr}}
    {{$index}} {{$value}} // $index下标 $value 值
{{/each}}
过滤器
{{value | filterName}} // 使用过滤器
定义过滤器
template.defaults.imports.filterName = function(value){/*return处理的结果*/}
模板结构
<script type="text/html" id="tpl-user"> // 定义模板结构
    <h1>{{name}}    ------    {{age}}</h1>
    {{@ test}}

    <div>
      {{if flag === 0}}
      flag的值是0
      {{else if flag === 1}}
      flag的值是1
      {{/if}}
    </div>

    <ul>
      {{each hobby}}
      <li>索引是:{{$index}},循环项是:{{$value}}</li>
      {{/each}}
    </ul>

    <h3>{{regTime | dateFormat}}</h3>
  </script>
数据
<script>
    // 定义处理时间的过滤器
    template.defaults.imports.dateFormat = function (date) {
      var y = date.getFullYear()
      var m = date.getMonth() + 1
      var d = date.getDate()

      return y + '-' + m + '-' + d
    }

    // 2. 定义需要渲染的数据
    var data = { name: 'zs', age: 20, test: '<h3>测试原文输出</h3>', flag: 1, hobby: ['吃饭', '睡觉', '写代码'], regTime: new Date() }

    // 4. 调用 template 函数 
    // HTML结构 = template(模板结构,数据)
    var htmlStr = template('tpl-user', data)
    console.log(htmlStr)
    // 5. 渲染HTML结构
    $('#container').html(htmlStr)
</script>

模板基本原理

基本语法

RegExpObject.exec(string) 函数用于检索字符串中的正则表达式的匹配

var str = 'hello'
var pattern = /o/
// 输出的结果["o", index: 4, input: "hello", groups: undefined]
console.log(pattern.exec(str)) 
分组

正则表达式中 ( ) 包起来的内容表示一个分组,可以通过分组来提取自己想要的内容

var str = '<div>我是{{name}}</div>'
 var pattern = /{{([a-zA-Z]+)}}/
 var patternResult = pattern.exec(str)
 console.log(patternResult)
 // 得到 name 相关的分组信息
 // ["{{name}}", "name", index: 7, input: "<div>我是{{name}}</div>", groups: undefined]
replace函数

replace() 函数用于在字符串中用一些字符替换另一些字符

var result = '123456'.replace('123', 'abc') // 得到的 result 的值为字符串 'abc456'
简易模板引擎
<div id="user-box"></div>
<script type="text/html" id="tpl-user">
    <div>姓名:{{name}}</div>
    <div>年龄:{{ age }}</div>
    <div>性别:{{  gender}}</div>
    <div>住址:{{address  }}</div>
</script>
<script>
    // 定义数据
    var data = { name: 'zs', age: 28, gender: '男', address: '北京顺义马坡' }
    // 调用模板引擎
    var htmlStr = template('tpl-user', data)
    // 渲染HTML结构
    document.getElementById('user-box').innerHTML = htmlStr
</script>
function template(id, data) { // 获取模板 和数据
  var str = document.getElementById(id).innerHTML 
  var pattern = /{{\s*([a-zA-Z]+)\s*}}/
  var pattResult = null
  while (pattResult = pattern.exec(str)) { // 正则匹配{{}} 为null 结束
    str = str.replace(pattResult[0], data[pattResult[1]]) // 字符替换
  }
  return str
}

XMLHttpRequest

XMLHttpRequest(简称 xhr)是浏览器提供的 Javascript 对象,通过它,可以请求服务器上的数据资源

GET请求

// 1. 创建 XHR 对象
var xhr = new XMLHttpRequest()
// 2. 调用 open 函数,指定 请求方式 与 URL地址
xhr.open('GET', 'http://www.liulongbin.top:3006/api/getbooks?id=1') // ?id=1 是携带的参数
// 3. 调用 send 函数,发起 Ajax 请求
xhr.send()
// 4. 监听 onreadystatechange 事件
xhr.onreadystatechange = function() {
    // 4.1 监听 xhr 对象的请求状态 readyState ;与服务器响应的状态 status
    if (xhr.readyState === 4 && xhr.status === 200) {
        // 4.2 打印服务器响应回来的数据
        console.log(xhr.responseText)
    }
}

查询字符串

查询字符串(URL 参数)是指在 URL 的末尾加上用于向服务器发送信息的字符串(变量)
格式:将英文的 ? 放在URL 的末尾,然后再加上 参数=值 ,想加上多个参数的话,使用 & 符号进行分隔

?id=1&bookname=西游记

URL编码

encodeURI('黑马程序员') //  编码的函数
// 输出字符串  %E9%BB%91%E9%A9%AC%E7%A8%8B%E5%BA%8F%E5%91%98
decodeURI('%E9%BB%91%E9%A9%AC') // 解码的函数
// 输出字符串  黑马

POST请求

// 1. 创建 xhr 对象
var xhr = new XMLHttpRequest()
// 2. 调用 open()
xhr.open('POST', 'http://www.liulongbin.top:3006/api/addbook')
// 3. 设置 Content-Type 属性(固定写法)
// 在Ajax中会设置
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded')
// 4. 调用 send(),同时将数据以查询字符串的形式,提交给服务器
xhr.send('bookname=水浒传&author=施耐庵&publisher=天津图书出版社')
// 5. 监听 onreadystatechange 事件
xhr.onreadystatechange = function() {
    if (xhr.readyState === 4 && xhr.status === 200) {
        console.log(xhr.responseText)
    }
}

JSON

JSON 就是 Javascript 对象和数组的字符串表示法,它使用文本表示一个 JS 对象或数组的信息,因此,JSON 的本质是字符串

JSON的两种结构

对象结构

对象结构在 JSON 中表示为 { } 括起来的内容。数据结构为 { key: value, key: value, … } 的键值对结构。其中,key 必须是使用英文的双引号包裹的字符串,value 的数据类型可以是数字、字符串、布尔值、null、数组、对象6种类型。

数组结构

数组结构在 JSON 中表示为 [ ] 括起来的内容。数据结构为 [ “java”, “javascript”, 30, true … ] 。数组中数据的类型可以是数字、字符串、布尔值、null、数组、对象6种类型。

注意事项

①属性名必须使用双引号包裹
②字符串类型的值必须使用双引号包裹
③JSON 中不允许使用单引号表示字符串
④JSON 中不能写注释
⑤JSON 的最外层必须是对象或数组格式
⑥不能使用 undefined 或函数作为 JSON 的值

与JS对象的互转

方法说明
JSON.parse()从 JSON 字符串转换为 JS 对象
var obj = JSON.parse('{"a": "Hello", "b": "World"}') // 字符串转换为数据对象 反序列化
//结果是 {a: 'Hello', b: 'World'}
var json = JSON.stringify({a: 'Hello', b: 'World'}) // 数据对象转换为字符串 序列化
//结果是 '{"a": "Hello", "b": "World"}'

封装Ajax函数

XMLHttpRequest Level2

请求限时

xhr.timeout = 3000 //设置等待时间过了这个时间
xhr.ontimeout = function(event){//执行这个回调函数
     alert('请求超时!')
 }

FormData

// 1. 新建 FormData 对象
var fd = new FormData()
 // 2. 为 FormData 添加表单项
fd.append('uname', 'zs')
fd.append('upwd', '123456')
// 3. 创建 XHR 对象
var xhr = new XMLHttpRequest()
// 4. 指定请求类型与URL地址
xhr.open('POST', 'http://www.liulongbin.top:3006/api/formdata')
// 5. 直接提交 FormData 对象,这与提交网页表单的效果,完全一样
xhr.send(fd)
xhr.onreadystatechange = function() {}
// 获取表单元素
var form = document.querySelector('#form1')
// 监听表单元素的 submit 事件
form.addEventListener('submit', function(e) {
	e.preventDefault()
	// 根据 form 表单创建 FormData 对象,会自动将表单数据填充到 FormData 对象中
	var fd = new FormData(form)
	var xhr = new XMLHttpRequest()
     xhr.open('POST', 'http://www.liulongbin.top:3006/api/formdata')
     xhr.send(fd)
     xhr.onreadystatechange = function() {}
})

在使用原生的请求表单在上传文件数据时,用默认设置 enctype=“multipart/form-data”,那么请求报文会设置 “Content-Type: multipart/form-data”,但是我们在发送 Ajax 请求时会设置其他的请求报文,就不会去设置他

上传文件

1.定义 UI 结构

<!-- 导入 jQuery -->
<script src="./lib/jquery.js"></script>
<!-- 文件选择框 -->
<input type="file" id="file1" />
<!-- 上传文件按钮 -->
<button id="btnUpload">上传</button>
 <br />
<img src="./images/loading.gif" alt="" style="display: none;" id="loading" />

2.验证是否选择了文件

$('#btnUpload').on('click', function() {
     // 1. 将 jQuery 对象转化为 DOM 对象,并获取选中的文件列表
     var files = $('#file1')[0].files // .files用于获取文件列表
     // 2. 判断是否选择了文件
     if (files.length <= 0) {
         return alert('请选择图片后再上传!‘)
     }
 })

3.向 FormData 中追加文件

var fd = new FormData()
fd.append('avatar', files[0])

4.使用 xhr 发起上传文件的请求

// 1. 创建 xhr 对象
var xhr = new XMLHttpRequest()
// 2. 调用 open 函数,指定请求类型与URL地址。其中,请求类型必须为 POST
xhr.open('POST', 'http://www.liulongbin.top:3006/api/upload/avatar')
// 3. 发起请求
xhr.send(fd)
//监听 onreadystatechange 事件
xhr.onreadystatechange = function() {
  if (xhr.readyState === 4 && xhr.status === 200) {
    var data = JSON.parse(xhr.responseText)
    if (data.status === 200) { // 上传文件成功
      // 将服务器返回的图片地址,设置为 <img> 标签的 src 属性
      document.querySelector('#img').src = 'http://www.liulongbin.top:3006' + data.url
    } else { // 上传文件失败
      console.log(data.message)
    }
  }
}

使用jQuery发起上传文件的请求

$.ajax({
    method: 'POST',
    url: 'http://www.liulongbin.top:3006/api/upload/avatar',
    data: fd,
// 不修改 Content-Type 属性,使用 FormData 默认的 Content-Type 值
    contentType: false,
// 不对 FormData 中的数据进行 url 编码,而是将 FormData 数据原样发送到服务器
    processData: false,
    success: function(res) {
        console.log(res)
    }
})

Ajax请求开始:ajaxStart(callback)

$(document).ajaxStart(function() { // 会监听当前文档内所有的 Ajax 请求
     $('#loading').show() // 图片显示
 })

Ajax请求结束:ajaxStop(callback)

$(document).ajaxStop(function() { // 请求结束执行
     $('#loading').hide() // 图片隐藏
})

Axios

Axios 是专注于网络数据请求的库

get请求

axios.get(‘url’, { params: { /参数/ } }).then(callback)

// 请求的 URL 地址
var url = 'http://www.liulongbin.top:3006/api/get'
// 请求的参数对象
var paramsObj = { name: 'zs', age: 20 }
// 调用 axios.get() 发起 GET 请求
axios.get(url, { params: paramsObj }).then(function(res) {
     // res.data 是服务器返回的数据
     var result = res.data
     console.log(res)
})

post请求

axios.post(‘url’, { /参数/ }).then(callback)

// 请求的 URL 地址
var url = 'http://www.liulongbin.top:3006/api/post'
// 要提交到服务器的数据
var dataObj = { location: '北京', address: '顺义' }
// 调用 axios.post() 发起 POST 请求
axios.post(url, dataObj).then(function(res) {
    // res.data 是服务器返回的数据
    var result = res.data
    console.log(result)
})

axios请求

axios({
    method: 'GET',
    url: 'http://www.liulongbin.top:3006/api/get',
    params: { // GET 参数要通过 params 属性提供
        name: 'zs',
        age: 20
    }
}).then(function(res) {
    console.log(res.data)
})

axios({
    method: 'POST',
    url: 'http://www.liulongbin.top:3006/api/post',
    data: { // POST 数据要通过 data 属性提供
        bookname: '程序员的自我修养',
        price: 666
    }
}).then(function(res) {
     console.log(res.data)
})

同源策略

浏览器规定,A 网站的 JavaScript,不允许和非同源的网站 C 之间,进行资源的交互

同源

同源是两个页面的协议,域名和端口都相同,则两个页面具有相同的源

跨域

非同源反之,则是跨域

JSONP

JSONP (JSON with Padding) 是 JSON 的一种“使用模式”,可用于解决主流浏览器的跨域数据访问的问题
由于浏览器同源策略的限制,网页中无法通过 Ajax 请求非同源的接口数据。但是

自定义JSONP
<script src="http://ajax.frontend.itheima.net:3006/api/jsonp?callback=success&name=zs&age=20"></script> // 查询字符串 callback = success 指定调用success函数
// 响应回来的是 success{ massage: "JSONP响应成功",data: { name: 'zs' , age: '22'} }
<script>
   function success(data) { // 调用了这个函数
     console.log('获取到了data数据:')
     console.log(data)
   }
</script>
jQuery的JSONP

是通过

$.ajax({
    url: 'http://ajax.frontend.itheima.net:3006/api/jsonp?name=zs&age=20',
    // 如果要使用 $.ajax() 发起 JSONP 请求,必须指定 datatype 为 jsonp
    dataType: 'jsonp', // 会自动携带一个 callback=jQueryxxx 
    // 发送到服务端的参数名称,默认值为 callback
    jsonp: 'callback',
    // 自定义的回调函数名称,默认值为 jQueryxxx 格式
    jsonpCallback: 'abc',
    success: function(res) {
       console.log(res)
    }
 })

防抖

防抖策略(debounce)是当事件被触发后,延迟 n 秒后再执行回调,如果在这 n 秒内事件又被触发,则重新计时

var timer = null                    // 1. 防抖动的 timer

function debounceSearch(keywords) { // 2. 定义防抖的函数
    timer = setTimeout(function() { // 定时函数
    // 发起 JSONP 请求
    getSuggestList(keywords)
    }, 500)
}

$('#ipt').on('keyup', function() {  // 3. 在触发 keyup 事件时,立即清空 timer
    clearTimeout(timer)
    // ...省略其他代码
    debounceSearch(keywords) // 重新创建定时函数
})

节流

节流策略(throttle),顾名思义,可以减少一段时间内事件的触发频率。

$(function() {
  var angel = $('#angel')
  var timer = null // 1.预定义一个 timer 节流阀
  $(document).on('mousemove', function(e) {
    if (timer) { return } // 3.判断节流阀是否为空,如果不为空,则证明距离上次执行间隔不足16毫秒
    timer = setTimeout(function() {
      $(angel).css('left', e.pageX + 'px').css('top', e.pageY + 'px')
      timer = null // 2.当设置了鼠标跟随效果后,清空 timer 节流阀,方便下次开启延时器
    }, 16)
  })
})

HTTP

HTTP协议

(主体)+(内容)+(方式) (服务器+客户端) + (内容) + (响应)

通信协议

HTTP(规则)因此网页内容的传输协议又叫做超文本传输协议

HTTP请求消息

请求行、请求头部 、空行 和 请求体 4 个部分组成

HTTP响应消息

状态行、响应头部、空行 和 响应体 4 个部分组成

HTTP请求方法

GET(查询)POST(新增)PUT(修改)DELETE(删除)

HTTP相应状态码

状态码英文名称说明
200ok请求成功
201Created已创建
301MovedPermanently永久移动
404Not Found服务器无法根据客户端的请求找到资源(网页)。
500Internal Server Error服务器内部错误,无法完成请求
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值