Lỗi trong MySQL khi đặt giá trị mặc định cho DATE hoặc DATETIME


124

Tôi đang chạy MySql Server 5.7.11 và câu này:

updated datetime NOT NULL DEFAULT '0000-00-00 00:00:00'

không làm việc. Đưa ra lỗi:

ERROR 1067 (42000): Invalid default value for 'updated'

Nhưng sau đây:

updated datetime NOT NULL DEFAULT '1000-01-01 00:00:00'

chỉ hoạt động .

Trường hợp tương tự cho DATE.

Như một chú thích phụ , nó được đề cập trong tài liệu MySQL :

Loại DATE được sử dụng cho các giá trị có phần ngày nhưng không có phần thời gian. MySQL truy xuất và hiển thị các giá trị DATE ở định dạng 'YYYY-MM-DD'. Phạm vi được hỗ trợ là '1000-01-01' đến '9999-12-31'.

ngay cả khi họ cũng nói:

Các giá trị DATE, DATETIME hoặc TIMESTAMP không hợp lệ được chuyển đổi thành giá trị “không” của loại thích hợp ('0000-00-00' hoặc '0000-00-00 00:00:00').

Cũng đã tính đến trích dẫn thứ hai từ tài liệu MySQL, bất cứ ai có thể cho tôi biết tại sao nó lại đưa ra lỗi đó không?


11
Tại sao bạn muốn một mặc định rõ ràng là vô nghĩa? Nếu ngày không xác định thì đó chính xác NULLlà những gì dành cho.
Tom H

Lưu ý: Điều này hoạt động trong phiên bản 5.6 trên SQL Fiddle - sqlfiddle.com/#!9/02c98 .
Gordon Linoff

@Karlos kiểm tra câu trả lời được cập nhật.
geeksal

Câu trả lời:


209

Lỗi là do chế độ sql có thể là chế độ nghiêm ngặt theo tài liệu MYSQL 5.7 mới nhất

Tài liệu MySQL 5.7 cho biết :

Chế độ nghiêm ngặt ảnh hưởng đến việc máy chủ có cho phép '0000-00-00' là ngày hợp lệ hay không: Nếu chế độ nghiêm ngặt không được bật, '0000-00-00' được cho phép và việc chèn không tạo ra cảnh báo. Nếu chế độ nghiêm ngặt được bật, '0000-00-00' không được phép và các phần chèn tạo ra lỗi, trừ khi cũng đưa ra BỎ QUA. Đối với CHÈN BỎ QUA và CẬP NHẬT BỎ QUA, cho phép '0000-00-00' và các phần chèn tạo ra cảnh báo.

Để kiểm tra chế độ MYSQL

SELECT @@GLOBAL.sql_mode global, @@SESSION.sql_mode session

Đang tắt chế độ STRICT_TRANS_TABLES

Tuy nhiên, để cho phép định dạng, 0000-00-00 00:00:00bạn phải tắt chế độ STRICT_TRANS_TABLES trong tệp cấu hình mysql hoặc bằng lệnh

Theo lệnh

SET sql_mode = '';

hoặc là

SET GLOBAL sql_mode = '';

Việc sử dụng từ khóa này GLOBALđòi hỏi các nguyên tắc siêu cấp và nó ảnh hưởng đến hoạt động của tất cả các khách hàng kết nối từ đó trở đi

nếu ở trên không hoạt động, hãy truy cập /etc/mysql/my.cnf(theo ubuntu) và nhận xétSTRICT_TRANS_TABLES

Ngoài ra, nếu bạn muốn thiết lập vĩnh viễn chế độ sql lúc khởi động máy chủ sau đó bao gồm SET sql_mode=''trong my.cnftrên Linux hoặc hệ điều hành MacOS. Đối với cửa sổ, điều này phải được thực hiện trong my.initệp.

Ghi chú

