PHP / MYSQL - 使用mysql_insert_id插入多行?

时间:2011-09-08 22:20:35

标签: php mysql

背景

我正在从客户端表单将基本客户端信息上载到数据库。当客户点击“已完成”时,输入的信息(如(名称城市国家等))将存储在 client_info中表。

作为替代方案,客户端也可以将图像上传到服务器。然后,这些图像的信息( image_path image_name )将存储在名为 client_images 的另一个单独的表中,该表具有外键 client_info 相关联。

client_info表:

CREATE TABLE IF NOT EXISTS `client_info` (
  `client_id` int(11) NOT NULL AUTO_INCREMENT,
  `client_name` varchar(100) NOT NULL,
  `client_city` varchar(100) NOT NULL,
  `client_country` varchar(10) NOT NULL,
  PRIMARY KEY (`client_id`)
) ENGINE=InnoDB  DEFAULT CHARSET=latin1 AUTO_INCREMENT=10446 ;

client_images表:

CREATE TABLE IF NOT EXISTS `client_images` (
  `image_id` int(11) NOT NULL AUTO_INCREMENT,
  `image_path` varchar(100) NOT NULL,
  `image_name` varchar(100) NOT NULL,
  `client_id` int(11) NOT NULL,
  PRIMARY KEY (`image_id`),
  KEY `client_id` (`client_id`)
) ENGINE=InnoDB  DEFAULT CHARSET=latin1 AUTO_INCREMENT=415;

问题:

我已成功将一行上传到 client_images 表。添加多行会导致以下错误

Cannot add or update a child row: a foreign key constraint fails (`db`.`client_images`, CONSTRAINT `client_images_ibfk_1` FOREIGN KEY (`client_id`) REFERENCES `client_info` (`client_id`) ON DELETE CASCADE ON UPDATE CASCADE) : 1452

但是,当我通过控制台测试它添加多行和已经存在的 client_id 时,它运行得非常好。

示例:

mysql> INSERT INTO client_images (image_path, image_name, client_id) VALUES ('pathname1', 'name1', 10241);
Query OK, 1 row affected (0.00 sec)

mysql> INSERT INTO client_images (image_path, image_name, client_id) VALUES ('pathname2', 'name2', 10241);
Query OK, 1 row affected (0.00 sec)

mysql> INSERT INTO client_images (image_path, image_name, client_id) VALUES ('pathname3', 'name3', 10241);
Query OK, 1 row affected (0.00 sec)

我需要指出的是,我把它排成一行:

文件:(create_article.php)

insert_client_info($client_name, $client_city, $client_country);    

include('uploads.php');

?>

<?php mysql_close($connection); ?>

uploads.php 里面我有一个 foreach 循环,它计算要上传的文件数量,在循环结束时我目前有一个函数< strong> insert_images($ image_path,$ image_name);

功能(insert_images):

function insert_images($directory_path, $image_name) {  

    global $connection;
    $client_id=mysql_insert_id();

    $query = "INSERT INTO client_images (
        image_path, image_name, client_id
    ) VALUES (
        '{$directory_path}', '{$image_name}', '{$client_id}'
    )";
$result = mysql_query($query, $connection);

if ($result) {
    // Success!
    echo "<p></p>";
    echo "Successfully uploaded the file.";
    echo "<p></p>";

} else {
    // Display error message.
    echo "<p>" . mysql_error() ." : ".mysql_errno(). "</p>";
}
return $result;
}

阅读其他类似的问题我明白Achilles的脚跟可能是 mysql_insert_id(),并且它会以某种方式使用与 foreach 循环中的第一个不同的另一个连接,因此导致获得前面提到的MYSQL错误。

由于 / EDZ

2 个答案:

答案 0 :(得分:1)

这一部分:

$query = "INSERT INTO client_images (
        image_path, image_name, client_id
    ) VALUES (
        '{$directory_path}', '{$image_name}', '{$client_id}'
    )";

将是:

$query = "INSERT INTO client_images (
        image_path, image_name, client_id
    ) VALUES (
        '$directory_path', '$image_name', $client_id
    )";

您必须使用mysql_real_escape_string()转义变量以阻止sql injection

$client_id也是数字,因此无需引用'$client_id'

答案 1 :(得分:0)

  

添加多行会导致以下错误:

最后一个插入标识是由当前MySQL会话中最近一次生成自动递增标识的INSERT生成的标识。

插入第一个图像时,最新的INSERT是client_info的INSERT。

但是一旦你到达第二个图像,最新的INSERT就是第一个图像的INSERT,所以mysql_insert_id()返回的值是不同的。同样,每个后续INSERT都会更改mysql_insert_id()返回的值。

你应该在你的foreach循环开始插入图像之前设置变量$ client_id ,并将该变量传递给每次调用insert_images()。在uploads.php中:

$client_id=mysql_insert_id();
foreach (...) {
    insert_images($client_id, ...);
}

将$ client_id作为insert_images()函数的参数传递,而不是在函数中设置$ client_id。

function insert_images($client_id, $directory_path, $image_name) {  
    ...
}

P.S。:@tttony有一个很好的观点,他提醒你要注意SQL注入。

另请参阅我的演示文稿SQL Injection Myths and Fallacies或我的书SQL Antipatterns: Avoiding the Pitfalls of Database Programming中有关SQL注入的章节。