Подтвердить что ты не робот

Регулярное выражение для принудительного применения сложных паролей, соответствующих 3 из 4 правил

У меня есть следующие критерии для создания регулярного выражения для пароля, который соответствует следующим правилам:

  • Пароль должен содержать 8 символов (это я могу сделать: -)).

Затем пароль должен содержать символы не менее трех из следующих 4 правил:

  • Верхний регистр
  • Нижний регистр
  • Номера
  • Неабелевое число

Я могу заставить выражение соответствовать всем этим правилам со следующим выражением:

/^(?=.*\d)(?=.*[a-z])(?=.*[A-Z])(?=.[\W]).{8,}$/

Но я борюсь с тем, как сделать это таким образом, что ему нужно только решить любые 3 из 4 правил.

Может ли кто-нибудь помочь мне с этим?

4b9b3361

Ответ 1

Не используйте одно регулярное выражение для его проверки.

if (password.length < 8)
  alert("bad password");
var hasUpperCase = /[A-Z]/.test(password);
var hasLowerCase = /[a-z]/.test(password);
var hasNumbers = /\d/.test(password);
var hasNonalphas = /\W/.test(password);
if (hasUpperCase + hasLowerCase + hasNumbers + hasNonalphas < 3)
  alert("bad password");

Если вы должны использовать одно регулярное выражение:

^(?:(?=.*[a-z])(?:(?=.*[A-Z])(?=.*[\d\W])|(?=.*\W)(?=.*\d))|(?=.*\W)(?=.*[A-Z])(?=.*\d)).{8,}$

Это регулярное выражение не оптимизировано для эффективности. Он строится на A·B·C + A·B·D + A·C·D + B·C·D с некоторой факторизацией. Структура:

^
(?:
    (?=.*[a-z])       # 1. there is a lower-case letter ahead,
    (?:               #    and
        (?=.*[A-Z])   #     1.a.i) there is also an upper-case letter, and
        (?=.*[\d\W])  #     1.a.ii) a number (\d) or symbol (\W),
    |                 #    or
        (?=.*\W)      #     1.b.i) there is a symbol, and
        (?=.*\d)      #     1.b.ii) a number ahead
    )
|                     # OR
    (?=.*\W)          # 2.a) there is a symbol, and
    (?=.*[A-Z])       # 2.b) an upper-case letter, and
    (?=.*\d)          # 2.c) a number ahead.
)
.{8,}                 # the password must be at least 8 characters long.
$

Ответ 2

Вы можете написать действительно сложное регулярное выражение для этого. Вместо этого Id предлагает писать четыре различных регулярных выражения, по одному для каждого правила и тестировать их один за другим, подсчитывая, сколько из них соответствует. Если три из четырех сделали, примите пароль.

Ответ 3

Вы можете использовать следующее Regex:

(^(?=.*\d)(?=.*[a-z])(?=.*[A-Z]).*$)?(^(?=.*\d)(?=.*[a-z])(?=.*[@#$%^&+=]).*$)?(^(?=.*\d)(?=.*[A-Z])(?=.*[@#$%^&+=]).*$)?(^(?=.*[a-z])(?=.*[A-Z])(?=.*[@#$%^&+=]).*$)?

При минимальной длине пароля 8 и максимальной длине 32 вы можете использовать следующее Regex:

(^(?=.*\d)(?=.*[a-z])(?=.*[A-Z]).{8,32}$)?(^(?=.*\d)(?=.*[a-z])(?=.*[@#$%^&+=]).{8,32}$)?(^(?=.*\d)(?=.*[A-Z])(?=.*[@#$%^&+=]).{8,32}$)?(^(?=.*[a-z])(?=.*[A-Z])(?=.*[@#$%^&+=]).{8,32}$)?

Ответ 4

Id предлагает выполнить проверку отдельно, а затем просто суммируя количество совпадений.

(Я бы также не использовал регулярное выражение ни в одном из них, но это только мой персональный POV, а именно, что они препятствуют читаемости и обычно представляют собой код с однократной записью)