// string_data.h
/* Copyright 2010 10gen Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License, version 3,
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see .
*
* As a special exception, the copyright holders give permission to link the
* code of portions of this program with the OpenSSL library under certain
* conditions as described in each individual source file and distribute
* linked combinations including the program with the OpenSSL library. You
* must comply with the GNU Affero General Public License in all respects
* for all of the code used other than as permitted herein. If you modify
* file(s) with this exception, you may extend this exception to your
* version of the file(s), but you are not obligated to do so. If you do not
* wish to do so, delete this exception statement from your version. If you
* delete this exception statement from all source files in the program,
* then also delete it in the license file.
*/
#pragma once
#include // for min
#include
#include
#include
#include
namespace mongo {
/**
* A StringData object wraps a 'const std::string&' or a 'const char*' without copying its
* contents. The most common usage is as a function argument that takes any of the two
* forms of strings above. Fundamentally, this class tries go around the fact that string
* literals in C++ are char[N]'s.
*
* Notes:
*
* + The object StringData wraps around must be alive while the StringData is.
*
* + Because std::string data can be used to pass a substring around, one should never assume a
* rawData() terminates with a null.
*/
class StringData {
public:
/** Constructs an empty std::string data */
StringData()
: _data(NULL), _size(0) {}
/**
* Constructs a StringData, for the case where the length of std::string is not known. 'c'
* must be a pointer to a null-terminated string.
*/
StringData( const char* str )
: _data(str), _size((str == NULL) ? 0 : std::strlen(str)) {}
/**
* Constructs a StringData explicitly, for the case where the length of the std::string is
* already known. 'c' must be a pointer to a null-terminated string, and len must
* be the length that strlen(c) would return, a.k.a the index of the terminator in c.
*/
StringData( const char* c, size_t len )
: _data(c), _size(len) {}
/** Constructs a StringData, for the case of a string. */
StringData( const std::string& s )
: _data(s.c_str()), _size(s.size()) {}
/**
* Constructs a StringData explicitly, for the case of a literal whose size is known at
* compile time.
*/
struct LiteralTag {};
template
StringData( const char (&val)[N], LiteralTag )
: _data(&val[0]), _size(N-1) {}
/**
* Returns -1, 0, or 1 if 'this' is less, equal, or greater than 'other' in
* lexicographical order.
*/
int compare(StringData other) const;
/**
* note: this uses tolower, and therefore does not handle
* come languages correctly.
* should be use sparingly
*/
bool equalCaseInsensitive( StringData other ) const;
void copyTo( char* dest, bool includeEndingNull ) const;
StringData substr( size_t pos, size_t n = std::numeric_limits::max() ) const;
//
// finders
//
size_t find( char c , size_t fromPos = 0 ) const;
size_t find( StringData needle ) const;
size_t rfind( char c, size_t fromPos = std::string::npos ) const;
/**
* Returns true if 'prefix' is a substring of this instance, anchored at position 0.
*/
bool startsWith( StringData prefix ) const;
/**
* Returns true if 'suffix' is a substring of this instance, anchored at the end.
*/
bool endsWith( StringData suffix ) const;
//
// accessors
//
/**
* Get the pointer to the first byte of StringData. This is not guaranteed to be
* null-terminated, so if using this without checking size(), you are likely doing
* something wrong.
*/
const char* rawData() const { return _data; }
size_t size() const { return _size; }
bool empty() const { return size() == 0; }
std::string toString() const { return std::string(_data, size()); }
char operator[] ( unsigned pos ) const { return _data[pos]; }
/**
* Functor compatible with std::hash for std::unordered_{map,set}
* Warning: The hash function is subject to change. Do not use in cases where hashes need
* to be consistent across versions.
*/
struct Hasher {
size_t operator() (StringData str) const;
};
//
// iterators
//
typedef const char* const_iterator;
const_iterator begin() const { return rawData(); }
const_iterator end() const { return rawData() + size(); }
private:
const char* _data; // is not guaranted to be null terminated (see "notes" above)
size_t _size; // 'size' does not include the null terminator
};
inline bool operator==(StringData lhs, StringData rhs) {
return (lhs.size() == rhs.size()) && (lhs.compare(rhs) == 0);
}
inline bool operator!=(StringData lhs, StringData rhs) {
return !(lhs == rhs);
}
inline bool operator<(StringData lhs, StringData rhs) {
return lhs.compare(rhs) < 0 ;
}
inline bool operator<=(StringData lhs, StringData rhs) {
return lhs.compare(rhs) <= 0;
}
inline bool operator>(StringData lhs, StringData rhs) {
return lhs.compare(rhs) > 0;
}
inline bool operator>=(StringData lhs, StringData rhs) {
return lhs.compare(rhs) >= 0;
}
std::ostream& operator<<(std::ostream& stream, StringData value);
} // namespace mongo
#include "mongo/base/string_data-inl.h"