Tuy nhiên, chế độ nghiêm ngặt không được bật theo mặc định trong MYSQL 5.6. Do đó, nó không tạo ra lỗi theo tài liệu MYSQL 6 cho biết

MySQL cho phép bạn lưu trữ giá trị "0" của '0000-00-00' dưới dạng "ngày giả". Trong một số trường hợp, điều này thuận tiện hơn so với việc sử dụng giá trị NULL và sử dụng ít không gian dữ liệu và chỉ mục hơn. Để không cho phép '0000-00-00', hãy bật chế độ SQL NO_ZERO_DATE.

CẬP NHẬT

Về vấn đề lỗi như đã nói bởi @ Dylan-Su:

Tôi không nghĩ đây là lỗi mà nó là cách MYSQL được phát triển theo thời gian do đó một số thứ được thay đổi dựa trên cải tiến hơn nữa của sản phẩm.

Tuy nhiên, tôi có một báo cáo lỗi liên quan khác liên quan đến NOW()chức năng

Trường ngày giờ không chấp nhận NOW () mặc định

Một lưu ý hữu ích khác [xem Tự động khởi tạo và cập nhật cho TIMESTAMP và DATETIME ]

Kể từ MySQL 5.6.5, các cột TIMESTAMP và DATETIME có thể được tự động khởi tạo và cập nhật thành ngày và giờ hiện tại (nghĩa là dấu thời gian hiện tại). Trước 5.6.5, điều này chỉ đúng với TIMESTAMP và tối đa một cột TIMESTAMP trên mỗi bảng. Các ghi chú sau đây đầu tiên mô tả việc khởi tạo và cập nhật tự động cho MySQL 5.6.5 trở lên, sau đó là sự khác biệt cho các phiên bản trước 5.6.5.

Cập nhật về NO_ZERO_DATE

Đối với MySQL kể từ ngày 5.7.4, chế độ này không được dùng nữa. Đối với phiên bản trước, bạn phải nhận xét dòng tương ứng trong tệp cấu hình. Tham khảo tài liệu MySQL 5.7 vào NO_ZERO_DATE


5
CẬP NHẬT BỎ QUA là những gì tôi đang tìm kiếm 👍🏻
Afanasii Kurakin

2
Sai lầm. Tôi có STRICT_TRANS_TABLEScho cả phiên bản MySQL, cục bộ và máy chủ của mình. Tuy nhiên, tôi có thể dễ dàng chèn 0000-00-00vào phiên bản cục bộ của mình, nhưng không thể chèn trong phiên bản máy chủ của tôi - lỗi được đưa ra. Tại sao? Vì cấu hình MySQL máy chủ của tôi đã NO_ZERO_DATEđược kích hoạt. Còn địa phương của tôi thì không có.
Màu xanh lá cây,

ok @Green tôi sẽ tìm ra và cập nhật các câu trả lời nếu có
geeksal

4
Trong trường hợp của tôi, 'SET sql_mode = ""' không hoạt động. 'ĐẶT TOÀN CẦU sql_mode = "";' đã làm công việc cho tôi.
arjen Stens

Cũng NO_ZERO_DATEcần được loại bỏ
Preshan Pradeepa

18

Tôi đã gặp lỗi này với WAMP 3.0.6 với MySql 5.7.14.

Giải pháp :

thay đổi dòng 70 (nếu tệp ini của bạn không bị ảnh hưởng) trong c:\wamp\bin\mysql\mysql5.7.14\my.initệp từ

sql-mode= "STRICT_ALL_TABLES,ERROR_FOR_DIVISION_BY_ZERO,NO_ZERO_DATE,NO_ZERO_IN_DATE,NO_AUTO_CREATE_USER"

đến

sql-mode="ERROR_FOR_DIVISION_BY_ZERO,NO_ZERO_DATE,NO_ZERO_IN_DATE,NO_AUTO_CREATE_USER"

và khởi động lại tất cả các dịch vụ.

