数据库错误:[mysqli.mysqli]:用户已经有超过'max_user_connections'活动连接

时间:2009-04-22 14:31:27

标签: php mysql database mysqli

我的网站每天只有大约100人,但是当我以用户身份登录时收到此错误消息:

Warning: mysqli::mysqli() [mysqli.mysqli]: (42000/1203): User mexautos_Juan already has more than 'max_user_connections' active connections in /home/mexautos/public_html/kiubbo/data/model.php on line 26

Warning: mysqli::query() [mysqli.query]: Couldn't fetch mysqli in /home/mexautos/public_html/kiubbo/data/model.php on line 87
Query failed 

我刷新页面几次,现在还可以,但由于我没有那么多用户,我怀疑我的代码中出现了错误,我应该在哪里寻找它?

THX

编辑:这是模型文件:

<?php
/* 

    Model is the base class from which the other
    model classes will be derived. It offers basic
    functionality to access databases

*/ 
require_once($_SERVER['DOCUMENT_ROOT'].'/config.php'); 
require_once(SITE_ROOT.'includes/exceptions.php'); 

class Model {

    private $created;
    private $modified;

    static function getConnection()
    {
        /* 

            Connect to the database and return a
            connection or null on failure

        */

        $db = new mysqli (DB_HOST, DB_USER, DB_PASS, DB_NAME);
        if(!$db) {
            echo mysql_error();
            throw new Exception('Could not connect to database', EX_NO_DATABASE_CONNECTION);
        }

        return $db;

    }

    static function execSQL($query)
    {
    /*
            Execute a SQL query on the database
            passing the tablename and the sql query.
            Returns the resultset
    */


        $db = null;
        $results = null;
        //echo "query is $query";

        try
        {
            $db = Model::getConnection();
            $results = $db->query($query);
            if(!$results) {
                throw new Exception('Query failed', EX_QUERY_FAILED );
            }
        }
        catch(Exception $e)
        {
            /*  errors are handled higher in the
                    object hierarchy
            */

            throw $e;
        }

        Model::closeConnection($db);

        return $results;
    }

    static function execSQl2($query)
    {
    /*
            Execute a SQL query on the database
            passing the tablename and the sql query.
            Returns the LAST_INSERT_ID
    */


        $db = null;
        $lastid = null;
        //echo "query is $query";

        try
        {
            $db = Model::getConnection();
            $results = $db->query($query);
            if(!$results) {
                throw new Exception('Query failed', EX_QUERY_FAILED );
            }
            $lastid = $db->insert_id;
        }
        catch(Exception $e)
        {
            /*  errors are handled higher in the
                    object hierarchy
            */

            throw $e;
        }

        Model::closeConnection($db);

        return $lastid;
    }

    function delete($table, $id, $conditions = '')
    {
        $query = "delete from $table where id = $id";
        try
        {
            $db = Model::getConnection();
            $results = Model::execSQL($query);
            if(!$results){
                throw new Exception('Could not delete this object', EX_DELETE_ERROR);
            }
            return $results->num_rows;
        }

        catch(Exception $e)
        {
            throw $e;
        }
    }

    static function closeConnection($db)
    {
        $db->close();
    }

    function getCreated()
    {
        return $this->created;
    }

    function setCreated($value)
    {
        $this->created = $value;
    }

    function getModified()
    {
        return $this->modified;
    }

    function setModified($value)
    {
        $this->modified = $value;
    }



}

?>

4 个答案:

答案 0 :(得分:5)

您网站上打开数据库连接的每次点击都使用相同的数据库用户名和密码。您的数据库设置限制了每个用户的连接数,并且您将超过该最大值。

查看此MySQL手册页:http://dev.mysql.com/doc/refman/5.0/en/user-resources.html

您的模型类不是很好,因为它似乎是在每个单独的查询上打开和关闭数据库连接。这是一种非常糟糕的资源使用,因为打开和关闭连接很昂贵。我会在你的模型对象上编写一个析构函数,它调用$ db-&gt; close(),并改变getConnection()来打开一次连接,然后每次都返回它。这意味着将您的模型类转换为非静态用法,但在数据库上这样做会更容易。

无论如何,所有连接和断开连接都可能使得MySQL有连接备份,并且在达到最大用户限制之前它们没有足够快地清除。

答案 1 :(得分:2)

查看处理数据库连接的代码。你在用游泳池吗?您使用的是PHP数据库抽象框架之一吗?

您是否在每个数据库访问中处理连接?如果是这样,您正在寻找未明确释放/关闭数据库连接的代码。 (如果你这样做的话,我建议看一下这样的文章:http://www.devshed.com/c/a/PHP/Database-Abstraction-With-PHP/

答案 2 :(得分:1)

这应该解决你的问题,但我没有测试它。差异:如果第一次调用getConnection(),则建立并存储连接。其余时间,使用已建立的连接。

我删除了closeConnection中的操作,因为这会使第一次更改无效。尽管从execSQL中删除closeConnection调用会更好。

通常(据我所知),当脚本终止时,数据库连接会自动关闭(只要mysqli不支持持久性)。但是在完成所有数据库之后,最好手动调用(工作)closeConnection。

<?php
class Model {

    private $created;
    private $modified;

    private static $db = false;

    static function getConnection()
    {
        /* 

            Connect to the database and return a
            connection or null on failure

        */


        if (self::$db === false) {
            self::$db = new mysqli (DB_HOST, DB_USER, DB_PASS, DB_NAME);
        }

        if(!self::$db) {
            echo mysql_error();
            throw new Exception('Could not connect to database', EX_NO_DATABASE_CONNECTION);
        }

        return self::$db;
    }

    static function closeConnection()
    {
        // self::$db->close();
    }

    // moar functions (...)
}

?>

和...我建议使用现有的数据库访问抽象层来避免将来出现此类问题。

答案 3 :(得分:1)

这里有两个问题;一个加剧了另一个。

@zombat发现了一个更大的问题:您不需要为每个查询连接和断开连接。虽然MySQL具有快速设置和拆卸周期,但它浪费了其他资源。在代码的设置步骤中打开连接一次更有意义,然后对每个查询重复使用连接,直到页面结束。我建议为mysqli对象使用一个实例变量。

(如果您的结构中有多个数据库,哪一个依赖于对象,那么您需要增强数据库处理程序以跟踪它已打开的数据库连接,以便它只打开它所需的数据库。但是这是一个更高级的主题,大多数人都不需要这样做。)

这里播放的“其他资源”是MySQL连接。如果mysqli正在创建持久连接,它实际上不会关闭与MySQL的连接(事实上,它也应该重新使用连接,所以你甚至不会遇到这个问题,但我离题了)。 MySQL超时此类连接的默认值是几个小时,所以你可能遇到了这个限制。如果你在SHOW PROCESSLIST中看到数百个“休眠”线程,那么这就是正在发生的事情。要更改的参数是wait_timeoutmax_connections也可能太低了。

但我建议您先修复数据库处理程序。