如何知道MySQL表最近一次更新?

在我的页脚,我想添加一些东西,如“上次更新xx/xx/200x”,这个日期是最后一次某个mySQL表已更新。

最好的方法是什么?是否有检索最近更新日期的函数?每次需要这个值时,我都应该访问数据库吗?

346909 次浏览

当查询不可用时,将查询缓存在全局变量中。

创建一个网页,在更新缓存时强制重新加载缓存。

将对重新加载页面的调用添加到部署脚本中。

在MySQL的后续版本中,你可以使用information_schema数据库来告诉你何时更新了另一个表:

SELECT UPDATE_TIME
FROM   information_schema.tables
WHERE  TABLE_SCHEMA = 'dbname'
AND TABLE_NAME = 'tabname'

这当然意味着打开到数据库的连接。


另一种选择是每当MySQL表更新时“触摸”一个特定的文件:

数据库更新:

  • O_RDRW模式打开时间戳文件
  • close它再次

或者

  • 使用touch()(相当于PHP中的utimes()函数)来更改文件时间戳。

页面显示:

  • 使用stat()读取文件修改时间。

我没有information_schema数据库,使用mysql版本4.1.16,所以在这种情况下,你可以查询这个:

SHOW TABLE STATUS FROM your_database LIKE 'your_table';

它将返回这些列:

| Name      | Engine | Version | Row_format | Rows | Avg_row_length 
| Data_length | Max_data_length | Index_length | Data_free | Auto_increment
| Create_time | Update_time | Check_time | Collation
| Checksum | Create_options | Comment |

正如你所看到的,有一个名为:"Update_time"的列,它显示了your_table的最后更新时间。

这就是我所做的,希望能有所帮助。

<?php
mysql_connect("localhost", "USER", "PASSWORD") or die(mysql_error());
mysql_select_db("information_schema") or die(mysql_error());
$query1 = "SELECT `UPDATE_TIME` FROM `TABLES` WHERE
`TABLE_SCHEMA` LIKE 'DataBaseName' AND `TABLE_NAME` LIKE 'TableName'";
$result1 = mysql_query($query1) or die(mysql_error());
while($row = mysql_fetch_array($result1)) {
echo "<strong>1r tr.: </strong>".$row['UPDATE_TIME'];
}
?>
我会创建一个触发器,捕捉所有更新/插入/删除,并在自定义表中写入时间戳,类似于 表名|时间戳

只是因为我不喜欢直接读取db服务器的内部系统表的想法

对于最近的表更改列表,使用这个:

SELECT UPDATE_TIME, TABLE_SCHEMA, TABLE_NAME
FROM information_schema.tables
ORDER BY UPDATE_TIME DESC, TABLE_SCHEMA, TABLE_NAME

只需从文件系统中获取修改过的文件日期。用我的语言来说就是:

 tbl_updated = file.update_time(
"C:\ProgramData\MySQL\MySQL Server 5.5\data\mydb\person.frm")

输出:

1/25/2013 06:04:10 AM

如果您正在运行Linux,您可以使用inotify来查看表或数据库目录。inotify可以从PHP, node.js, perl和大多数其他语言中获得。当然,你必须安装inotify或让你的ISP安装它。很多ISP不会这么做。

我很惊讶没有人建议跟踪每行最后更新时间:

mysql> CREATE TABLE foo (
id INT PRIMARY KEY
x INT,
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
ON UPDATE CURRENT_TIMESTAMP,
KEY (updated_at)
);


mysql> INSERT INTO foo VALUES (1, NOW() - INTERVAL 3 DAY), (2, NOW());


mysql> SELECT * FROM foo;
+----+------+---------------------+
| id | x    | updated_at          |
+----+------+---------------------+
|  1 | NULL | 2013-08-18 03:26:28 |
|  2 | NULL | 2013-08-21 03:26:28 |
+----+------+---------------------+


mysql> UPDATE foo SET x = 1234 WHERE id = 1;

这将更新时间戳,尽管我们没有在UPDATE中提到它。

mysql> SELECT * FROM foo;
+----+------+---------------------+
| id | x    | updated_at          |
+----+------+---------------------+
|  1 | 1235 | 2013-08-21 03:30:20 | <-- this row has been updated
|  2 | NULL | 2013-08-21 03:26:28 |
+----+------+---------------------+

现在你可以查询MAX():

