我在MySQL中有一个包含各种VARCHAR字段的表。我想通过PHP从表单中插入一些用户数据。显然,如果我知道PHP中的字段长度,我可以使用substr()
限制数据长度。但是那种违反DRY(存储在MySQL中的字段长度以及我的PHP脚本中的常量)。有没有办法让我配置一个INSERT,以便它自动切断过长的字符串,而不是失败?
编辑:当我有过长的字符串时,它在PHP / PDO中失败(或至少导致异常)。不知道我在PHP / PDO中必须做什么才能做正确的事。
编辑2:呃。这是错误的做法;即使我在INSERT上运行正常,如果我想检查一个重复的字符串,它也不会正常匹配。答案 0 :(得分:8)
实际上,默认情况下,MySQL会将字符串截断为列宽。它会生成警告,但允许插入。
mysql> create table foo (str varchar(10));
mysql> insert into foo values ('abcdefghijklmnopqrstuvwxyz');
Query OK, 1 row affected, 1 warning (0.00 sec)
mysql> show warnings;
+---------+------+------------------------------------------+
| Level | Code | Message |
+---------+------+------------------------------------------+
| Warning | 1265 | Data truncated for column 'str' at row 1 |
+---------+------+------------------------------------------+
mysql> select * from foo;
+------------+
| str |
+------------+
| abcdefghij |
+------------+
如果设置严格SQL模式,它会将警告变为错误并拒绝插入。
重新评论:SQL模式是MySQL服务器配置。它可能不是导致它的PDO,但另一方面它是可能的,因为任何客户端都可以为其会话设置SQL模式。
您可以使用以下语句检索当前的全局或会话sql_mode值:
SELECT @@GLOBAL.sql_mode;
SELECT @@SESSION.sql_mode;
默认值应为空字符串(不设置模式)。您可以在my.cnf
文件中设置SQL模式,使用mysqld的--sql-mode
选项,或使用SET
语句。
有关SQL模式的完整文档,请参阅http://dev.mysql.com/doc/refman/5.1/en/server-sql-mode.html。
答案 1 :(得分:0)
您可以查询information_schema以获取列的长度,并在插入之前使用该数据进行截断,但这比必要的开销更大。只需关闭此语句的严格SQL模式:
SET @prev_mode = @@sql_mode;
SET sql_mode = '';
INSERT blah....;
SET sql_mode = @prev_mode;
答案 2 :(得分:0)
您可以使用列元数据来检查字符串长度。 (PHP Manual on PDOStatement->getColumnMeta)
以这种方式获取整个表的元数据
$query = $conn->query("SELECT * FROM places");
$numcols = $query->columnCount();
$tablemeta = array();
for ($i = 0; $i < $numcols; $i++) {
$colmeta = $query->getColumnMeta($i);
$tablemeta[$colmeta['name']] = $colmeta;
}