std::string, subscripting, and npos

M

Marcus Kwok

std::string::npos is described in _TC++PL:SE_ (Section 20.3.4) as the
"all characters" marker. I tried to use it this way, but my program
crashes:

#include <iostream>
#include <string>

int main()
{
std::string s = "hi/hello/now";
std::cout << s << '\n';

for (std::string::size_type i = 0; i != std::string::npos; ++i) {
if (s == '/') {
s = '\\';
}
}

std::cout << s << '\n';
}


If I change "std::string::npos" to "s.length()" then of course the
program works correctly. Why doesn't std::string::npos work in this
context?
 
B

Bob Hairgrove

std::string::npos is described in _TC++PL:SE_ (Section 20.3.4) as the
"all characters" marker. I tried to use it this way, but my program
crashes:

#include <iostream>
#include <string>

int main()
{
std::string s = "hi/hello/now";
std::cout << s << '\n';

for (std::string::size_type i = 0; i != std::string::npos; ++i) {
if (s == '/') {
s = '\\';
}
}

std::cout << s << '\n';
}


If I change "std::string::npos" to "s.length()" then of course the
program works correctly. Why doesn't std::string::npos work in this
context?


std::string::npos is usually just an alias for ((unsigned)(-1)). You
are therefore incrementing i way past the end of the string's length.

std::string::npos is returned by most string functions which involve
some kind of find, and which would normally return the index or
position of the element being sought. Since there can never be such a
position, this value was chosen to indicate "not found" or not
present.
 
V

Victor Bazarov

Marcus said:
std::string::npos is described in _TC++PL:SE_ (Section 20.3.4) as the
"all characters" marker. I tried to use it this way, but my program
crashes:

#include <iostream>
#include <string>

int main()
{
std::string s = "hi/hello/now";
std::cout << s << '\n';

for (std::string::size_type i = 0; i != std::string::npos; ++i) {
if (s == '/') {
s = '\\';
}
}

std::cout << s << '\n';
}


If I change "std::string::npos" to "s.length()" then of course the
program works correctly. Why doesn't std::string::npos work in this
context?


'npos' is a constant usually defined as (size_type)(-1). Its specific
purpose is to indicate "not found" condition when returning from 'find'
members and the "end" position when using 'substr'.

You mistakenly used it as if it is different for every 'string' object.
It's not. IIRC it's a static member.

V
 
M

Mike Wahler

Marcus Kwok said:
std::string::npos is described in _TC++PL:SE_ (Section 20.3.4) as the
"all characters" marker.

Its meaning depends upon context. It's most often used to
indicate 'not found' with e.g. 'find()'.
I tried to use it this way, but my program
crashes:

#include <iostream>
#include <string>

int main()
{
std::string s = "hi/hello/now";
std::cout << s << '\n';

for (std::string::size_type i = 0; i != std::string::npos; ++i) {

'npos' is not a valid index for a string. Even if you use the
meaning 'all characters', what would that value as an index
mean? The book doesn't say 'how many characters' does it? :)
(and even if it did, then the highest valid index would be
'how many, less one' (because first index is 0).)

The valid index values are those integers in the set beginning with
zero and ending with 'string::size() - 1', inclusive.

'size() - 1' can never be equal to 'npos'. That's part of
'npos's definition. The value of 'npos' must be chosen by
an implementation to be some value that is *not* a valid
index. That's how e.g. 'find()' (which returns an index
value), can distinguish between a found element and 'not found'.

Once 'i's value passes 'size() - 1', using it as an index
will give undefined behavior. (The 'at()' member functinon
would throw an exception).

Write this instead:

for (std::string::size_type i = 0; i != s.size(); ++i) {
if (s == '/') {
s = '\\';
}
}

std::cout << s << '\n';
}


If I change "std::string::npos" to "s.length()"


Note that 'size()' and 'length()' are two functions which
have exactly the same effect.
then of course the
program works correctly. Why doesn't std::string::npos work in this
context?

Because you're trying to use it for something other than
it's defined purpose. Like trying to cut a watermelon with
a feather. :)

-Mike
 
M

Mike Wahler

Victor Bazarov said:
'npos' is a constant usually defined as (size_type)(-1). Its specific
purpose is to indicate "not found" condition when returning from 'find'
members and the "end" position when using 'substr'.

You mistakenly used it as if it is different for every 'string' object.
It's not. IIRC it's a static member.

I suspect that Marcus misunderstood after seeing npos used in
a different context, e.g. 'string::substr()'s second parameter
specifies a default value of 'npos', meaning 'all'.

std::string s("hello");
std::string sub(s.substr(0)); // now s == sub

-Mike
 
M

Marcus Kwok

Bob Hairgrove said:
std::string::npos is described in _TC++PL:SE_ (Section 20.3.4) as the
"all characters" marker. I tried to use it this way, but my program
crashes:

#include <iostream>
#include <string>

int main()
{
std::string s = "hi/hello/now";
std::cout << s << '\n';

for (std::string::size_type i = 0; i != std::string::npos; ++i) {
if (s == '/') {
s = '\\';
}
}

std::cout << s << '\n';
}


If I change "std::string::npos" to "s.length()" then of course the
program works correctly. Why doesn't std::string::npos work in this
context?


std::string::npos is usually just an alias for ((unsigned)(-1)). You
are therefore incrementing i way past the end of the string's length.

std::string::npos is returned by most string functions which involve
some kind of find, and which would normally return the index or
position of the element being sought. Since there can never be such a
position, this value was chosen to indicate "not found" or not
present.


Thanks. I was thinking maybe it would be similar to the .end()
iterator (as in, 1 greater than the last element).
 
D

David Harmon

On Fri, 27 Jan 2006 21:06:31 +0000 (UTC) in comp.lang.c++,
(e-mail address removed) (Marcus Kwok) wrote,
std::string::npos is described in _TC++PL:SE_ (Section 20.3.4) as the
"all characters" marker.

That statement refers specifically to the use of npos as an argument
to the string constructor that creates a substring of an existing
string. Your code is not doing anything even slightly resembling
that. Give me a break.
 
D

Duane Hebert

Thanks. I was thinking maybe it would be similar to the .end()
iterator (as in, 1 greater than the last element).

You could always just use an iterator.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
473,995
Messages
2,570,236
Members
46,825
Latest member
VernonQuy6

Latest Threads

Top