Thinkphp 如何解决表单重复提交问题

Thinkphp 如何解决表单重复提交问题
问:表单为何会被重复提交?
答:在开发中,如果一个新增或修改的表单,在后台完成数据库操作后我们设定的不是跳转到其他页面,而是返回本页面,这时点击浏览器的‘后退再提交’或‘刷新页面’,会导致form表单重复提交,即这条记录会被增加或修改两次。

导致表单重复提交的原因是:第一次提交的表单会被缓存到内存中,直到页面下次提交或页面关闭或转向其他页面时才消失。在自调用返回时,内存中的数据依然在,这时页面中的判断提交的代码依然可以检测到提交的值,顾会产生重复提交的效果。

解决办法
方法1:这里写代码片`:最简单:页面提交后转到另一个页面而不是本页面,举个栗子,比如你的 页面地址

http://yourdomain.com/User/Index/login

则该页面的表单action地址可以为另外的处理地址,如

<form action="{:U('User/Index/check_login')}" method="post">

但如果报错返回,或者用户点击回退按钮,还是会回到上一个地址,所以这种方法也不保险。还要搭配方法2,一起使用比较保险

方法2:提交表单后提交按钮变灰/隐藏提交按钮

这种方式一般是结合方法1来做的,通过JS来动态监听用户的点击动作,动态将按钮属性置成disabeld,即为灰色不可用。代码如下:
HTML

<form action="{:U('User/Index/check_login')}" method="post">
   <input type="text" name="username" value="" id="username" />
   <input type="password" name="userpwd" id="userpwd" />
   <input type="submit" name="login_btn" id="login_btn" value="登陆"/>
</form>

JS

$().ready(function(){
     $("#login_btn").on('click',function(){
            $(this).attr('disabled',true);
      });
});

方法1+方法2 结合后,基本上90%以上的重复提交问题都能解决,但是不好的是如果你点击了提交,而提交前的js验证没有通过,这时候就比较尴尬了,你无法再次点击提交…
所以这里还是要说下第三种方法,即在服务端解决这个问题

方法3:使用隐藏随机TOKEN值的方法进行重复提交判断

首先,在项目的公共文件functions.php中添加如下方法

//创建TOKEN
function createToken($tokenName) {
   $code = chr(mt_rand(0xB0, 0xF7)) . chr(mt_rand(0xA1, 0xFE)) .       chr(mt_rand(0xB0, 0xF7)) . chr(mt_rand(0xA1, 0xFE)) . chr(mt_rand(0xB0, 0xF7)) . chr(mt_rand(0xA1, 0xFE));
   session($tokenName, authcode($code));
}
//判断TOKEN
function checkToken($tokenName) {
    if ($tokenName == session('tokenName')) {
       session('tokenName', NULL);
       return TRUE;
    } else {
      return FALSE;
    }
}
//加密TOKEN 
function authcode($str) {
    $key = "YOURKEY";
    $str = substr(md5($str), 8, 10);
    return md5($key . $str);
}

在表单页面form中填入以下HTML代码

<input type="hidden" name="TOKEN" value="{:session('TOKEN')}" />

在页面展示前调用creatToken()方法生成token,在相应控制器POST请求中 使用 checkToken() 进行判断是否重复提交

if(IS_POST) {
    $post_token = I('post.TOKEN');
    if(!checkToken($post_token)){
        $this->error('请不要重复提交页面',U('User/Index/login'));
    }
}

基本上,这3个方法配合着使用,就能解决ThinkPHP开发中表单重复提交问题,当然也可以使用ThinkPHP的令牌机制,这样其实就更简单了,TP会默认在表单中生成一个隐藏域,到时候判断这个隐藏域是否存在以及和session中的值是否想的即可,原理和方法3是一样的。

涉及函数:
mt_rand() 生成一个随机数。
chr() 根据指定的 ASCII 值返回字符。

rand()和mt_rand()作用都是产生一个随机整数,都有两种使用形式:
1.int rand(void) / int mt_rand(void)
2.int rand(int $min, int $max) / int mt_rand($min, $max)

对第一种形式:
     rand()产生的随机数为0到getrandmax()之间
     mt_rand()产生的随机数为0到mt_getrandmax()之间

对第二种形式:
     rand()产生从$min$max之间的随机数
     mt_rand()产生从$min$max之间的随机数    

对比:
     mt_rand()是更好地随机数生成器,性能上比rand()快4倍,mt_getrandmax()所表示的数值范围也更大
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值