Симон Моурир дал этот пример :
object o = null;
DateTime d = (DateTime)o; // NullReferenceException
, где unboxing преобразование (литье) из object
(или из одного из классов System.ValueType
или System.Enum
или из типа интерфейса) - тип значения (кроме Nullable<>
) сам по себе дает NullReferenceException
.
В другом направлении конверсия бокса из a Nullable<>
, которая имеет HasValue
, равную false
, на ссылочный тип, может дать ссылку null
, которая затем может привести к NullReferenceException
. Классический пример:
DateTime? d = null;
var s = d.ToString(); // OK, no exception (no boxing), returns ""
var t = d.GetType(); // Bang! d is boxed, NullReferenceException
Иногда бокс происходит по-другому. Например, с помощью этого не общего метода расширения:
public static void MyExtension(this object x)
{
x.ToString();
}
следующий код будет проблематичным:
DateTime? d = null;
d.MyExtension(); // Leads to boxing, NullReferenceException occurs inside the body of the called method, not here.
Эти случаи возникают из-за специальных правил, используемых во время выполнения при боксе Nullable<>
экземпляров.
Лучше сделать разбивку на / и проверить все отдельные части. Но если вы действительно хотите использовать регулярное выражение, вы можете попробовать следующее:
#\A(?:(?:(?:(?:0?[13578])|(1[02]))/31/(19|20)?\d\d)|(?:(?:(?:0?[13-9])|(?:1[0-2]))/(?:29|30)/(?:19|20)?\d\d)|(?:0?2/29/(?:19|20)(?:(?:[02468][048])|(?:[13579][26])))|(?:(?:(?:0?[1-9])|(?:1[0-2]))/(?:(?:0?[1-9])|(?:1\d)|(?:2[0-8]))/(?:19|20)?\d\d))\Z#
Объяснение:
\A # start of string
(?: # group without capture
# that match 31st of month 1,3,5,7,8,10,12
(?: # group without capture
(?: # group without capture
(?: # group without capture
0? # number 0 optionnal
[13578] # one digit either 1,3,5,7 or 8
) # end group
| # alternative
(1[02]) # 1 followed by 0 or 2
) # end group
/ # slash
31 # number 31
/ # slash
(19|20)? #numbers 19 or 20 optionnal
\d\d # 2 digits from 00 to 99
) # end group
|
(?:(?:(?:0?[13-9])|(?:1[0-2]))/(?:29|30)/(?:19|20)?\d\d)
|
(?:0?2/29/(?:19|20)(?:(?:[02468][048])|(?:[13579][26])))
|
(?:(?:(?:0?[1-9])|(?:1[0-2]))/(?:(?:0?[1-9])|(?:1\d)|(?:2[0-8]))/(?:19|20)?\d\d)
)
\Z
Я объяснил первую часть, оставив остальное как упражнение.
Это соответствует одной недействительной дате: 02/29/1900, но подходит для любых других дат между 01/01/1900 и 12/31/2099
Лучшее, что вы можете сделать с регулярным выражением, - это проверка формата, например. что-то вроде:
[0-1][0-9]/[0-3][0-9]/[0-9]{2}(?:[0-9]{2})?
Все, что не может быть надежно выполнено без какого-либо слова даты. Срок действия даты зависит от того, является ли это високосный год или нет, например.
, так что вы хотите, чтобы регулярное выражение соответствовало как mm / dd / yy
^((0?1?1){1}|(0?1?2){1}|([0]?3|4|5|6|7|8|9))\/((0?1?2?3?1){1}|(0?1?2?(2|3|4|5|6|7|8|9|0))|(30))\/[1-90]{4}$
, это регулярное выражение будет точно соответствовать вашему желанию в этом формате. mm / dd / yy a не будет проверять фальшивую дату вы можете проверить регулярное выражение на regex101 , которое вы можете проверить для дат 12/30/2040 и 09/09/2020 и что вы хотите для этого формата. Я думаю, что это также кратчайшее регулярное выражение, которое вы можете найти для этого формата
Лучше, чем сумасшедшее огромное Regex (предполагая, что это для проверки, а не сканирования):
require 'date'
def valid_date?( str, format="%m/%d/%Y" )
Date.strptime(str,format) rescue false
end
И как редакционная статья в сторону: Eww! Зачем вам использовать такой ужасно разбитый формат даты? Перейдите для ISO8601, YYYY-MM-DD
, который является действующим международным стандартом, имеет последовательный порядок частей и также сортирует лексикографически.
Для MM-DD-YYYY
вы можете использовать следующее регулярное выражение. Он будет работать в течение високосных лет и будет соответствовать правильным датам, если только год не будет превышать 2099.
(?:(09|04|06|11)(\/|-|\.)(0[1-9]|[12]\d|30)(\/|-|\.)((?:19|20)\d\d))|(?:(01|03|05|07|08|10|12)(\/|-|\.)(0[1-9]|[12]\d|3[01])(\/|-|\.)((?:19|20)\d\d))|(?:02(\/|-|\.)(?:(?:(0[1-9]|1\d|2[0-8])(\/|-|\.)((?:19|20)\d\d))|(?:(29)(\/|-|\.)((?:(?:19|20)(?:04|08|12|16|20|24|28|32|36|40|44|48|52|56|60|64|68|72|76|80|84|88|92|96))|2000))))
Матчи выписки в http://regexr.com/
Или вы просто используете Date.parse "some random date"
. Вы получите исключение ArgumentException, если он не выполнит разбор (=> Дата недействительна).
См., Например, http://santoro.tk/mirror/ruby-core/classes/Date.html#M000644
Вот код, который вы можете использовать :), попробуйте и скажите мне:
^([0-2][0-9]|(3)[0-1])(\/)(((0)[0-9])|((1)[0-2]))(\/)\d{4}$