[PERL] Вот как оказывается можно писать...
Поэтому, люди, если вы не знаете как сделать простую вещь без изврата, лучше спросите.У тебя спрашивать? А ты часто в форуме бываешь?
пеаришься?
ну да, конечно спрашивай! я всегда рад помочь! только тут почему-то очень мало вопросов по Perl... Обычно C да PHP...
как в цикле использовать значения $1, $2 и т.д. без евала ибо он тормозной что песдетц.
я сделал кое-как, но хотелось бы услышать мнение специалиста по этому поводу(вдруг ему придется копаться в моем коде)
Вообще эти переменные не обязательно использовать. Можно сразу в одну строчку сохранять значения из регулярного выражения, например:
my @strs = ( "Hello, world!" , "Larry Wall" );
foreach my $str (@strs)
{
my ($first, $second) = $str =~ /(\S+)\s+(\S+)/;
print "
first = '$first'
1 = '$1'
second = '$second'
2 = '$2'
";
}
Напечатается вот что:
first = 'Hello,'
1 = 'Hello,'
second = 'world!'
2 = 'world!'
first = 'Larry'
1 = 'Larry'
second = 'Wall'
2 = 'Wall'
В твои переменные $first и $second попадает то же, что и в переменные $1 и $2.
По поводу eval. Я не совсем понимаю, зачем его нужно в твоем случае использовать. Вообще eval юзают аналогично явавской конструкции try{ ... }catch( ... ){ ... }, т.е. в местах, где вызываются потенциально опасные функции (т.е. функции, которые могут упасть). И в случае, если функция в евале падает, то программа, которая этот евал вызвала не падает, а продолжает работать.
А вообще лучше приведи код, по которому у тебя вопрос..
То есть смысла такой:
/$pattern/;
for (1 .. num_columns) {
сделать что-то с $i, ну то есть не с переменной i, а с $1, etc.
}
один из вариантов - написать евал
eval "my \$a=\$$_;";
Но он работает очень медленно(известный факт а это происходит в большом цикле.
поэтому пришлось сделать по-другому, но хотелось бы узнать как правильно.
насчет try/catch, это другой вариант евала, не eval EXPR, а eval BLOCK в основном
юзай @- и @+
а что будет быстрее, @+ и @-, или использовать $^R в теле регэкспа?
Прикольно, да. Решить эту задачу можно двумя намного более простыми способами:Это будет медленнее.
Способ 1 - регулярное выражение:
my ($res) = $str =~ /(\S+)$/;
Способ 2 - можно сразу узнать последний элемнт массива. Не обязательно делать reverse:Раз уж выёбываться, то до конца! Зачем временные переменные?
my @arr = split( / / , $str );
my $res = $arr[-1];
my $res = (split(/ /, $str[-1];
Как будет решаться такая задача в любом нормальном ЯП? Примерно так:
1) двигаясь с конца строки, найти первый непробельный символ
2) двигаясь с этой позиции дальше, найти первый пробел
3) подстрока между позициями из 1 и 2 - искомая
Количество операций равно (длина строки - позиция начала слова если нет финальных пробелов - длина слова.
Теперь смотрим на перл.
Решение с массивом как минимум просматривает всю строку + (что ещё печальнее) делает дохуя не нужных аллокаций памяти и копирования туда кусков строки. А если в строке миллион символов и тысяча слов? Тогда как нам нужно только последнее?
С регэкспом оценить трудоёмкость сложнее, но есть надежда, что такой регэксп должен прорабатывать за О(длина строки при этом лишней памяти не аллоцируя. Так что для достаточно больших строк с большим числом слов регэскп может порулить.
Ты, наверное, думаешь, что на перле нельзя реализовать твой алгоритм?
Я перл не рюхаю, там разве посимвольный доступ к строкам есть? Субстры не предлагать - это скорее всего опять маллоки, опять сосут.
Ну если есть совсем правильный способ - то автор темы - лох бп
Как будет решаться такая задача в любом нормальном ЯП? Примерно так:Выходит C уже не нормальный ЯП.
1) двигаясь с конца строки, найти первый непробельный символ
Я перл не рюхаю, там разве посимвольный доступ к строкам есть?Прикинь!
Если хочешь считать маллоки - пиши на C или ассемблере.
Нормальный, зависит от того, что считать строками. ;-)
ЗЫ Даже для 0-терминэйтед строк, когда с ними делаются какие-то пертурбации, зачастую кем-то уже посчитано, какая у них длина.
ЗЗЫ Упражение специально для Глеба: модифицировать пункты 1-2 "моего алгоритма" на случай 0-терминэйтед строк. Подумать, что принципиально изменилось по отношению к перлу.
что хотел сделать программист, написавший a = exp(b*log(c должно быть понятно любому другому нормальному программисту с полуслова. Даже если в рамках конкретного перла эта запись излишня.
Монжо пример из одной строчки, чото не могу найти как это.
Как будет решаться такая задача в любом нормальном ЯП? Примерно так:Если это решать, как математическую задачку - то да.
1) двигаясь с конца строки, найти первый непробельный символ
2) двигаясь с этой позиции дальше, найти первый пробел
3) подстрока между позициями из 1 и 2 - искомая
А на практике, очень редко бывает, что последнее слово нужно, а предыдущие слова - нет. Поэтому язык должен уметь достаточно понятно и эффективно разбивать строки на слова. Perl в области его типичных применений это умеет неплохо.
Монжо пример из одной строчки, чото не могу найти как это.Это substr. Он может быть и rhs.
Раз уж выёбываться, то до конца! Зачем временные переменные?согласен. еще можно так писать:
code:
my $res = (split(/ /, $str[-1];
my $res = unshift split(/ /, $str);
ржачный трэд. Не поленился прочитать весь на курорте с кпк+гпрс. Кстати, некоторые адекватные вещи порадовали - буду пользовать, спасибо.
Решение с массивом как минимум просматривает всю строкуФункция split позволяет просматривать не всю строку, а только до определенного количества кусочков, например:
my($head,$body,$tail) = split / /, '1 2 3 4 5', 3;
# Here:
# $head = '1'
# $body = '2'
# $tail = '3 4 5'
в данном случае, на первый взгляд, это не поможет, т.к. строку надо просматривать с конца
my ($res) = split( / / , reverse $str, 1 );
как минимум, ты уже допустил неочевидную ошибку, т.к. в исходной постановке не говорилось, что значения состоят из одного символа
имхо, все эти выкрутасы сводят к нулю принцип: написана единожды, используется везде, т.к. все эти выкрутасы приводят к неочевидным ошибкам, когда, кажется, что у автора кода все работает, а при попытках сделать шажок влево/вправо - все рассыпается.
кстати, все вышеприведенные примеры на пустой строке как будут себя вести?
Есть функция, которая получает паттерн регэкспа. Но скобок которые из него нужны конечно всегда разное число, зато есть правила по которым их можно и нужно обрабатывать(независимые от их числа).а от чего тогда зависят эти правила?
Мне кажется, что этот случай довольно сложный для того, чтобы решить его с помощью одного универсального регулярного выражения (даже если это и можно, то получится регулярное выражения с кучей условий, которое тяжеловатo будет понимать).
Я бы в данном случае постарался разбить все возможные варианты обрабатываемой строки на несколько типов (исходя из каких-либо характерных для каждого варианта кусков в строке). А потом определял для каждой строки что это за вариант и применял к нему свое регулярное выражение:
if( /характерное_свойство_варианта_1/ )
{
# регулярное выражение для варианта строки 1
}
elsif( /характерное_свойство_варианта_2/ )
{
# регулярное выражение для варианта строки 2
}
...
хотя мне кажется я не ответил на вопрос. если не сложно, приведи пожалуйста разные варианты строк, а также, что из них нужно вытащить.
значения состоят из одного символасхема такая
split /PATTERN/, EXPR, LIMIT
третий аргумент ( у меня он равен 1) определяет не длину значения, а количество раз, которое нужно применить к строке EXPR регулярное выражение EXP.
если только под словом значение ты подразумевал кусочки, получаемы после разбиения исходной строки...
кстати, все вышеприведенные примеры на пустой строке как будут себя вести?очень просто - переменным присвоятся значения undef. При этом Perl НЕ выругается (только что проверил что, конечно, не очень хорошо. Но вообще это всегда нужно проверять...
так что будет на выходе, если входная строка - '13', для варианта с reverse-ом?
my ($res) = split( / / , reverse $str, 1 );
для $str = 13
получается $res = 31
Ты прав, этот вариант неправильный....
в данном случае тоже можно всю строку не просматривать:А что, reverse $str не просматривает всю сроку? Он ещо и память на копию жрёт наверняка.
my ($res) = split( / / , reverse $str, 1 );
Прикинь! Про субстры я написал на 20 постов выше.
В порицание за невнимательность (а ещо модератор, ай-яй-яй) можешь попробовать заботать исходники перла и доказать мне, что субстр реализован не совсем примитивно, и я не прав. Было бы приятно порадовацо за перл
Type of arg 1 to unshift must be array (not split)
В порицание за невнимательность (а ещо модератор, ай-яй-яй) можешь попробовать заботать исходники перла и доказать мне, что субстр реализован не совсем примитивно, и я не прав.Почему я должен искать в исходниках и что-то доказывать, когда высказал спорное предположение ты, а не я?
P.S. Кстати я не модератор.
я кстати не до конца понимаю почему разговор пошел о производительности перла. Хочешь скорости и оптимальности - пиши на ассемблере
может надо шифт а не аншифт?
Хочешь скорости и оптимальности - пиши на ассемблереДавно писал на ассемблере?
ну я конечно немного утрирую, но смысл вопроса думаю ясен
может надо шифт а не аншифт?если мы хотим последний элемент массива, то нужен именно unshift.
Но конструкция, которую я применил действительно не работает. Перл ругается....
Это странно, потому что мне кажется такое написание вполне естественно...
Извините, конечно, но, если мы говорим о Perl, то тут уж давайте не будем заострять внимание на производительности. Потому что главная сила перла, это не скорость работы перловых скриптов, а скорость их написания.
Кстати, если кому-нибудь интересно, то есть офигенный модуль - Inline.pm.
http://search.cpan.org/search?query=inline&mode=all
Он позволяет объединять совершенно разные языки программирования в перловом скрипете. Это просто офигенно! Потому что места программы, где нужны сложные и большие вычисления можно писать на С++, а разбор файлов и работу со строками - на перле.
Этот модуль вообще позволяем почти все языки программирования использовать (только вот ассемблера там, кажется, нет)
если мы хотим последний элемент массива, то нужен именно unshift.и вовсе не unshift, а pop
Но конструкция, которую я применил действительно не работает. Перл ругается....
Это странно, потому что мне кажется такое написание вполне естественно...
Ботай разницу между list и array: perldoc -q "list and an array".
split создаёт list, в то время как pop и shift требуют array.
Выражаясь иначе, list, возвращаемый split модифицировать нельзя, в то время как shift и pop требуют lvalue (которое можно модифицировать).
What is the difference between a list and an array?
An array has a changeable length. A list does not. An array is something you can push
or pop, while a list is a set of values. Some people make the distinction that a list
is a value while an array is a variable. Subroutines are passed and return lists, you
put things into list context, you initialize arrays with lists, and you foreach across
a list. "@" variables are arrays, anonymous arrays are arrays, arrays in scalar context
behave like the number of elements in them, subroutines access their arguments through
the array @_, and push/pop/shift only work on arrays.
> Type of arg 1 to unshift must be array (not split)[/quote]
Можно ещё анонимный array сделать
my $res = pop @{[split / /, $str ]};
да дада,уншифт добавляет с конца в массив же =)
Есть функция, которая получает паттерн регэкспа. Но скобок которые из него нужны конечно всегда разное число, зато есть правила по которым их можно и нужно обрабатывать(независимые от их числа).Когда-то сам столкнулся с этой проблемой. Помогло следующее: ${$i}.
То есть смысла такой:
/$pattern/;
for (1 .. num_columns) {
сделать что-то с $i, ну то есть не с переменной i, а с $1, etc.
}
один из вариантов - написать евал
eval "my \$a=\$$_;";
Но он работает очень медленно(известный факт а это происходит в большом цикле.
Вот иллюстрация: программа, находящая одно из решений линейного диофантова уравнения с положительными коэффициентами и положительной правой частью (и то и другое передаётся через командную строку). С помощью регулярных выражений. Зацените идею! Идея, к сожалению, не моя, я лишь в учебно-тренировочных целях адаптировал программу на случай произвольного количества неизвестных. Магическую букву 'o' можно заменить на любую другую.
#!/usr/bin/perl -w
sub solve
{
my $b=pop @_;
my @a=@_;
my @x;
my $k;
map {$_--} @a;
my $re='^';
for $k(1..@a)
{
$re.="(o+)\\$k\{".$a[$k-1]."}";
}
$re.='$';
if'o' x $b)=~m/$re/)
{
for $k(1..@a)
{
$x[$k-1]=length ${$k};
}
}
else
{
die "Не могу решить уравнение!\n";
}
return @x;
}
print join ' ', solve(@ARGV "\n");
${$i} не работает с use strict
Я вот подумал, что если use strict совершенно необходим в программе, то ведь можно хулиганские места заключить в блок и вставить no strict... Так что это не очень серьёзное ограничение. Всё же эстетичнее, чем eval.
ничего в программе не понял, но выглядит красиво
Ищется решение уравнения a1*x1+...+a_n*x_n=b (все a и b>0). Формируется строка, состоящая из b букв (в данном примере 'o'). Строка сопоставляется с RE: одна или больше строк 'o....o' длины a_1 (обозначим, как в Перле, 'o' x a_1 затем одна или больше строк 'o....o' длины a_2 ('o' x a_2) и т. д. Все эти фрагменты заключаем в скобки, чтобы запомнить найденное в переменных $1, $2,...
Затем считаем длины $1, $2, ... Это в точности a_1*x_1, a_2*x_2,... Делим на a_1, a_2,... Получаем x_1, x_2,...
Особенности этой программы: необычное применение RE, формирование RE в цикле, и использование в цикле ${$k}, о чём спрашивал .
Вот и всё.
"\\$k"внутри выражения и зачем
'ooooooo'=~m/^(o+)\1{2}(o+)\2{1}$/
То есть (o+ затем то же самое ещё 2=3-1 раза; затем снова (o+ затем то же самое ещё 1=2-1 раз. '\3' в RE — то же самое, что и $3, но предназначен для использования в самом RE. Перл не любит, когда $1, $2,... вставляют в RE.
Я был не совсем точен, описывая словами данный алгоритм. Искомые иксы — это длины строк $1, $2,...
А почему не "o+){" . $a[$k] . "})" или типа того? Вот меня и смутили обратные слеши.
"(?:(o+){" . $a[$k-1] . "})"
И не делать map {$_--} @a.
Ботай разницу между list и array: perldoc -q "list and an array".А как еще можно из листа сделать массив, кроме следующего геморойного варианта:
split создаёт list, в то время как pop и shift требуют array.
my $res = pop @{ [ split(/,/ , "1,2,3") ] };
Т.е., чтобы не далать сначала ссылку, а потом из ссылки массив... Должен же быть более красивый способ
и вовсе не unshift, а popда да. это я что-то глупость написал. извиняйте...
просто этими функциями не часто приходится пользоваться, вот и перепутал
Проще наверное все таки не литерал сплиту передавать,а переменную,тогда с этим проблем не будет.
${$i} не работает с use strictА что это вообще за переменная? или это типа приведения ссылки на массив к массиву:
@{[$aref]}
тольок для скалаяров?
никогда не сталкивался...
Проще наверное все таки не литерал сплиту передавать,а переменную,тогда с этим проблем не будет.ты это имеешь ввиду:
my $var = '1 2 3';
my $head = pop split(/ /, $var);
все равно не работает...
Когда я спрашивал про преобразование листа в массив я хотел избежить такой конструкции:
my $head = pop @{[split(/ /, '1 2 3')]};
my $var = '1 2 3';а так
my $head = pop split(/ /, $var);
my @var=split(/ /,'1 2 3');
my $head = pop @var;
? =)
>> А что это вообще за переменная?
Это способ обратиться к переменной, имя которой содержится в $i. Либо, если $i — ссылка на скаляр, то ${$i} или даже $$i — то, на что она указывает. Перл различает ссылки и нессылки и разбирает это выражение по ситуации.
сделай сначала сплит,засунь это куда нить,а потом от результата уже и попай =)
my $space = ' ';
my $str = $space . 2 + 3;
print "without brackets: '$str'\n";
$str = $space . (2+3);
print "with brackets: '$str'\n";
# output is:
# without brackets: '5'
# with brackets: ' 5'
Разница в том, что без скобок пробел в начале строчки не печатается...
Я долго не мог понять эту ошибку. Фишка вот в чем. перл читает строку без скобок слева направо. т.е. сначала он склеивает $space и цифру 2. Затем он видит операцию сложения и автоматически преобразует строку ' 2' к числу 2 и склдывает. Поэтому-то пробел в первом случае и пропал.
my $two = ' 2'; # space at the begining!
my $three = 3;
my $res = $two + $three;
print "'$res'";
# output is: '5'
Прикольно, да?
а такну так в том и фишка, что я хочу в одну строчку без создания промежуточной ссылки на массив!
хз,а нах тебе? =)
Это способ обратиться к переменной, имя которой содержится в $iа почему тогда не работает?
my $var = 'Hello!';
my $var_name = 'var';
print "${$var_name}";
без strict и warnings он вообще ничего не печатает...
а так, наверное все дело в приоритете =)
А можно ещё давать переменным неприличные имена:
${'Привет!'} = 'Hello!';
my $var_name = 'Привет!';
print "${$var_name}";
print "\n";
хз,а нах тебе? =)ну так просто... красивее и понятнее было бы смотреть на код в одной строчке...
вариант с двумя строчками кода, конечно, понятный и прозрачный:
my @var=split(/ /,'1 2 3');
my $head = pop @var;
Но это ж целые ДВЕ строчки кода на Perl'e! к тому же дополнительная переменная... какое расточительство!
можно, конечно, и без pop. Вот варинт, который предложил :
my $head = (split(/ /, '1 2 3'[-1];
Из-за my в первой строчкеда, заработало... Буду знать. Но пользовать, наверное, не буду. Уж больно андерграундный вариант....
print (2+3)*4;
Да, строчка крайне неудачная. print напечатает 5, вернёт какой-то результат, который без рользы в пустом контексте будет умножен на 4. Нужны скобки.
Однако, в твоем примере перл warning'и выдает (у меня он вообще ничего ни на что не руглася):
print (...) interpreted as function at D:\Programming\Perl\_tmp\test1.pl line 7.
Useless use of multiplication (*) in void context at D:\Programming\Perl\_tmp\test1.pl line 7.
а ошибка понятна. просто функция print думает, что скбочки это не для математичской формулы, а скобочки для нее (для функции print то есть). Поэтому и вылазеет warning про void context. Если написать так:
print 4*(2+3);
то все работает и без ругательств....
ну так в том и фишка, что я хочу в одну строчку без создания промежуточной ссылки на массив!Да дело не в ссылке, а в самом массиве - без него не обойтись, и это логично.
это-то я понял. вопрос исходно стоял, как перейти от листа к массиву без ссылки на массив
print (2+3)*4; # Oops!
When Perl sees this line of code, it prints 5, just as you asked. Then it takes the return value from print, which is 1, and multiplies that times 4. It then throws away the product, wondering why you didn't tell it to do something else with it. And at this point, someone looking over your shoulder says, "Hey, Perl can't do math! That should have printed 20, rather than 5!"
substr($line, $-[$_ + 1], $+[$_ + 1] - $-[$_ + 1])
Начать с фразы "добрый день", закончить фразой "влоушов(dlfeушуоов;;*d=d dkdkjвлваоовал". И говорить, что обе понятны и прекрасны.
Оставить комментарий
sakura
Я программирую на Perl в некой фирме. Часто приходиться копаться в чужих кодах. При этом находятся вещи, которые довольно трудны до понимания обычному смертному.Пример 1:
my $x = ...; # некое число
my $res = exp( $x * log(10) )
Если вы внимательно посмотрите, то увидите, что значение переменной $res - это десять в степени $x. Возведение в степень в Perl пишется очень просто - две звездочки:
$res = 10 ** $x;
Пример 2:
my $str = "1 2 3 4 5"; # Некая строка со значениями, разденными пробелами. Нужно вытищить последний элемент, т.е. в данном случае число 5
my @arr = split( / / , $str );
@arr = reverse @arr;
my $res = $arr[0];
Прикольно, да. Решить эту задачу можно двумя намного более простыми способами:
Способ 1 - регулярное выражение:
my ($res) = $str =~ /(\S+)$/;
Способ 2 - можно сразу узнать последний элемнт массива. Не обязательно делать reverse:
my @arr = split( / / , $str );
my $res = $arr[-1];
P.S. Это происходит из-за неполного знания прекрасного языка Perl. Поэтому, люди, если вы не знаете как сделать простую вещь без изврата, лучше спросите. А то потом довольно тяжело понимать, что там хотел сделать программист, который это написал.