开发者社区> 问答> 正文

PHP和mysql中,如何防止数据重复insert?

因为运行的时候有延迟。

sql1 = SELECT * FROM posts WHERE mid = 2

sql2 = INSERT INTO posts (id,mid,other) VALUES ('',".$c.",'def');

因为用了for 循环。首先是检测是否存在Sql1,如果不存在,则sql2。
再重复循环。可是,当前sql2第一次没有运行insert时,第二次循环的时候执行Sql1是通过的,然后再次执行sql2,导致运行了两次sql2。

可能同时触发了一个链接,这样导致重复插入了。
综合几位朋友的说法,我决定不在这个循环里面插入数据。

但好像:
INSERT INTO posts (id,mid,other) VALUES ('',".$c.",'def'),('','c2','def2'),('','c2','def2'),('','c2','def2');
这样子不行,因为假如其中一个出错,那么整条就好像就不能插入了。
这个要怎么处理呢?

听说要用事务处理,但是不懂如何使用。手册已经看过了,但不明白。。。

[以下只是举例子]
$c 是 变量, mid 字段不能设置为“唯一”

$c = '10000';
for($a = 0;$a < 100; $a++){
    $sql1 =  "SELECT * FROM `posts` WHERE `mid` = ".$c;
    $query1 = mysql_query($sql1);
    $row = mysql_num_rows($query1);

    if($row == 0){
        $sql2 = "INSERT INTO `posts` (`id`,`mid`,`other`) VALUES ('',".$c.",'def')";
        $query2 = mysql_query($sql2);
    }
}

展开
收起
落地花开啦 2016-06-17 17:14:58 5152 0
1 条回答
写回答
取消 提交回答
  • 喜欢技术,喜欢努力的人

    首先,在循环里面做sql查询是相当低效的,强烈建议你修改逻辑,避免这种情况

    关于你举例子的那段for循环代码,两个优化方案
    其一,在循环里拼接insert的数组,大概代码如下

    $d = [];
    for($i = 0; $i < 100; $i++)
    {
        $r = mt_rand(1, 100);
        //排除重复值,
        if(!in_array($r, $d))
        {
             $d[] = $r;
        }
    }
    $pdo = new PDO($dsn, $user, $passwd);
    $pdo->prepare("insert into table(col1,col2) values(?,?)");
    foreach($d as $v)
    {
        $pdo->execute(array($v, $v));
    }

    如上,复用insert语句模板,同时也排除了可重复插入的数值,如果对每一组值有特殊处理,也可以放在foreach处理
    我很推荐你使用此种方式,这种方式也是安全的,能良好的预防sql注入,而无需你再去考虑太多sql注入的问题

    其二,拼接一个比较长的insert的语句,

    insert into table_name(col1,col2, ......) values(v1,v2,......),(v1,v2,......),......

    上述二者可能并不切合题意,因为可能之前库里就存在和当前插入有重复的数据
    在mysql里有如下sql语法

    INSERT INTO TABLE (a,b,c)
    VALUES (1,2,3) ON DUPLICATE KEY UPDATE c=c+1;

    上句中的意思就是,插入一条语句到表中,如果插入的数据和表中唯一索引或者主键重复,更新字段c的值

    2019-07-17 19:43:11
    赞同 展开评论 打赏
问答排行榜
最热
最新

相关电子书

更多
阿里云栖开发者沙龙PHP技术专场-深入浅出网络编程与swoole内核-吴镇宇 立即下载
PHP安全开发:从白帽角度做安全 立即下载
PHP 2017.北京 全球开发者大会——高可用的PHP 立即下载

相关镜像