Jadual Kandungan
回复内容:
Rumah pembangunan bahagian belakang tutorial php PHP和mysql中,如何防止数据重复insert?

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

Jun 06, 2016 pm 08:46 PM
php

因为运行的时候有延迟。

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 字段不能设置为“唯一”

<br>$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);
    }
Salin selepas log masuk
Salin selepas log masuk

}

回复内容:

因为运行的时候有延迟。

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 字段不能设置为“唯一”

<br>$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);
    }
Salin selepas log masuk
Salin selepas log masuk

}

问题描述的不是很清晰,

可是,当前sql2第一次没有运行insert时,第二次循环的时候执行Sql1是通过的,然后再次执行sql2,导致运行了两次sql2。

上面这句话没看懂什么意思

首先,在循环里面做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));
}
Salin selepas log masuk

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

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

insert into table_name(col1,col2, ......) values(v1,v2,......),(v1,v2,......),......
Salin selepas log masuk

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

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

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

结合这个sql语句,能很完美解决插入值重复的问题吧?

循环里敢这么玩!

不懂你这个$sql1判断的意义何在,设置id这个是唯一key不就好了么?而且同 @宋小北 循环里面敢这么玩!!

今天看到个文章和这个问题非常契合,特此补上:http://blog.sae.sina.com.cn/archives/3491 -2014/04/17 12:41

只要在'posts'的表,加上唯一UNIQUE即可,或者當然可以用

INSERT INTO
table(column1,column2,column3 ...columnN)
SELECT value1,value2,value3 ...valueN
FROM dual
WHERE NOT EXISTS(
SELECT *
FROM table
WHERE value = ?
);

先查询可用的mid, 再插入数据

$variable_array = array();
$c = 1000;
for($a = 0;$a < 100; $a++){
    $variable_array[] = $c;    
}
if (count($variable_array) == 0) {
    return false;
}
// 去重
$variable_array = array_unique($variable_array);
// 获取可用mid
$sql_head = "SELECT `mid` FROM `posts` WHERE `mid` in (#variable#) ";
// 加上单引号, 如果确定都是数字,那就不加
$mid_array = array_map(function($value){return "'$value'";}, $variable_array);
$mid_list = array();
foreach (array_chunk($mid_array, 500) as $piece) {
    $variable_str = implode(",", $piece);
    $sql_piece = str_replace("#variable#", $variable_str, $sql_head);
    $query_piece = mysql_query($sql_piece);
    // 获取mid $mid_piece
    $mid_piece = mysql_fetch_array($query_piece);
    $mid_list = array_merge($mid_list, array_diff($piece, $mid_piece););
}
// 插入数据库
$insert_str_array = array_map(function($value){return "('', '$value', 'def')";}, $mid_list);
$insert_head = "INSERT INTO `posts` (`id`,`mid`,`other`) VALUES ";
foreach (array_chunk($insert_str_array, 500) as $piece) {
    $variable_str = implode(",", $piece);
    $insert_sql = $insert_head.$variable_str;
    mysql_query($insert_sql);
}
Salin selepas log masuk

很久没写php代码, 怎么获取mysql_query结果忘记了!!!直接用得mysql_fetch_array.

1.唯一索引。2.按照特征拿相关数据,进行对比

试试

<code>replace into `table` values set  key=val
</code>
Salin selepas log masuk
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn

Artikel Panas

R.E.P.O. Kristal tenaga dijelaskan dan apa yang mereka lakukan (kristal kuning)
2 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
Repo: Cara menghidupkan semula rakan sepasukan
4 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Adventure: Cara mendapatkan biji gergasi
3 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌

Artikel Panas

R.E.P.O. Kristal tenaga dijelaskan dan apa yang mereka lakukan (kristal kuning)
2 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
Repo: Cara menghidupkan semula rakan sepasukan
4 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Adventure: Cara mendapatkan biji gergasi
3 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌

Tag artikel panas

Notepad++7.3.1

Notepad++7.3.1

Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina

SublimeText3 versi Cina

Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1

Hantar Studio 13.0.1

Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6

Dreamweaver CS6

Alat pembangunan web visual

SublimeText3 versi Mac

SublimeText3 versi Mac

Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Panduan Pemasangan dan Naik Taraf PHP 8.4 untuk Ubuntu dan Debian Panduan Pemasangan dan Naik Taraf PHP 8.4 untuk Ubuntu dan Debian Dec 24, 2024 pm 04:42 PM

Panduan Pemasangan dan Naik Taraf PHP 8.4 untuk Ubuntu dan Debian

Konfigurasi Projek CakePHP Konfigurasi Projek CakePHP Sep 10, 2024 pm 05:25 PM

Konfigurasi Projek CakePHP

Tarikh dan Masa CakePHP Tarikh dan Masa CakePHP Sep 10, 2024 pm 05:27 PM

Tarikh dan Masa CakePHP

Muat naik Fail CakePHP Muat naik Fail CakePHP Sep 10, 2024 pm 05:27 PM

Muat naik Fail CakePHP

Penghalaan CakePHP Penghalaan CakePHP Sep 10, 2024 pm 05:25 PM

Penghalaan CakePHP

Bincangkan CakePHP Bincangkan CakePHP Sep 10, 2024 pm 05:28 PM

Bincangkan CakePHP

Cara Menyediakan Kod Visual Studio (Kod VS) untuk Pembangunan PHP Cara Menyediakan Kod Visual Studio (Kod VS) untuk Pembangunan PHP Dec 20, 2024 am 11:31 AM

Cara Menyediakan Kod Visual Studio (Kod VS) untuk Pembangunan PHP

Panduan Ringkas CakePHP Panduan Ringkas CakePHP Sep 10, 2024 pm 05:27 PM

Panduan Ringkas CakePHP

See all articles