Communities

Writing
Writing
Codidact Meta
Codidact Meta
The Great Outdoors
The Great Outdoors
Photography & Video
Photography & Video
Scientific Speculation
Scientific Speculation
Cooking
Cooking
Electrical Engineering
Electrical Engineering
Judaism
Judaism
Languages & Linguistics
Languages & Linguistics
Software Development
Software Development
Mathematics
Mathematics
Christianity
Christianity
Code Golf
Code Golf
Music
Music
Physics
Physics
Linux Systems
Linux Systems
Power Users
Power Users
Tabletop RPGs
Tabletop RPGs
Community Proposals
Community Proposals
tag:snake search within a tag
answers:0 unanswered questions
user:xxxx search by author id
score:0.5 posts with 0.5+ score
"snake oil" exact phrase
votes:4 posts with 4+ votes
created:<1w created < 1 week ago
post_type:xxxx type of post
Search help
Notifications
Mark all as read See all your notifications »
Q&A

Welcome to Software Development on Codidact!

Will you help us build our independent community of developers helping developers? We're small and trying to grow. We welcome questions about all aspects of software development, from design to code to QA and more. Got questions? Got answers? Got code you'd like someone to review? Please join us.

How to use grep to print only specific word from a string

+7
−0

I have a variable that contains a string:

$CCSR = "branches/features/arm_and_musl"

I want to get only the part after the last /. In this case it's "arm_and_musl" but it can be anything.

So something like:

def dirname= sh " echo $CCSR | grep ????? "

But the main issue is that it can be anything. It is "branches/features" this time, though it will always be in this format, with only two forward slashes.

I'm not sure what to put here so that only the part I want is passed to the variable.

Could you please suggest any solutions for this?

History
Why does this post require moderator attention?
You might want to add some details to your flag.
Why should this post be closed?

0 comment threads

3 answers

You are accessing this answer with a direct link, so it's being shown above all other answers regardless of its score. You can return to the normal view.

+7
−0

While I agree with Dirk Herrmann that basename is the right tool for the job, I think it is still worthwhile to know how to do it with grep, because you might one day encounter a sufficiently similar situation that is not covered by a specialised tool.

The question can be broken into two questions:

  1. What regular expression matches exactly the part after the last slash.

  2. How to get grep to only output the part that matches, instead of the whole line.

For the first question, note that the part after the last slash is characterized by the fact that it doesn't contain a slash (otherwise the previous slash would not have been the last one) and extends to the end of the line.

To match a character that's anything but a slash, you can use [^/]. You want to match any number of them, so put a star afterwards: [^/]*. But you only want to match it if it extends to the end of line (so you do not match e.g. the branches in your example), this is done by terminating the regular expression with a dollar sign. So the complete regular expression you need is [^/]*$

For the second question, the option -o does exactly what you need. From the man page:

       -o, --only-matching
              Print  only  the  matched  (non-empty) parts of a matching line,
              with each such part on a separate output line.

So putting it all together, we get the desired behaviour:

$ echo "branches/features/arm_and_musl" | grep -o '[^/]*$'
arm_and_musl
History
Why does this post require moderator attention?
You might want to add some details to your flag.

1 comment thread

Thank you soo much (2 comments)
+8
−0

grep is not the right tool for your case. You can use basename:

basename a/b/c   -->   c
basename a/b/c/  -->   c

or, in your case

basename branches/features/arm_and_musl  -->  arm_and_musl

which, within a sh script, you would use as

dirname=`basename "$CCSR"`

Note, that if you need to call sh explicitly, you have to give the -c command line argument to sh if you want to pass commands on the command line.

History
Why does this post require moderator attention?
You might want to add some details to your flag.

1 comment thread

Thank you sooo much, I had no idea about "basename" I'm writing this down (1 comment)
+5
−0

I wouldn't use grep. As the other answers already said, it's not the right tool for this job.

Considering your specific case (fields separated by /), basename is the most straighforward way, as stated in Dirk's answer. I just want to provide some other alternatives.


cut

The cut command can separate the input in fields and allows you to get only the ones you need. If you know for sure that the inputs will always be branches/features/something, then you can use / as separator and get the third field:

echo $CCSR | cut -d / -f 3

Or, if you don't know how many fields there will be, there's a little trick to get the last one:

echo $CCSR | rev | cut -d / -f 1 | rev

rev reverses the input, so first I reverse it, get the first field and reverse it again. The result will be the last field.

awk

The awk command works in a similar way: we tell it what the separator is and can get specific fields. The difference is that it has a better way to get the last one, by using the predefined variable NF:

echo $CCSR | awk -F / '{print $NF}'

Although basename is the most straighforward solution for your case, please note thatcut and awk are more generic, as they can work with different separators and allows you to get specific fields (not only the last one).

History
Why does this post require moderator attention?
You might want to add some details to your flag.

1 comment thread

Thank you sooooo much for such a detailed answer I reallyy appreciate it (1 comment)

Sign up to answer this question »