Điều này sẽ tắt chế độ nghiêm ngặt. Theo tài liệu, "chế độ nghiêm ngặt" có nghĩa là chế độ có hoặc cả hai STRICT_TRANS_TABLEShoặc STRICT_ALL_TABLESđược bật. Các tài liệu cho biết:

"Chế độ SQL mặc định trong MySQL 5.7 bao gồm các chế độ sau: ONLY_FULL_GROUP_BY, STRICT_TRANS_TABLES, NO_ZERO_IN_DATE, NO_ZERO_DATE, ERROR_FOR_DIVISION_BY_ZERO, NO_AUTO_CREATE_USER và NO_ENGINE_SUBSTITUTION."


14

Tôi gặp trường hợp dữ liệu bị trộn giữa NULL và 0000-00-00 cho trường ngày tháng. Nhưng tôi không biết cách cập nhật '0000-00-00' thành NULL, vì

 update my_table set my_date_field=NULL where my_date_field='0000-00-00'

không được phép nữa. Cách giải quyết của tôi khá đơn giản:

update my_table set my_date_field=NULL where my_date_field<'1000-01-01'

bởi vì tất cả các my_date_fieldgiá trị không chính xác (cho dù ngày chính xác hay không) đều có từ trước ngày này.


2
Giải pháp nhanh chóng hoàn hảo. Trên thực tế, bạn cũng có thể sử dụng <'0000-01-01'vì tất nhiên đó là ngày hợp lệ.
Ricky McMaster

5

Cấu hình cú pháp vấn đề

Trên một số phiên bản MYSQL (đã thử nghiệm 5.7. *) Theo hệ thống * nix, bạn nên sử dụng cú pháp sau:

[mysqld]

sql-mode="NO_BACKSLASH_ESCAPES,STRICT_TRANS_TABLE,NO_ENGINE_SUBSTITUTION"

Những điều này sẽ không hoạt động:

gạch ngang không có dấu ngoặc kép

sql-mode=NO_ENGINE_SUBSTITUTION

gạch dưới không có dấu ngoặc kép

sql_mode=NO_ENGINE_SUBSTITUTION

gạch dưới và dấu ngoặc kép

sql_mode="NO_ENGINE_SUBSTITUTION"

Đánh giá đầy đủ hơn về các giá trị cấu hình và chế độ sql:

Cách thiết lập cờ Chế độ Sql vĩnh viễn


5

Đầu tiên hãy chọn phiên hiện tại sql_mode:

SELECT @@SESSION.sql_mode;

Sau đó, bạn sẽ nhận được một cái gì đó giống như giá trị mặc định đó :

'ONLY_FULL_GROUP_BY, STRICT_TRANS_TABLES, NO_ZERO_IN_DATE, NO_ZERO_DATE, ERROR_FOR_DIVISION_BY_ZERO, NO_AUTO_CREATE_USER, NO_ENGINE_SUBSTITUTION'

và sau đó đặt sql_modemà không có 'NO_ZERO_DATE':

SET SESSION sql_mode = 'ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION';

Nếu bạn có tài trợ, bạn cũng có thể làm điều đó cho GLOBAL:

SELECT @@GLOBAL.sql_mode;
SET GLOBAL sql_mode = '...';

4

Chỉ cần thêm dòng: sql_mode = "NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION"

tệp bên trong: /etc/mysql/mysql.conf.d/mysqld.cnf

sau đó sudo service mysql restart


1
Hoạt động cho 5.7.23.
dùng2513149

1
Tôi có thể sẽ đề xuất SELECT @@SESSION.sql_mode;trước và họ xóa NO_ZERO_IN_DATE, NO_ZERO_DATE và STRICT_TRANS_TABLES khỏi những gì nó mang lại cho bạn. Bằng cách đó, bạn giữ lại bất kỳ cài đặt nào khác mà bạn đã bật. Tôi đã có nhiều thứ hơn chỉ là hai mục đó được đặt cho chế độ sql của tôi. Không biết tất cả chúng làm gì, nhưng tôi không muốn mạo hiểm xóa chúng vào thời điểm này.
Radley Bền vững

