Javascript не соответствует str в кавычках - PullRequest
1 голос
/ 08 июля 2011

Я пишу регулярное выражение в javascript , которое заменяет пробелы , за исключением , когда:

  1. Некоторый специфический синтаксис перед пробела
  2. Он заключен в одинарные двойные кавычки (экранированные кавычки в кавычках исключены)

Теперь у меня большая часть работы.Он соответствует всем шаблонам, которые не имеют определенного синтаксиса перед пробелом, однако я застрял с частью цитаты.

return str.replace(/(function|new|return|var)?\s/g, function($0, $1) {
    return $1 ? $0 : '';
});

Я провел довольно небольшое тестирование, но я просто могуне понять это.Заранее спасибо.

Ответы [ 2 ]

2 голосов
/ 08 июля 2011

Вы можете использовать:

var str = "foo  \"b a \\\" r\" new y 'l o l' foo lol; var x = new 'fo \\' o' ";

var result = str.replace(/(function|new|return|var)?\s+(?=(?:[^\\"']|\\.)*(?:(?:"(?:[^\\"]|\\.)*"|'(?:[^\\']|\\.)*'))*(?:[^\\"']|\\.)*$)/gm,
function($0, $1) { return $1 ? $0 : ''; });

См. http://jsfiddle.net/qCeC4/

Часть Lookahead в форме Perl /x:

s/
\s+
(?=
    (?:[^\\"']|\\.)*
    (?:
        (?:
            "(?:[^\\"]|\\.)*"
            |
            '(?:[^\\']|\\.)*'
        )
    )*
    (?:[^\\"']|\\.)*$
)
//xmg;

Примечание: Как я уже говорил, это не очень хороший способ для анализа JS, и он будет разбит на комментарии, цитирование регулярных выражений и кто знает что еще.

Примечание 2: Забыл добавить, что это работает только для "действительного" цитирования, все цитаты должны быть закрыты.

1 голос
/ 08 июля 2011

Мои предложения:

  • имитировать взгляд назад в javascript (хотя этот хак может не быть идеальным).

  • использовать парсер рекурсивного спуска (возможно, antlr)?

  • Или вручную написать код, чтобы сделать это за вас.Ниже приведен мой первый черновой вариант того, что я думаю (есть еще псевдокод):

function go(str) {
    var quoteStart, quoteEnd, quotedRanges, from, retval;
    quotedRanges = []; //quotedRanges holds the indexes inclusively within which nothing should be changed because it's quoted.


    quoteStart = str.indexOf('"');

    if( quoteStart > -1 ) {
        from = quoteStart;
        while (from !== false) {
            quoteEnd = str.indexOf('"', from);

            if (quoteEnd == -1) { //There is an unmatched quote. We pretend that it is closed off at the end of the string.
                quoteEnd = str.len;
                from = false;
            } else if(str.charAt(quoteEnd - 1) == "\\") {
                from = quoteEnd;
            } else { //we found the ending quote index.
                from = false;
            }
        }
        quotedRanges.push([quoteStart, quoteEnd]);
    }


    retval = str.replace(/(function|new|return|var)?\s/g, function($0, $statement) {
        if($0 within on of quotedRanges)
            return $0;
        return $statement ? $0 : '';
    });
    return retval;
}

assert(1, go("") == "");
assert(2, go("function ") == "function ");
assert(3, go(" ") == "");
assert(4, go('" "') == '" "');
assert(5, go('" ') == '" ');
assert(6, go('"x x"') == '"x x"');
assert(6, go('"new x"') == '"new x"');
assert(7, go(' "x x"') == '"x x"');
assert(8, go("' '") == "' '");
assert(9, go("' \\' '") == "' \\' '");


function assert(num, statement) {
    if(!statement) {
        document.write('test #' + num + ' failed! <br/>');
    }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...