页面数据交互的发展过程
付款是我们日常中常见的一种金钱交易,用户在页面中点击付款按钮,网页提交请求给服务器,服务器收到请求后在数据库对金额进行扣减,然后将消息返回给页面,告诉用户给付款成功。
在讲JSONP之前,我们先看看以前的前端程序员是用什么办法实现一个网页数据交互过程的。
用form表单向服务器发起请求
- 页面代码如下:用表单提交post请求,请求路径为/pay
Title
您的账户余额是&&&amount&&&
- 服务器代码
当请求的路径为/pay时,数据库进行扣减金额一块钱,然后给页面返回success
if(path === '/pay'&& method.toUpperCase()==='POST'){
//如果的路径是path,而且你的方法是post
var amount = fs.readFileSync('./db', 'utf8') //100
//那么我就去读一下你当前有多少钱
var newAmount = amount - 1
//然后把这个钱扣一块,扣了一块之后就是99
fs.writeFileSync('./db', newAmount)
//然后再把剩下的钱存到数据库里
response.write('success')
//然后告诉你扣费成功了
response.end()
//结束
}
当我们点击付款的时候可以看到扣费了一块钱
再回到我们之前自己用文件db存的一个数据库,库里就只有一个数字100,上面点击扣费后,数据库里的余额剩99了
再来模拟一下失败付款扣费的方案
if(path === '/pay'&& method.toUpperCase()==='POST'){
var amount = fs.readFileSync('./db', 'utf8') //100
var newAmount = amount - 1
if(Math.random()>0.5{
fs.writeFileSync('./db', newAmount)
response.write('success')
}else{
response.write('fail')
}
response.end()
这里有一个比较重要的细节,在点击付款按钮的form表单,一旦提交了,就一定会刷新当前页面,不管是成功付款还是失败付款,都会刷新页面。这样用户体验效果就很差了,我们可以用另外一种方式优化一下,但是这个方式也是比较古老的,就是用iframe
来实现。
您的账户余额是&&&amount&&&
//使用iframe来实现,这个方法也很古老
img发请求法
- 浏览器有一个特点,一旦发现你在内存里面创建了一个img,它就会去请求这个img的source.
- 这种办法有一个缺陷,就是没办法去post,所以它一定会去get
- 使用用img发起一个请求,服务器会同时执行两句话
- 页面代码
用图片发起get请求
button.addEventListener('click', (e) => {
let image = document.createElement('img')
image.src = '/pay'
image.onload = function() {
alert('打钱成功')
amount.innerText = amount.innerText - 1
}
image.onerror = function() {
alert('打钱失败')
}
}
- 服务器代码
if(Math.random()>0.5{
fs.writeFileSync('./db', newAmount)
response.setHeader('Content-Type','image/ipg')
response.statusCode = 200
response.write('fs.writeFileSync('./dog.jpg'))
//只能传送一张真的图片,才能让服务器成功发起请求
}else{
response.status=400
response.write('fail')
}
response.end()
script发请求法(SRJ)
- 尝试用script发起请求,但是到服务器运行后发现没有丝毫变化
通过其他前端工程师不断地尝试,发现只有把放到body(页面)里面,服务器才能发起请求,与img发起请求法不同,img是不需要这一步的,直接就可以向服务器发请求,而用
script
请求法是需要这一步的
button.addEventListener('click', (e) => {
let script = document.createElement('script')
script.src = '/pay'
document.body.appendChild(script)
script.onload=function(){
alert('success')
}
script.onerror = function() {
alert('fail')
}
if(path === '/pay'){
var amount = fs.readFileSync('./db', 'utf8') //100
var newAmount = amount - 1
if(Math.random()>0.5{
fs.writeFileSync('./db', newAmount)
response.setHeader('Content-Type', 'application/javascript')
response.statusCode = 200
response.write('')
}else{
response.status=400
response.write('fail')
}
response.end()
通过以上方法,我们可以用script
成功向服务器发起请求,比用img发起请求法要快一些,不需要再返回一张图片了。
但是,这种方法有一个问题还没解决。就是当我们向服务器请求了script
之后,这个script
是会执行的呀。什么叫会执行?请看下图:
我们尝试一下把服务器代码的空字符串换一下
下图可以看到,当我们执行完多出来的script
的内容后,才会继续执行onload事件(success)
当我们点击打钱,就会创建一个script
标签,由于页面中多出了一个script
,于是浏览器就会执行pay这个script
里面的内容。也就是alert
里面的"我是pay",然后才继续执行onload事件。
既然先执行响应内容,然后再调用onload,连续触发两次alert事件,如此多此一举。我们为何不直接把所有的内容都放进pay里面?同时本地页面监听script标签的onload,onerror事件来删除动态创建的script标签,节省内存。
这个方案就叫做SRJ(Server Rendered JavaScript),服务器返回的JavaScript
- 一些重要的操作,比如付款、取款,都不要用get来操作,一律都用post操作。
用SRJ实现不同的网站之间(不同的域名)的数据交互
如果想让当前的网站跟另一个服务器,另一个域名下的接口交流,怎么实现呢?
- 首先我们设置两个hosts
- 再开启两个sever (tip:可以用PORT来设置端口)
- 现在打开网页访问我们设置的第一个网站
frank.com:8001
再把jack.com:8002
也打开
由于我们在刚刚写了127.0.0.1 frank.com
和127.0.0.1 jack.com
所以相当于127.0.0.1 8001
和127.0.0.1 8002
- 现在我们让
frank.com:8001
的前端去访问jack.com:8002
的后端
button.addEventListener('click', (e) => {
let script = document.creatElement('script')
script.src = 'http://jack.com:8002/pay'
document.body.appendChild(script)
script.onload = function(e) {
e.currentTarget.remove()
}
script.onerror = function(e) {
alert('fail')
e.currentTarget.remove()
}
})
回到frank.com
页面
这就是两个网站之间通过SRJ交流
以上就是一个成用数据库实现一个付款扣费的方案,和失败的付款扣费方案,但是上面的方法是旧的时代前后端用数据库配合的方法,下面我们用现在流行的方案再来实现一遍
JSONP
JSONP(JSON with Padding) 是 json 的一种"使用模式",可以让网页从别的域名(网站)那获取资料,即跨域读取数据。
上面的SRJ方案有一个很大的缺陷,就是双方网站的代码耦合度太高,为了代码的解耦,可以使用JSONP方案。
JSONP完整过程:
- 请求方:前端程序员(浏览器)
请求方动态创建一个
script
标签,script
的src路径指向响应方,同时传一个查询参数callback函数,根据约定必须是callback,函数名可随机化