Bootstrap

【C++】 —— string的使用

前言

        string类虽然不在STL的容器中,但string类十分重要,string类是对字符串的存储和相关操作。

        basic_string

        std::basic_string类是C++的一个模版类,它支持多种字符类型。

  1. char :用于表示一个字节的字符,使用ASCII编码。
  2. wchar_t :用于表示宽字符,可以支持更广泛的字符集。
  3. char8_t :用于表示8位的Unicode字符(UTF-8)。
  4. char16_t :用于表示16位的Unicode字符(UTF-16)。
  5. char32_t :用于表示32位的Unicode字符(UTF-32)。

basic_string 实例化出来的4种类型:

string
wstring
u16string
u32string

        string类是basic_string模版类的一个实例,它使用char来实例化basic_string模版类,并使用char_traits和allocator 作为basic_string的默认参数。

一、string类

1、string类是表示字符序列的类。

2、标准的字符串类提供了对此对象的支持,其接口类似于标准字符容器的接口,但是添加了专门用于操作单字节字符字符串的设计特性。

3、string类封装了字符数组,并提供了一系列成员函数来执行字符串的创建、修改、连接、查找、替换等操作。

        string类出现实际上是比STL要早的,后来才分入STL库里面;所以,string一开始设计比较冗余,有很多重复功能的接口;我们没必要记住所有的接口,只需将核心接口记住并熟练使用就好啦;(如果遇到陌生的接口,直接查看文档就好了)。

string类文档

二、string默认成员函数

        2.1、构造函数

先来看一下C++库里面string的构造函数。

1、string();

        这是没有参数的构造函数(也就是默认构造函数),就是在创建string类对象时不需要传参数。

#include<iostream>
#include<string>

using namespace std;
void test()
{
	string s1;
}

int main()
{
	test();
	return 0;
}

2、string(const char* s);

        在创建string类对象时,传字符串(指针/字符数组),进行类对象的初始化。

使用:用字符串来初始化对象。

void test()
{
	string s1; //string();
	string s2("I Love You"); //string(const char* s);
}

3、string(const char* s, size_t n);

        与上面的构造函数不同的时,在传字符串指针的时,多一个参数n。

使用:用字符数组的前n个字符来初始化string类对象。

void test()
{
	string s1; //string();
	string s2("I Love You"); //string(const char* s);
	string s3("I Love You", 6); //string(const char* s, size_t n);
}

可以通过调试来看对象s1,s2,s3的值

4、string(const string& str);

        在创建string类对象,传同为string类类型的对象进行对象的初始化(也就是拷贝构造函数)。

使用:通过同类型的对象来初始化;

void test()
{
	string s1; //string();
	string s2("I Love You"); //string(const char* s);
	string s3("I Love You", 6); //string(const char* s, size_t n);
	string s4(s2); // string(const string& str);
}

5、string(const string& str, size_t pos, size_t len=npos);

        与拷贝构造不同的时,这个构造函数,还需要多传两个参数pos和len;

使用:用str中的第pos个位置后面len个字符进行初始化对象。

void test()
{
	string s1; //string();
	string s2("I Love You"); //string(const char* s);
	string s3("I Love You", 6); //string(const char* s, size_t n);
	string s4(s2); // string(const string& str);
	string s5(s2, 2, 4); //string(const string& str, size_t pos, size_t len=npos);
}

6、string(size_t n, char ch);

        创建string类对象,并将其内容设置成n个ch。

void test()
{
	string s1; //string();
	string s2("I Love You"); //string(const char* s);
	string s3("I Love You", 6); //string(const char* s, size_t n);
	string s4(s2); // string(const string& str);
	string s5(s2, 2, 4); //string(const string& str, size_t pos, size_t len=npos);
	string s6(6, 'H'); //string(size_t n, char ch);
}

7、

        C++文档中最后一个构造函数,涉及到迭代器,这里先不介绍,后面再使用。

        2.2、析构函数

        析构函数没有函数重载,就不过多叙述。

        2.3、赋值运算符重载

1、string& operator= (const string& str);

        使用:这个就是对于两个已经存在的string类类型对象,进行赋值操作。

void test1()
{
	string s1;
	string s2("Hello World");//拷贝构造
	s1 = s2;//赋值运算符重载

}

2、string& operator= (const char* s);

        使用:将一个字符串赋值给一个已经存在的string类类型对象。

void test1()
{
	string s1;
	string s2("Hello World");//拷贝构造
	s1 = s2;//赋值运算符重载

	string s3; 
	s3 = "I Love ";//operator= (const char* s);
}

3、string& operator= (char ch);

        使用:将一个字符赋值给一个已经存在的string类类型对象。

void test1()
{
	string s1;
	string s2("Hello World");//拷贝构造
	s1 = s2;//赋值运算符重载

	string s3; 
	s3 = "I Love ";//operator= (const char* s);

	string s4;
	s4 = 'c';//operator= (char ch);
}

调试看一下赋值结果:

三、string其他成员函数

        3.1、Capacity

这里就看几个常用的(size、lengh、resize、capacity、clear和empty)。

