跨域问题、回调函数、同步与异步、封装Jsonp、封装ajax

129 阅读2分钟

跨域问题

同源策略

浏览器的安全功能,协议、域名、端口都相同的地址称为同源

 同源策略

           同源策略是浏览器的一个安全功能,不同源的网页脚本在没有明确授权的情况下,不能读写对方资源。所谓同源是指"协议+域名+端口"三者相同。

           => 浏览器

           => 同源:  当访问从一个地方访问另一个地方资源时,如果 协议+域名+端口 相同就是同源访问

                协议+域名+端口

                http://www.qianfeng.com:8080/login.html

                 |

                http://www.qianfeng.com:8080/index.html

跨域问题

使用AJAX技术(XMLHttpRequest 对象),从一个网页去请求另一个网页资源时,违反浏览器同源策略限制,引起的安全问题,称为跨域。 域名:

 主域名不同 www.baidu.com/index.html –>www.sina.com/test.js
 子域名不同 www.666.baidu.com/index.html –>www.555.baidu.com/test.js
 域名和IP地址 www.baidu.com/index.html –>http://180.149.132.47/test.js

端口:  www.baidu.com:8080/index.html–… www.baidu.com:8081/test.js

协议:  www.baidu.com:8080/index.html–… www.baidu.com:8080/test.js

备注:  1、localhost和127.0.0.1虽然都指向本机,但也属于跨域

跨域错误

image.png

解决跨域问题

  1. jsonp(不使用ajax技术)

首先,不知道大家有没有注意,不管是我们的script标签的src还是img标签的src,或者说link标签的href他们没有被同源策略所限制, 比如我们有可能使用一个网络上的图片,就可以请求得到,<img src="ss3.baidu.com/aa.jpg>

jsonp就是使用同源策略这一“漏洞”,实现的跨域请求

其基本原理是利用HTML的

该协议的一个要点就是允许用户传递一个callback参数给服务端,然后服务端返回数据时会将这个callback参数作为函数名来包裹住JSON数据,这样客户端就可以随意定制自己的函数来自动处理返回数据了。

jsonp弊端 : JSONP仅仅用于get请求

代码实现

服务端 jsonp封装;将响应数据封装到名为callback的函数中返回

router.get('/test', (request,response) => { response.write( “callback( {code:1,data:{content:'内容'}} )”); }

image.png 2. 授权跨域资源共享

CORS是一个W3C标准,全称是"跨域资源共享"(Cross-origin resource )。 它允许浏览器向跨源服务器,发出XMLHttpRequest请求,从而克服了AJAX只能同源使用sharing的限制。

image.png

  1. 代理服务器

image.png

image.png

回调函数

一个函数作为另一个函数参数,在另一个函数中调用执行,这个函数称为回调函数

      回调函数作用:

          将函数中的数据作为参数传给回调函数处理
 回调函数作用:

             1. 将函数中的数据作为参数传给回调函数处理

             2. 处理异步任务的结果

同步与异步

<!-- 

       现实生活

           异步与同步任务

           准备茶水

              第一步, 烧开水    5分钟

              第二步, 买茶叶    2分钟

              第三步,  泡茶      1分



              同步操作: 一个任务完成后再处理下一个任务

              异步操作: 多个任务同时进行

       软件世界

           等待一个任务执行完成后,再执行下一个执行,同步任务

           一个任务开始后,不等主程序执行完,继续向下执行,这个任务称为异步任务

         异步任务

            定时器

            ajax网络任务





    -->

同步

在开始一个任务后,等待主程序结束,再开始下一个任务

异步

在开始一个任务后,不等待主程序结束,继续下一个任务

jsonp封装

<script>

			class MyJsonp {

				constructor(url) {

					this.scriptEle = document.createElement('script') // <script>

					this.scriptEle.setAttribute('src', url)

					document.body.appendChild(this.scriptEle)

				}

                getResult(cb){

                    // callback函数名与后端约定

                    window.callback = function(result){

                        // console.log('data >> ',data)

                        cb(result)

                    }

                }

			}

			const myJsonp = new MyJsonp('http://10.7.162.150:8089/api/jsonp/list')

            myJsonp.getResult(function(data){

                console.log('data>>11',data)

            })

            

		</script>

ajax封装

/**

 let options = {

            method:  'get',

            url: '',

            data:{

                username:'jack'

                age:18

            },

            success:function(data){

                    

            }

        }

 */

function ajax(options) {

	// 1. 创建XMLHttpRequest

	let xhr = new XMLHttpRequest()

	let param = formateParam(options.data) // name=jack&age=18

    let method = options.method.toUpperCase()

	// 2. 建立连接

	if (method == 'GET') {

		xhr.open(options.method, options.url + '?' + param)

		// 3. 发送请求

		xhr.send()

	}else if(method == 'POST'){

        xhr.open(options.method,options.url)

        xhr.setRequestHeader('content-type', 'application/x-www-form-urlencoded')

        xhr.send(param)

    }



	// 4. 接收响应数据

	xhr.onreadystatechange = function () {

		// 4.1 是否响应完成

		if (xhr.readyState === 4) {

			// 4.2 是否成功响应

			if (xhr.status === 200) {

				let data = xhr.responseText // 响应内容

				data = JSON.parse(data)

				options.success(data)

			} else {

				alert('网络出错 ' + xhr.status)

			}

		}

	}

}



/**

 * 格式化参数

 *  {name:'jack',age:18}  =>  name=jack&age=18

 *   遍历对象,属性转换成名称=值形式,存储到数组, 再将数组元素用&符号拼接join('&)

 *    ['name=jack','age=18']  ->

 */

function formateParam(obj) {

	let arr = []

	for (const key in obj) {

		let item = `${key}=${obj[key]}` // name=jack   age=18

		arr.push(item) // ['name=jack','age=18;]

	}

	return arr.join('&') // name=jack&age=18

}



// let str = formateParam({name:'jack',age:18})

// console.log(str)