PHP preg_match с регулярным выражением: только одиночные дефисы и пробелы между словами

Как я могу разрешить одиночные дефисы и одиночные пробелы только внутри слов, но не в начале или в конце слов?

if(!preg_match('/^[a-zA-Z0-9\-\s]+$/', $pg_tag))
    {
        $error = true;
        echo '<error elementid="pg_tag" message="TAGS - only alphanumbers and hyphens are allowed."/>';
    }

Я не хочу принимать эти данные ниже

---stack---over---flow---
stack-over-flow- stack-over-flow2
   stack    over    flow

но только они приемлемы,

stack-over-flow stack-over-flow2 stack-over-flow3
stack over flow
stacoverflow

Спасибо.


person laukok    schedule 08.02.2011    source источник
comment
Вероятно, было бы проще нормализовать ввод, то есть сократить все последовательные пробелы и тире до одного и удалить их с начала и с конца.   -  person Felix Kling    schedule 08.02.2011
comment
@Felix Kling: спасибо за эту идею! как нормализовать ввод?   -  person laukok    schedule 08.02.2011


Ответы (3)


$aWords = array(
    'a',
    '---stack---over---flow---',
    '   stack    over    flow',
    'stack-over-flow',
    'stack over flow',
    'stacoverflow'
);

foreach($aWords as $sWord) {
    if (preg_match('/^(\w+([\s-]\w+)?)+$/', $sWord)) {
        echo 'pass: ' . $sWord . "\n";
    } else {
        echo 'fail: ' . $sWord . "\n";
    }
}

И вывод:

pass: a
fail: ---stack---over---flow---
fail:    stack    over    flow
pass: stack-over-flow
pass: stack over flow
pass: stacoverflow

Разбивка регулярного выражения:

^             # Match from the very beginning of the string
(             # Start Group
    \w+       # At least one "word" character
    (         # Start Subgroup
       [\s-]  # A single space or a dash
       \w+    # At least one "word" character
    )?        # End Subgroup is optional
)+            # End group - allow it multiple times
$             # Match until the very end of the string
person enobrev    schedule 08.02.2011
comment
Верно, хотя оператор никогда не говорил, что это должно пройти. - person enobrev; 08.02.2011
comment
aa проходит, но a не проходит. Я не вижу твоей логики. - person Tim Cooper; 08.02.2011
comment
@tim-cooper И теперь это исправлено. Моя логика заключается в том, что не было никакого способа узнать, что одна строка символов должна пройти в соответствии с примерами того, что должно пройти или не пройти. - person enobrev; 08.02.2011

Что касается комментариев: другая идея состоит в том, чтобы «нормализовать» ввод, т.е. сократить все последовательные пробелы и тире до одного и удалить их с начала и с конца:

$pg_tag = preg_replace(array('/\s+/', '/-+/'), 
                       array(' ', '-'), 
                       trim($pg_tag, ' -'));

Ссылка: preg_replace, trim

person Felix Kling    schedule 08.02.2011

Чтобы принять символ или цифру с пробелом между ними. В случае только пробела он не проходит тест или не принимается только.

Это регулярное выражение принимает строку типа «Домашняя работа», но не работает с «».

var pattern = /^(?=.*\S)[.+a-z0-9A-Z-.#:!*$^_|?` ]+$/;
     return pattern.test(value);

Это возвращает true, если строка принята.

person Vineet Jain    schedule 28.10.2014