1、size*

        size就是返回string对象中的字符(数据)个数(也就是字符串的长度)。

2、lengh

        lengh和size一样,都是计算string对象中字符串的长度。

3、resize*

        resize是设置string对象中字符串的内容,(如果n要小于size就将字符串数据个数设为n)如果n大于size,(空间不够就扩容)设置size后面的字符内容。如果给了参数c就设置成 c;如果没有给参数c,就默认设置成'\0'。

4、capacity*

        capacity 返回string对象中可用空间的大小(就像动态顺序表中capacity一样)。

5、clear*

        clear是清理string中字符串的内容,让string对象中字符串变为空,并修改对象中size的值。

6、empty*

        empty判断string对象是否为空;为空就返回true,否则返回false。

7、reserve*

        

        reserve是设置string的空间大小,如果传参n比capacity大就扩容;如果小,不一定会缩小容量。

void test2()
{
	string s1("I Love You !!!");
	//size 返回s1中字符串数据个数
	cout << "size = " << s1.size() << endl;
	//lengh 返回s1中字符串长度
	cout << "lengh = " << s1.length() << endl;
	//capacity 返回s1中可用空间大小
	cout << "capacity = " << s1.capacity() << endl;
	//设置s1中数据个数
	s1.resize(10);
	cout << s1 << endl;
	s1.resize(20, 'x');
	cout << s1 << endl;
	//清理s1中字符串内容,使其变成空
	s1.clear();
	cout << s1 << endl;
	//empty 判断s1中字符串是否为空
	cout << s1.empty() << endl;
}

        3.2、Element access 数据访问

        Element access 就是元素访问,(访问字符串中的数据)。

        这里就主要看 operator[] 和at;

        除了这样访问元素,还可以通过迭代器就行访问(下面在迭代器内容进行使用)。

1、operator[]  *

        operator[] 库里面实现了两个,一个用于普通对象的元素访问(可读可写),另一个就用于const修饰的对象的元素访问(只读)。

        有了operator[] 这个运算符重载的函数,我们就可以像访问字符数组那样去通过下标访问string类型对象。

2、at

        at这个函数是传递一个参数pos,获得string对象中字符串下标为pos的字符。(使用起来和operator[]一样)。

void test3()
{
	string s1("Hello world");
	for (int i = 0; i < s1.size(); i++)
	{
		s1[i]++; //普通对象调用char& operator[](size_t pos); 可以进行修改
		cout << s1[i] << " ";
	}
	cout << endl;
	const string s2("Hello World");
	for (int i = 0; i < s2.size(); i++)
	{
		//s2[i]++; //err
		//const修饰的对象调用const char& operator[](size_t pos) cosnt; 不能进行修改
		cout << s2[i] << " ";
	}
	cout << endl << endl;
	string s3("Hello world");
	for (int i = 0; i < s3.size(); i++)
	{
		s3.at(i)++; //普通对象调用char& at(size_t pos); 可以进行修改
		cout << s3.at(i) << " ";
	}
	cout << endl;
	const string s4("Hello World");
	for (int i = 0; i < s4.size(); i++)
	{
		//s4.at[i]++; //err
		//const修饰的对象调用const char& at(size_t pos) cosnt; 不能进行修改
		cout << s4.at(i) << " ";
	}
	cout << endl;

}

        3.3、Modifiers

        Modifiers ,库里面提供了以下这些函数,对string对象中字符串进行设置(修改)

1、operator+=   *

        库里面实现了三个+=运算符重载函数,我们在使用时就可以像内置类型那样+=;

这里可以+=一个string类类型对象、字符串(字符数组)或者一个字符。

void test4()
{
	string s1("hello ");
	string s2("world ");
	s1 += s2;
	cout << s1 << endl;
	
	char str[] = "ever";
	s1 += str;
	cout << s1 << endl;

	s1 += 'o';
	s1 += 'n';
	s1 += 'e';
	cout << s1 << endl;
}

2、append

        库里面实现的append感觉有点冗余;

        append可以增加string类型对象(也可以是其中的一部分)、字符串(可以是其中的一部分)、n个字符;传参也可以是迭代器。

void test5()
{
	string s1("hello ");
	string s2("world ");
	//s1.append(s2); 也可以直接添加s2的全部
	s1.append(s2, 1, 3);
	//从下标为1的位置开始,添加后面3个字符
	cout << s1 << endl;

	s1.append("everone");
	//s1.append("everone",4);
	//也可以这样,只添加前4个字符
	cout << s1 << endl;

	s1.append(3, 'x');
	//添加3个字符 'x'
	cout << s1 << endl;
}

3、push_back

push_back就是在string对象后面添加一个字符。(比较简单就不测试了)。

4、assgin

        assgin是设置string类对象的内容,(可以用同类型对象来设置(也可以是一部分)、也可以用字符串设置(或者其中一部分)、也可以将strng对象内容设置成n个字符;(也可以传迭代器))。

