5

我正在尝试在我的数据库中重新散列用户 ID,但我被卡住了。

<?php
include("session.php");
include("functions.php");
$conn = ConnectMySQL();
setTimezone($session->username);

$sql = "SELECT username, userid FROM users";
$result = mysql_query($sql) or die(mysql_error());
while($rows = mysql_fetch_array($result)){
    $username = $rows['username'];
    $old = mysql_real_escape_string($rows['userid']);
    $new = mysql_real_escape_string($session->generateRandID());

    $moresql = "START TRANSACTION;
                UPDATE users SET userid='$new' WHERE userid='$old';
                UPDATE comments SET user='$new' WHERE user='$old';
                UPDATE forum_posts SET poster_name='$new' WHERE poster_name='$old';
                UPDATE forum_topics SET topic_poster_name='$new' WHERE topic_poster_name='$old';
                UPDATE images SET author='$new' WHERE author='$old';
                UPDATE likes SET user='$new' WHERE user='$old';
                UPDATE music SET author='$new' WHERE author='$old';
                UPDATE ratings SET user='$new' WHERE user='$old';
                COMMIT;";
    $newresult = mysql_query($moresql) or die(mysql_error());
    if(!$newresult){echo "There was a problem with changing $username's hash. \n";}
    else{echo "Changed $username's hash<i>!</i> \n";}
}
mysql_close($conn);
?>

整个查询完全可以通过 PHPMyAdmin 工作,显然只返回零行。但是,当我尝试使用 PHP 传递实际值时,它会变得很糟糕:

You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'UPDATE users SET userid='b8aca4b680707453fa4dfe1bf1d0fddb' WHERE userid='8' at line 2

没有其他错误 - 我不知道它是什么。提前致谢,

山姆

4

3 回答 3

7

mysql_query() 不支持多个查询。您必须一次执行一个查询。见http://php.net/manual/en/function.mysql-query.php

于 2012-09-10T16:16:17.607 回答
2

文档中:

mysql_query() sends a unique query (multiple queries are not supported)

您不能通过 php 向数据库发送一系列查询。考虑在您的实际数据库上编写一个过程,然后从 php.ini 调用它。

此外,这些功能已被弃用。您应该考虑使用不同的库。 PDO是一种流行的方法。

于 2012-09-10T16:17:15.330 回答
1

我在处理事务时通常使用的想法是这样的(半伪代码):

try {
    // First of all, let's begin a transaction
    $db->beginTransaction();

    // A set of queries ; of one fails, an exception should be thrown
    $db->query('first query');
    $db->query('second query');
    $db->query('third query');

    // If we arrive here, it means that no exception was thrown
    // i.e. no query has failed ; and we can commit the transaction
    $db->commit();
} catch (Exception $e) {
    // An exception has been thrown
    // We must rollback the transaction
    $db->rollback();
}

请注意,有了这个想法,如果查询失败,则必须抛出异常:

PDO 可以做到这一点,具体取决于您如何配置它 请参阅 PDO::setAttribute 和 PDO::ATTR_ERRMODE 和 PDO::ERRMODE_EXCEPTION 否则,使用其他一些 API,您可能必须测试用于执行查询的函数的结果,并且自己抛出异常。

不幸的是,这并不涉及魔法:您不能只是在某处放置指令并自动完成事务:您仍然必须具体说明必须在事务中执行哪组查询。

例如,很多时候,在事务之前(开始之前)会有几个查询,在事务之后(在提交或回滚之后)会有另外几个查询;无论事务中发生(或不发生)什么,您都希望执行这些查询。

==================================================== ===

您也可以使用 mysqli 来实现相同的目的

例如

<?php
$link = mysqli_connect("localhost", "my_user", "my_password", "world");

/* check connection */
if (mysqli_connect_errno()) {
    printf("Connect failed: %s\n", mysqli_connect_error());
    exit();
}

/* disable autocommit */
mysqli_autocommit($link, FALSE);

mysqli_query($link, "CREATE TABLE myCity LIKE City");
mysqli_query($link, "ALTER TABLE myCity Type=InnoDB");
mysqli_query($link, "INSERT INTO myCity SELECT * FROM City LIMIT 50");

/* commit insert */
mysqli_commit($link);
于 2012-09-10T16:20:12.177 回答