2012-06-14 27 views
33

Estoy tratando de escribir una función para determinar si una cadena es una fecha/hora usando PHP. Básicamente una fecha/hora válida se vería así:Función para verificar si una cadena es una fecha

2012-06-14 01:46:28 

Obviamente, sin embargo es completamente dinámico cualquiera de los valores pueden cambiar, pero siempre debe ser en forma de XXXX-XX-XX XX:XX:XX, ¿cómo puedo escribir una expresión regular para comprobar si esta patrón y retorno verdadero si coincide.

+2

Posible dupli cate: http://stackoverflow.com/questions/37732/what-is-the-regex-pattern-for-datetime-2008-09-01-123545 – Fabian

Respuesta

56

Si ese es su cadena, simplemente intente analizarlo:

if (DateTime::createFromFormat('Y-m-d G:i:s', $myString) !== FALSE) { 
    // it's a date 
} 
+4

Esto no es una solución. Tengo 'DateTime :: createFromFormat ('m/d/Y', '10/38/2013 ')' y eso produce un objeto DateTime válido y no falso. La fecha se convierte en 'object (DateTime) # 39 (3) { [" date "] => string (19)" 2013-11-07 23:45:55 " [" timezone_type "] => int (3) ["zona horaria"] => cadena (3) "UTC" } ' – cj5

+1

Yikes. Me encanta cómo estas cosas nunca se señalan en la documentación ... ¿No has encontrado una alternativa? (Edit: [Formato de fecha] (http://www.php.net/manual/en/datetime.formats.date.php) documentos que puede sobre/subdesbordamiento meses y los días, pero no da una pista de lo hacer cuando * no * quiere ese comportamiento. – Joey

+0

Está bien, no hay solución o configuración para anular esto en PHP. Noté una solución en las respuestas http://stackoverflow.com/a/19666600/486863 – cj5

4

Utilizo esta función como un parámetro para la función PHP filter_var.

  • Comprueba si hay fechas en yyyy-mm-dd hh:mm:ss formato
  • Rechaza fechas que coinciden con el patrón, pero todavía no válidos (por ejemplo, Abr 31)

function filter_mydate($s) { 
    if (preg_match('@^(\d\d\d\d)-(\d\d)-(\d\d) (\d\d):(\d\d):(\d\d)[email protected]', $s, $m) == false) { 
     return false; 
    } 
    if (checkdate($m[2], $m[3], $m[1]) == false || $m[4] >= 24 || $m[5] >= 60 || $m[6] >= 60) { 
     return false; 
    } 
    return $s; 
} 
+5

^(\ d {4} (?: \ - \ d {2}) {2} \ d {2} (?: \: \ D {2}) {2}) $ – kioopi

1

yo no usaría una expresión regular para esto, sino que simplemente dividir la cadena y comprobar que la fecha es válida:

list($year, $month, $day, $hour, $minute, $second) = preg_split('%(|-|:)%', $mydatestring); 
if(!checkdate($month, $day, $year)) { 
    /* print error */ 
} 
/* check $hour, $minute and $second etc */ 
20

Aquí es un enfoque diferente sin necesidad de utilizar una expresión regular:

function check_your_datetime($x) { 
    return (date('Y-m-d H:i:s', strtotime($x)) == $x); 
} 
+1

Esto es genial, gracias –

0

Si su corazón está puesto sobre el uso de expresiones regulares entonces txt2re.com siempre es un buen recurso:

<?php 

    $txt='2012-06-14 01:46:28'; 
    $re1='((?:2|1)\\d{3}(?:-|\\/)(?:(?:0[1-9])|(?:1[0-2]))(?:-|\\/)(?:(?:0[1-9])|(?:[1-2][0-9])|(?:3[0-1]))(?:T|\\s)(?:(?:[0-1][0-9])|(?:2[0-3])):(?:[0-5][0-9]):(?:[0-5][0-9]))'; # Time Stamp 1 

    if ($c=preg_match_all ("/".$re1."/is", $txt, $matches)) 
    { 
     $timestamp1=$matches[1][0]; 
     print "($timestamp1) \n"; 
    } 

?> 
2

Aunque esto ha aceptado una respuesta, no se va a trabajar eficazmente en todos casos. Por ejemplo, pruebo la validación de fecha en un campo de formulario que tengo usando la fecha "10/38/2013", y obtuve un objeto fecha válido devuelto, pero la fecha fue lo que PHP llamó "desbordada", por lo que "10/38/2013 "se convierte" 11/07/2013 ". Tiene sentido, pero ¿deberíamos simplemente aceptar la fecha reformada o forzar a los usuarios a ingresar la fecha correcta? Para aquellos de nosotros que somos nazis de validación de formularios, podemos usar esta corrección sucia: https://stackoverflow.com/a/10120725/486863 y simplemente devolver falso cuando el objeto arroje esta advertencia.

La otra solución sería hacer coincidir la fecha de la cadena con la formateada, y comparar los dos por el mismo valor. Esto parece tan sucio. Oh bien. Tal es la naturaleza de PHP dev.

0
function validateDate($date, $format = 'Y-m-d H:i:s') 
{  
    $d = DateTime::createFromFormat($format, $date);  
    return $d && $d->format($format) == $date; 
} 

función fue copiado de esta answer o php.net

0

Si tiene respuesta PHP 5.2 de Joey no funcionará.Es necesario extender la clase DateTime de PHP:

class ExDateTime extends DateTime{ 
    public static function createFromFormat($frmt,$time,$timezone=null){ 
     $v = explode('.', phpversion()); 
     if(!$timezone) $timezone = new DateTimeZone(date_default_timezone_get()); 
     if(((int)$v[0]>=5&&(int)$v[1]>=2&&(int)$v[2]>17)){ 
      return parent::createFromFormat($frmt,$time,$timezone); 
     } 
     return new DateTime(date($frmt, strtotime($time)), $timezone); 
    } 
} 

y que se puede utilizar esta clase sin problemas:

ExDateTime::createFromFormat('d.m.Y G:i',$timevar); 
11

En caso de que no conoce el formato de fecha:

/** 
* Check if the value is a valid date 
* 
* @param mixed $value 
* 
* @return boolean 
*/ 
function isDate($value) 
{ 
    if (!$value) { 
     return false; 
    } 

    try { 
     new \DateTime($value); 
     return true; 
    } catch (\Exception $e) { 
     return false; 
    } 
} 

var_dump(isDate('2017-01-06')); // true 
var_dump(isDate('2017-13-06')); // false 
var_dump(isDate('2017-02-06T04:20:33')); // true 
var_dump(isDate('2017/02/06')); // true 
var_dump(isDate('3.6. 2017')); // true 
var_dump(isDate(null)); // false 
var_dump(isDate(true)); // false 
var_dump(isDate(false)); // false 
var_dump(isDate('')); // false 
var_dump(isDate(45)); // false 
+0

var_dump (isDate ('ahora')); // true var_dump (isDate ('tomorrow')); // true var_dump (isDate ('yesterday')); // true – Pocketsand

+1

¡Cuidado con los enteros! isDate ('123456') == true y isDate (123456) == true – Pocketsand

0

Una versión más pequeña, podría ayudar:

if(strtotime($date_string)){ 
    // it's in date format 
} 
Cuestiones relacionadas