void test5()
{
	string s1("hello ");
	string s2("world ");
	//s1.append(s2); 
	s1.assign(s2, 1, 3);
	cout << s1 << endl;

	s1.assign("everone");
	//s1.append("everone",4);
	cout << s1 << endl;

	s1.assign(3, 'x');
	cout << s1 << endl;
}

5、insert

        insert与assgin不同,要比assgin多一个参数,这个参数pos就决定了从string对象中字符串哪个位置开始设置字符串的内容。

void test5()
{
	string s1("hello ");
	string s2("world ");
	//s1.append(s2); 
	s1.insert(3, s2, 1, 3);
	cout << s1 << endl;

	s1.insert(8, "everone");
	//s1.append("everone",4);
	cout << s1 << endl;

	s1.insert(15, 3, 'x');
	cout << s1 << endl;
}

6、erase

        erase是删除string对象中字符串的数据,(删除pos位置后面的len和字符);如果不传参就是全部删除(如果只是不传len的参数,就默认删除pos后面所有的数据)。

void test6()
{
	string s1("hello world !!!");
	cout << s1 << endl;
	s1.erase(11, 3);
	cout << s1 << endl;
	s1.erase(5);
	cout << s1 << endl;
	s1.erase();
	cout << s1 << endl;
}

7、replace

replace 替换string对象字符串中的数据;

函数的前两个参数,就是指需要替换的位置(pos位置后面的len个字符)。

        可以替换成string类类型对象(也可以是一部分)、字符串(也可以是一部分)、n个字符或者迭代器指向的区间。

void test7()
{
	string s1("Hello World ");
	string s2("everone");
	cout << s1 << endl;
	s1.replace(6, 5, s2);
	cout << s1 << endl;

	s1.replace(6, 5, "I Love");
	cout << s1 << endl;

	s1.replace(6, 5, 3, '*');
	cout << s1 << endl;
}

8、swap

        swap是库里面实现的一个函数(交换两个string类类型对象)。

void test8()
{
	string s1("hello world");
	string s2("I Love you");
	cout << "s1: " << s1 << endl;
	cout << "s2: " << s2 << endl;
	s1.swap(s2);
	cout << "s1: " << s1 << endl;
	cout << "s2: " << s2 << endl;
}

        3.4、String operations 字符串操作

        字符串操作,主要操作有返回字符串指针、查找、拷贝、获得子串等。(这里就使用其中的一部分)。

1、c_str   *

        c_str返回string对象里的字符串指针。(需要注意的就是返回的指针是const修饰的,必须用const指针接受)。

void test9()
{
	string s1("Hello World");
	const char* s = s1.c_str();
	cout << s << endl;
}

2、find   *

        find查找, 在string对象里查找指定字符(string类类型对象、字符串、字符),也可以指定区间进行查找。找到就返回下标;没有找到返回npos。

void test9()
{
	string s1("Hello World");
	string s2("Hello");
	size_t f1 = s1.find(s2);
	size_t f2 = s1.find("Love");
	size_t f3 = s1.find('o');

	cout << f1 << endl;
	cout << f2 << endl;
	cout << f3 << endl;
}

3、substr   *

        substr获得string对象字符串中的子串。这结合上面的find,

实现将一个网址的协议、域名以及资源分开。

void test10()
{
	string s1("https://blog.csdn.net/LH__1314?type=blog");
	size_t f1 = s1.find("://", 0);
	string protocol = s1.substr(0, f1);

	size_t f2, f3;
	f2 = s1.find('/', f1 + 3);
	string domain = s1.substr(f1 + 3, f2 - (f1 + 3));
	string resource = s1.substr(f2 + 1);

	cout << protocol.c_str() << endl;
	cout << domain.c_str() << endl;
	cout << resource.c_str() << endl;
}

        3.5、Inerator

        inerator就是所谓的迭代器,在string中迭代器作用没有那么大,(甚至可以将它理解为指针(但迭代器不是指针))。

在使用这些函数之前,要先知道一种类型,迭代器 iterator

string::iterator it;

因为在其他容器里也存在迭代器,所以定义时要指定类域。

1、begin、end

        这里的两个函数返回的都是迭代器,begin返回的是起始位置,end返回的是终止位置。

        这里库里面还实现了const修饰的函数,const修饰的迭代器类型有所不同

string::const_iterator it;

这里直接使用:

void test11()
{
	string s1("I Love You");
	string::iterator it = s1.begin();
	while (it != s1.end())
	{
		cout << *it << " ";
		it++;
	}
	cout << endl;
}

2、范围for

有了迭代器,我们就可以实现一种语法 范围for

void test11()
{
	string s1("I Love You");
	for (auto ch : s1)
	{
		cout << ch << " ";
	}
}

到这里,string类的基本使用就结束了。

感谢各位大佬支持并指出问题,

                        如果本篇内容对你有帮助,可以一键三连支持以下,感谢支持!!!

悦读

道可道,非常道;名可名,非常名。 无名,天地之始,有名,万物之母。 故常无欲,以观其妙,常有欲,以观其徼。 此两者,同出而异名,同谓之玄,玄之又玄,众妙之门。

;