mysql> SELECT MAX(updated_at) FROM foo;
+---------------------+
| MAX(updated_at)     |
+---------------------+
| 2013-08-21 03:30:20 |
+---------------------+
不可否认,这需要更多的存储空间(TIMESTAMP每行4字节) 但是MySQL 5.7.15版本之前的这适用于InnoDB表,而INFORMATION_SCHEMA.TABLES.UPDATE_TIME没有

不知道你会不会对这个感兴趣。在mysql和客户端之间使用mysqlproxy,并使用lua脚本根据有趣的表变化更新memcached中的键值update,DELETE,INSERT是我最近做的解决方案。如果包装器支持php中的钩子或触发器,这可能会更容易。到目前为止,没有一个包装器这样做。

最简单的方法是检查磁盘上表文件的时间戳。例如,您可以在数据目录下检查

cd /var/lib/mysql/<mydatabase>
ls -lhtr *.ibd

这将为您提供所有表的列表,其中的表是最后一次修改该表的时间、最早的时间。

操作系统级别分析:

查找DB在磁盘上的存储位置:

grep datadir /etc/my.cnf
datadir=/var/lib/mysql

检查最近的修改

cd /var/lib/mysql/{db_name}
ls -lrt

应该适用于所有数据库类型。

虽然有一个公认的答案,但我不认为它是正确的答案。这是实现所需的最简单的方法,但即使已经在InnoDB中启用(实际上文档告诉你,你仍然应该得到NULL…),如果你读取MySQL 文档,即使在当前版本(8.0)使用UPDATE_TIME也不是正确的选择,因为:

当服务器重新启动或当 表被赶出InnoDB数据字典缓存。< / p >

如果我理解正确(现在不能在服务器上验证),时间戳在服务器重新启动后会重置。

至于真正的(和,嗯,昂贵的)解决方案,你有Bill Karwin的解决方案与CURRENT_TIMESTAMP,我想提出一个不同的,这是基于触发器(我正在使用那个)。

首先创建一个单独的表(或者可能有其他可以用于此目的的表),它将像存储全局变量(这里是时间戳)一样工作。您需要存储两个字段——表名(或者您想保留在这里作为表id的任何值)和时间戳。在你拥有它之后,你应该用这个表id +开始日期来初始化它(NOW()是一个很好的选择:))。

现在,你移动到你想要观察的表,并使用以下或类似的过程添加触发器AFTER INSERT/UPDATE/DELETE:

CREATE PROCEDURE `timestamp_update` ()
BEGIN
UPDATE `SCHEMA_NAME`.`TIMESTAMPS_TABLE_NAME`
SET `timestamp_column`=DATE_FORMAT(NOW(), '%Y-%m-%d %T')
WHERE `table_name_column`='TABLE_NAME';
END

我在phpMyAdmin中按名称做了一个列:update-at,并在我的代码(nodejs)中从Date()方法获得当前时间。表中的每一个变化,这一列都保存变化的时间。

a)它会显示你所有的表和最近的更新日期

SHOW TABLE STATUS FROM db_name;

然后,你可以进一步询问具体的表格:

SHOW TABLE STATUS FROM db_name like 'table_name';

b)在上面的例子中,你不能在'Update_time'上使用排序,但是使用SELECT你可以:

SELECT * FROM information_schema.tables WHERE TABLE_SCHEMA='db_name' ORDER BY UPDATE_TIME DESC;

要进一步询问特定的表:

SELECT * FROM information_schema.tables WHERE TABLE_SCHEMA='db_name' AND table_name='table_name' ORDER BY UPDATE_TIME DESC';

我让它在本地工作,但不是在我的公共网站的共享主机上(我想是配股问题)。

SELECT last_update FROM mysql.innodb_table_stats WHERE table_name = 'yourTblName';


'2020-10-09 08:25:10'

MySQL 5.7.20-log在win8.1上

和其他人一样,但是我使用了一些条件,以节省时间:

SELECT
UPDATE_TIME,
TABLE_SCHEMA,
TABLE_NAME
FROM
information_schema.tables
WHERE
1 = 1
AND UPDATE_TIME > '2021-11-09 00:00:00'
AND TABLE_SCHEMA = 'db_name_here'
AND TABLE_NAME not in ('table_name_here',)
ORDER BY
UPDATE_TIME DESC,
TABLE_SCHEMA,
TABLE_NAME;