-1

我正在为博客计划一个用 PHP 编码的视图计数。计数器的目的是指示阅读博客文章的频率,例如 StackOverflow 在问题页面上显示“查看了 n 次”。

首先,我想将计数器数据存储在文件中,但通过研究,我改变了主意,将 MySQL 与几个 InnoDB 表一起使用,因为这些表的执行速度比 HDD 查找快,从长远来看更省力,并且原生提供有用的功能。

我遇到的问题是如何每个用户只计算一次?

限制 IP 不是解决方案,因为动态 IP 和网络大多具有与所有连接设备(例如学校、办公室、公共接入点)共享连接的单个 IP。

编辑:

fideloper 的回答启发,我正在考虑基于会话的解决方案。

我已经为会话使用了单独的数据库表,因此我将按照建议将其合并到解决方案中。

这是我到目前为止所得到的(使用Laravel 4和Eloquent ORM):

SQL 表设置:

-- 
-- Sessions Table
-- 
CREATE TABLE sessions (
    id VARCHAR(255) NOT NULL COLLATE 'utf8_unicode_ci',
    payload TEXT NOT NULL COLLATE 'utf8_unicode_ci',
    last_activity INT(11) NOT NULL,

    UNIQUE INDEX sessions_id_unique (id)
) COLLATE='utf8_unicode_ci' ENGINE=InnoDB;

-- 
-- Views Table
-- 
CREATE TABLE post_views (
    id INT(11) UNSIGNED NOT NULL AUTO_INCREMENT,
    post_id INT(11) UNSIGNED NOT NULL,
    sess_id VARCHAR(255) NOT NULL COLLATE 'utf8_unicode_ci',
    viewed_at TIMESTAMP NULL DEFAULT NULL,

    PRIMARY KEY (id)
) COLLATE='utf8_unicode_ci' ENGINE=InnoDB;

-- 
-- Posts Table
-- 
CREATE TABLE posts (
    id INT(11) UNSIGNED NOT NULL AUTO_INCREMENT,
    user_id INT(11) UNSIGNED NOT NULL,
    title VARCHAR(255) NOT NULL COLLATE 'utf8_unicode_ci',
    excerpt TEXT NOT NULL COLLATE 'utf8_unicode_ci',
    slug VARCHAR(255) NOT NULL COLLATE 'utf8_unicode_ci',
    content LONGTEXT NOT NULL COLLATE 'utf8_unicode_ci',
    ...

    views INT(10) UNSIGNED NOT NULL DEFAULT '0',

    PRIMARY KEY (id)
) COLLATE='utf8_unicode_ci' ENGINE=InnoDB;


PHP 视图计数器类:

<?php namespace Services\Posts;
/**
 * Blog Post Views Counter
 *
 * Using Laravel 4 with Eloquent ORM
 */

class ViewsCounter {

    public $post_id;

    /**
     * Sets the post id
     * 
     * @param  int $id
     * @return void
     */
    public function setPostID($id)
    {
        $this->post_id = $id;
    }

    /**
     * Handles the view count
     * 
     * @return bool  True or false on failure
     */
    public function countView()
    {
        // Get the post data from the db
        try
        {
            $post = \Post::findOrFail($this->post_id);
        }
        catch (\Illuminate\Database\Eloquent\ModelNotFoundException $e)
        {
            return false;
        }

        // Get the current session id
        $sess_id = session_id();

        // Check if this view was already counted by 
        // looking up the session id in the views table
        $post_views = \PostViews::where('sess_id', '=', $sess_id)
                            ->where('post_id', '=', $this->post_id)
                            ->remember(10)
                            ->first();

        // The sess_id was not found so count the view
        if (empty($post_views))
        {
            $post_views->post_id = $this->post_id;
            $post_views->sess_id = $sess_id;
            $post_views->viewed_at = date('Y-m-d H:i:s');
            $post_views->save();

            // A cronjob is run every 24 hours to count the views per 
            // post and update the count values in the posts table.
            // The post_views table will be truncated when done.
        }

        return true;
    }

}

建议表示赞赏。

4

1 回答 1

8

计算每个用户的 +1 浏览量是一个有趣的问题,因为您需要将会话考虑在内。

您确实需要对每个会话 ID 进行 +1 计数(因为它会为每个用户重新生成,但更有可能使您每次访问每个用户 +1 视图计数......取决于您的代码)。请注意,这意味着在新会话中再次访问该站点的用户将再次获得查看计数。

如果您需要跟踪会话 ID,您可能需要执行以下操作:

  1. 每个视图添加一个新的数据库行(而不是增加单个值)。通过这种方式,您可以跟踪会话 ID 以及视图计数(这也意味着最终的大表!)
  2. 仅在未找到会话 ID 时添加新行(如果找到,则用户在此访问中的查看次数已 +1)。
  3. 对于一些可扩展性:添加第二个表来跟踪每个视图的访问次数。使用 CRON 或类似工具定期更新此内容。然后,您始终可以写入一个表(针对写入优化表)并从另一个表读取(针对读取进行优化)。
  4. 为了获得额外的可扩展性,请使用第二个表来计算每 24 小时的查看次数。然后您可以从视图计数表中删除视图。通过这种方式,您永远不会将您的观看次数表填得很大,但您可以节省每 24 小时的总观看次数。

也可以随意缩短 24 小时的时间。您可能(将)最终在更新视图计数时计算一些额外的视图计数,并且要与会话 ID 进行比较的会话 ID 被删除。

希望能给你带来一些想法。

于 2013-07-07T19:08:59.363 回答