原生js实现Ajax,JSONP

5次阅读

共计 3209 个字符,预计需要花费 9 分钟才能阅读完成。

Ajax 内部的几个执行步骤

创建 XMLHttpRequest 对象 (new XMLHttpRequest())
设置请求头 (setRequestHeader)
连接服务器 (open())
设置回调 (onreadyStateChange)
发送数据 (send())
在回调函数中获取数据

JSONP
利用 script 标签可以跨域请求资源解决跨域问题。详细解释可以看彻底弄懂跨域问题
前端代码
/*
* 原生 js 实现 Ajax
* */

function Ajax(params) {
params = params || {};
params.data = params.data || {};
var _json = params.jsonp ? jsonp(params): json(params); // 判断是 json 还是 jsonp
function json(params) {// 普通请求
params.type = (params.type || ‘GET’).toUpperCase(); // 设置请求默认类型
var urlData = formatParams(params.data); // 对数据进行格式化
var xhr = null; // 对 xhr 进行初始化
if (window.XMLHttpRequest) {
xhr = new window.XMLHttpRequest();
} else {
xhr = new ActiveXObject(‘Microsoft.XMLHTTP’);
}
var headers = params.headers || {};
if (params.type === ‘GET’) {
xhr.open(params.type, params.url + ‘?’ + urlData, true);
setHeaders(xhr, headers);
xhr.send(null);
} else {
xhr.open(params.type, params.url, true);
setHeaders(xhr, headers);
xhr.send(JSON.stringify(params.data));
}
xhr.onreadystatechange = function () {
if (xhr.readyState === 4) {
var status = xhr.status;
if (status >= 200 && status < 300) {
var response = ”;
var type = xhr.getResponseHeader(‘Content-Type’);
if (type.indexOf(‘xml’) !== -1 && xhr.responseXML) {// xml 格式
response = xhr.responseXML;
} else if (type.indexOf(‘application/json’) !== -1) {// JSON 格式
response = JSON.parse(xhr.responseText);
} else {
response = xhr.responseText; // 字符串格式
}
params.success && params.success(response);
} else {
params.error && params.error(status);
}
}
}
}
function jsonp(params) {
var callbackName = params.jsonp; // 回调函数名
var head = document.getElementsByTagName(‘head’)[0];
params.data[‘callback’] = callbackName;
var data = formatParams(params.data);
var script = document.createElement(‘script’);
head.appendChild(script);
// 创建 jsonp 函数,成功后自动让 success 函数调用,在自动删除
window[callbackName] = function (json) {// 设置回调,获取后台数据后才执行
head.removeChild(script);
clearTimeout(script.timer);
window[callbackName] = null;
params.success && params.success(json);
};
script.src = params.url + ‘?’ + data; // 设置 src 的时候才开始向后台请求数据
if (params.time) {// 限定时间
script.timer = setTimeout(function () {
window[callbackName] = null;
head.removeChild(script);
params.error && params.error({
message: ‘ 超时 ’
})
}, params.time)
}

}
function formatParams(data) {
// 使用 encodeURIComponent 对 URI 的某部分编码
var arr = [];
for (var key in data) {
arr.push(encodeURIComponent(key) + ‘=’ + encodeURIComponent(data[key]));
}
// 添加随机数,防止缓存
arr.push(‘v=’ + random());
return arr.join(‘&’);
}
function random() {
return Math.floor(Math.random() * 10000 + 500);
}
function setHeaders(xhr, headers) {
for (var key in headers) {
xhr.setRequestHeader(key, headers[key]);
}
}
}
使用方法

不用 jsonp 请求
Ajax({
url: ‘ 后端接口 ’,
type: ‘POST’,
headers: {
‘Content-Type’: ‘application/json’,
token: ‘xxx’
},
success(res) {
console.log(res);
},
error(status) {
console.log(`some error status = ${status}`);
}
})

jsonp 请求
Ajax({
url: ‘http://localhost:8080’,
headers: {
‘Content-Type’: ‘application/json’
},
jsonp: ‘getUser’,
time: 2000,
success(res) {
console.log(res);
},
error(status) {
console.log(`some error status = ${status.msg}`);
}
})

jsonp 后台配置代码
var querystring = require(‘querystring’);
var http = require(‘http’);
var server = http.createServer();

server.on(‘request’, function(req, res) {
var params = querystring.parse(req.url.split(‘?’)[1]);
var fn = params.callback;

// jsonp 返回设置
res.writeHead(200, { ‘Content-Type’: ‘text/javascript’});
var data = {
user: ‘xbc’,
password: ‘123456’
}
res.write(fn + ‘(‘ + JSON.stringify(data) + ‘)’);

res.end();
});

server.listen(‘8080’);
console.log(‘Server is running at port 8080…’);
参考文章

原生 JavaScript 实现 AJAX、JSONP
ajax 快速入门

正文完
 0