використовуючи awk з умовами значення стовпця


108

Я вивчаю awk з мови програмування AWK, і у мене є проблема з одним із прикладів.

Якщо я хотів надрукувати $ 3, якщо $ 2 дорівнює значенню (наприклад 1), я використовував цю команду, яка працює добре:

awk '$2==1 {print $3}' <infile> | more

Але коли я замінюю 1 іншими критеріями пошуку (наприклад findtext), команда не працює:

awk '$1== findtext {print $3}' <infile> | more

Він не повертає жодного виводу, і я впевнений, що у файлі введення існує "findtext".

Я також спробував це, але це не працює:

awk '$1== "findtext" {print $3}' <infile> | more

Ось мій тестовий файл під назвою 'test', і він містить 9 рядків та 8 полів, розділених пробілом:

1 11 0.959660297 0 0.021231423 -0.0073 -0.0031 MhZisp
2 14 0.180467091 0.800424628 0 0.0566 0.0103 ClNonZ
3 19 0.98089172 0 0 -0.0158 0.0124 MhNonZ
4 15 0.704883227 0.265392781 0.010615711 -0.0087 -0.0092 MhZisp
5 22 0.010615711 0.959660297 0.010615711 0.0476 0.0061 ClNonZ
6 23 0.715498938 0 0.265392781 -0.0013 -0.0309 Unkn
7 26 0.927813163 0 0.053078556 -0.0051 -0.0636 MhZisp
8 44 0.55626327 0.222929936 0.201698514 0.0053 -0.0438 MhZisp
9 31 0.492569002 0.350318471 0.138004246 0.0485 0.0088 ClNonZ

Ось що я зробив і результат:

$awk '$8 == "ClNonZ" {print $3}' test 

$ grep ClNonZ test 
2 14 0.180467091 0.800424628 0 0.0566 0.0103 ClNonZ
5 22 0.010615711 0.959660297 0.010615711 0.0476 0.0061 ClNonZ
9 31 0.492569002 0.350318471 0.138004246 0.0485 0.0088 ClNonZ

Я очікую побачити це - це 3 долари, які мають "ClNonZ" у своїх 8 доларах.

0.180467091 
0.010615711 
0.492569002

Не знаю, чому команда awk нічого не повернула. Будь-які думки?


Вам потрібно навести значення рядка "findtext", інакше це назва змінної
evil otto

Я спробував подвійні цитати з "findtext", але він не працює .. тому це мене турбує
user1687130

1
"Не працює" нічого не говорить нам. Покажіть нам точні входи, точний код, очікуваний вихід та фактичний вихід.
чепнер

Відповіді:


128

Якщо ви шукаєте певний рядок, поставте навколо нього лапки:

awk '$1 == "findtext" {print $3}'

Інакше awk припустить, що це ім'я змінної.


Я спробував це, але це не працює, я не знаю, чому. Я двічі перевірив грепом, і текст був там. :(
користувач1687130

1
@ user1687130, я думаю, вам знадобиться показати нам якийсь приклад і очікуваний вихід.
Карл Норум

1
Ви впевнені, що ваші дані розділені між собою. Чи можуть деякі з цих просторів бути вкладками? Спробуйте використати awk для відлуння одного поля. Чи awk '{ print $8 }'дає вам те, чого ви очікували?
Роб Девіс

1
Це може бути пов'язано з AWKреалізацією (перевірте це awk --version), подивіться на мою відповідь, вона працює GAWKі в тому, і в MAWKіншому.
arutaku

Це не працює, коли ми використовуємо подвійні лапки навколо сценарію awk. Likeawk "$1 == \"findtext\" {print $3}"
Thirupathi Thangavel

33

Цей метод використовує regexp, він повинен працювати:

awk '$2 ~ /findtext/ {print $3}' <infile>

Дякую, я шукав спосіб за допомогою awk знайти regex на $ NF без використання диявольських методів та grep ^^
Thibault Loison

20

Залежно від AWKреалізації, ви використовуєте ==це нормально чи ні.

Ви пробували ~?. Наприклад, якщо ви хочете, щоб 1 долар був "привіт":

awk '$1 ~ /^hello$/{ print $3; }' <infile>

^означає $ 1 початок, і $це $ 1 кінець.


4
Усі програми awk підтримують і "==", і "~".
Ед Мортон

2
@EdMorton - ОС X awkне вдалося співставитись ==, але це вдалося ~.
jww

2
@jww Не вдалося співставити що з чим? Вони еквівалентні: $1 == "hello"та $1 ~ /^hello$/. Ніколи не слід робити так, $1 ~ "^hello$"як показано у цій відповіді, оскільки це використовується рядок у контексті regexp, і таким чином awk повинен перетворити рядок у регулярний вираз, перш ніж використовувати його, і це має побічні ефекти (man awk).
Ед Мортон


2

Моя версія awk 3.1.5.

Так, вхідний файл розділений пробілом, немає вкладок.

Відповідно до відповіді arutaku, ось що я спробував, що спрацювало:

awk '$8 ~ "ClNonZ"{ print $3; }' test  
0.180467091
0.010615711
0.492569002


$ awk '$8 ~ "ClNonZ" { print $3}' test  
0.180467091
0.010615711
0.492569002

Що не вийшло (я не знаю чому і, можливо, через мою awk версію :),

$awk '$8 ~ "^ClNonZ$"{ print $3; }' test
$awk '$8 == "ClNonZ" { print $3 }' test

Дякую всім за відповіді, коментарі та допомогу!


9
Це не має нічого спільного з вашою версією awk. Ви створили свій тестовий файл у Windows, так що б який інструмент ви не використовували, додали контрольні Ms до кінця кожного рядка, щоб було останнє поле у ​​кожному рядку ClNonZ<control-M>, а не ClNonZтому порівняльне порівняння порівняння RE, виконане з grep або "~ "в awk знаходить це, але порівняння рівності не робить.
Ед Мортон

2
Так, має сенс. Я спробував $ dos2unix test, а потім використав "==" для заміни "~", і він працює. Дякую за пояснення!
користувач1687130

-3

будь ласка, спробуйте це

echo $VAR | grep ClNonZ | awk '{print $3}';

або

echo cat filename | grep ClNonZ | awk '{print $3}';

На жаль, ця відповідь насправді не використовує синтаксис Awk, який користувач спеціально попросив!
Асфанд Казі
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.