2

Nó hoạt động cho 5.7.8:

mysql> create table t1(updated datetime NOT NULL DEFAULT '0000-00-00 00:00:00');
Query OK, 0 rows affected (0.01 sec)

mysql> show create table t1;
+-------+-------------------------------------------------------------------------------------------------------------------------+
| Table | Create Table                                                                                                            |
+-------+-------------------------------------------------------------------------------------------------------------------------+
| t1    | CREATE TABLE `t1` (
  `updated` datetime NOT NULL DEFAULT '0000-00-00 00:00:00'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 |
+-------+-------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)

mysql> select version();
+-----------+
| version() |
+-----------+
| 5.7.8-rc  |
+-----------+
1 row in set (0.00 sec)

Bạn có thể tạo SQLFiddle để tạo lại sự cố của mình.

http://sqlfiddle.com/

Nếu nó hoạt động cho MySQL 5.6 và 5.7.8, nhưng không thành công trên 5.7.11. Sau đó, nó có thể là một lỗi hồi quy cho 5.7.11.


1

Để giải quyết vấn đề với MySQL Workbench (Sau khi áp dụng giải pháp ở phía máy chủ):

Xóa SQL_MODE thành TRADITIONAL trong bảng tùy chọn.

nhập mô tả hình ảnh ở đây


1

Câu trả lời này chỉ dành cho MySQL 5.7:

Tốt nhất là không thực sự đặt trống sql_mode, thay vào đó hãy sử dụng trong PHP một biến phiên với:

SET SESSION sql_mode= 'ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION'

Vì vậy, ít nhất bạn giữ các giá trị mặc định khác.

Thật là điên rồ khi tài liệu mysql không rõ ràng, bạn cần xóa các giá trị khử lỗi này trong sql_mode:

NO_ZERO_IN_DATE, NO_ZERO_DATE, tôi hiểu, nhưng trong tương lai, tính năng này sẽ bị ngừng.

STRICT_ALL_TABLES, với điều này, trước khi các tham số sẽ bị bỏ qua, vì vậy bạn cũng cần xóa nó.

Cuối cùng, TRADITIONAL cũng vậy, nhưng tài liệu nói về tham số này: "đưa ra lỗi thay vì cảnh báo" khi chèn giá trị không chính xác vào cột ", với tham số này, ngày có giá trị 0 không được chèn, nhưng không có.

MySQL không thực sự được tổ chức với các tham số và sự kết hợp này.


0

Các kết hợp tùy chọn cho mysql Ver 14.14 Distrib 5.7.18, for Linux (x86_64).

Không ném:

STRICT_TRANS_TABLES + NO_ZERO_DATE

Ném:

STRICT_TRANS_TABLES + NO_ZERO_IN_DATE

Cài đặt của tôi /etc/mysql/my.cnftrên Ubuntu:

[mysqld]
sql_mode = "STRICT_TRANS_TABLES,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION"

0
set global sql_mode = 'STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION';

0

Trong diretory xamp / mysql / bin Mở "my.ini" và thay đổi dòng: Sql_node cho ->

"sql_mode = NO_ENGINE_SUBSTITUTION, NO_ZERO_DATE"

XÓA "NO_ZERO_IN_DATE"


Chào mừng bạn đến với stackoverflow, vui lòng sử dụng định dạng mã thích hợp để cải thiện tính dễ đọc của câu trả lời.
vlizana
Khi sử dụng trang web của chúng tôi, bạn xác nhận rằng bạn đã đọc và hiểu Chính sách cookieChính sách bảo mật của chúng tôi.
Licensed under cc by-sa 3.0 with attribution required.