function inputIsValid(value) {
var r = /(^[0-9]{4}$)|(^(?:(?:[X]{0,2}(?:[I](?:[XV]?|[I]{0,2})?|(?:[V][I]{0,3})?))|(?:[X]{3}[I]{0,3}))\-[A-Z]{2}$)/ig;
return value.match(r);
}
Это будет соответствовать либо 4-значный вход, или римские цифры (в диапазоне 1 - 33), а затем тире и двух букв.
Чтобы объяснить регулярное выражение, ниже расширенный источник с комментариями:
// Test for a 4-digit number
( // Start required capturing group
^ // Start of string
[0-9]{4} // Test for 0-9, exactly 4 times
$ // End of string
) // End required capturing group
| // OR
// Test for Roman Numerals, 1 - 33, followed by a dash and two letters
( // Start required capturing group
^ // Start of string
(?: // Start required non-capturing group
// Test for 1 - 29
(?: // Start required non-capturing group
// Test for 10, 20, (and implied 0, although the Romans did not have a digit, or mathematical concept, for 0)
[X]{0,2} // X, optionally up to 2 times
(?: // Start required non-capturing group
// Test for 1 - 4, and 9
[I] // I, exactly once (I = 1)
(?: // Start optional non-capturing group
// IV = 4, IX = 9
[XV]? // Optional X or V, exactly once
| // OR
// II = 2, III = 3
[I]{0,2} // Optional I, up to 2 times
)? // End optional non-capturing group
| // OR
// Test for 5 - 8
(?: // Start optional non-capturing group
[V][I]{0,3} // Required V, followed by optional I, up to 3 times
)? // End optional non-capturing group
) // End required non-capturing group
) // End required non-capturing group
| // OR
// Test for 30 - 33
(?: // Start required non-capturing group
// Test for 30
[X]{3} // X exactly 3 times
// Test for 1 - 3
[I]{0,3} // Optional I, up to 3 times
) // End required non-capturing group
) // End required non-capturing group
// Test for dash and two letters
\- // Literal -, exactly 1 time
[A-Z]{2} // Alphabetic character, exactly 2 times
$ // End of string
) // End required capturing group
4-значный номер и замыкающие \-[A-Z]{2}
были (мне) самоочевидны. Мой метод для римских цифр состоял в том, чтобы:
- Открыть Excel Заполнить столбец 1-33.
- Преобразуйте эту колонку в римские цифры (всего 7 разных сортов).
- Проверьте, не отличались ли какие-либо разновидности от 1-33 (их не было).
- Свернутый с перемещением римских цифр в минимальное количество уникальных узоров, которые ограничивали их до 33 (т. Е. «Тогда вы будете считать до тридцати трех, не более, не менее». Тридцать три будут числом, которое вы должны подсчитать , а число подсчетов будет тридцать три. Тридцать четыре не засчитайте, и не посчитайте тридцать два, за исключением того, что тогда вы переходите к тридцати трем. Тридцать пять прямо. »)
- Понял, что до тридцати девяти - это один шаблон (
^(([X]{0,3}([I]([XV]?|[I]{0,2})?|([V][I]{0,3})?)))$
, который был изменен на группы захвата для большей ясности).
- Изменен шаблон, позволяющий до двадцати девяти.
- Добавлен еще один, чтобы разрешить тридцать-тридцать девять.
- Постройте весь шаблон и тест в RegexBuddy (бесценный инструмент для этого материала) против цифр 0 - 20 000 и римских цифр 1 - 150, а затем «-AA».
- Образец работал, поэтому я разместил его (затем схватил еще одну чашку кофе и самоуправлял «атта-мальчика» для завершения того, что, как я думал, было чудесным утром в субботу).
Посторонние скобки, я полагаю, вы имеете в виду группы, не захватывающие захват (?: ...)
. Я много использую для группировки вещей (и здесь очень нужна группировка). Я сделал их не захватывающими, потому что мне не нужно захватывать подгруппы, а только родительские группы (и в этом случае я не думаю, что они действительно должны быть захвачены, но это не мешает сделать это). Делая их не захватывая, они не будут создавать обратные ссылки, которые ускоряют обработку (хотя для одного входа время, полученное, незначительно).
Смотрите этот другой вопрос: http://stackoverflow.com/questions/267399/how -do-you-match-only-valid-roman-numbers-with-a-regular-expression – satoshi
@RobW, это может быть от 1 до 6 символов, поскольку ожидаемое значение равно от I до XXXIII (т.е. от 1 до 33). –