【C++ Primer】第3章:字符串、向量和数组

简介: 【C++ Primer】第3章:字符串、向量和数组

第3章 字符串、向量和数组


string表示可变长的字符序列


vector存放的是某种给定类型对象的可变长序列


使用命名空间


using namespace std;


头文件不应包含using声明


1. 标准库类型string


字符串初始化


string s1;
string s2(s1);
string s3("values");     //直接初始化
string s3 = "values";    //拷贝初始化
string s4(n, 'c');      //n个c组成字符串   直接初始化


拷贝初始化和直接初始化


使用等号初始化一个变量叫做拷贝初始化


不使用等号则是直接初始化


string s8 = string(10, 'c');   //拷贝初始化



在执行读写操作时,string对象会自动忽略开头的空白(即空格符、换行符、制表符等)并从第一个真正的字符开始读写,知道遇见下一个空白为止。


遇到第一个非空白符开始,之后再遇到空白符就结束


例如输入’ Hello World! ‘,会输出’Hello’


int main(){
    string s;
    cin >> s;           //  "  Hello World  "
    cout << s <<endl;   // "Hello"
    return 0;
}


string s1, s2;
cin >> s1 >> s2;             // "  Hello"  " World!"
cout << s1 << s2 <<endl;     //  "HelloWorld!"


一次读入一个单词


int main(){
    string word;
    while(cin >> word){
        cout << word <<endl;
    }
    return 0;
}


一次读入一整行


getline函数的参数是一个输入流和一个string对象。


函数从给定的输入流中读入内容,直到遇到换行符为止(注意换行符也被读进来了),然后把所读的内容存入到那个string对象中去(注意不存放换行符)。


int main(){
    string line;
    while(getline(cin, line)){
        cout << line << endl;
    }
    return 0;
}


string::size_type类型,它是一个无符号类型


auto len = line.size();   //len的类型是string::size_type类型


注意:字符串字面值不能相加,string能相加


切记:字符串字面值与string是不相同的类型



范围for


C++11新标准


for(auto ch : str){
  cout << ch << endl;
}


范围for还可以改变字符串中的字符


string s("Hello World!!!");
for(auto &c : s){
    c = toupper(c);
}
cout << s <<endl;   // "HELLO WORLD!!!"


2. 标准库类型vector


vector是一个类模板


使用模板时,需要指出编译器应把类或函数实例化成何种类型。


不存在包含引用的vector,因为引用非对象。



列表初始化时C++11新标准


vector<string> v{10};       //10个“”
vector<string> v{10, "hi"}  //10个"hi"



迭代器


auto b = v.begin();  //b表示v的第一个元素
auto e = v.end();    //e表示v尾元素的下一个位置     //b,e类型相同


尾后迭代器是容器中不存在的位置


特殊情况下如果容器为空,则begin和end返回的是同一个迭代器


所有标准库容器都可以使用迭代器,只有少数容器支持下标运算符



vector<int>::iterator it;   //能读写
string::iterator it2;       //能读写
vector<int>::const_iterator it3;  //只读不写
string::const_iterator it4;       //只读不写


C++11新标准引入cbegin和cend


auto it3 = v.cbegin();  //it3的类型是vector<int>::const_iterator



迭代器的距离类型:difference_type


用迭代器实现二分查找


auto beg = text.begin(), end = text.end();
auto mid = text.begin + (text.end - text.begin())/2;
while(mid!=end && *mid!=target){
    if(target < * mid){
        end = mid;
    }
    else
        beg = mid + 1;
    mid = beg + (end - beg)/2;
}


3. 数组


数组维度编译的时候已知,是一个常量表达式


默认情况下,数组的元素被默认初始化


不能将数组的内容拷贝给其他数组作为其初始值,也不能用数组为其他数组赋值


char a1[] = {'C', '+', '+'};       //维度是3
char a2[] = {'C', '+', '+', '\0'}   //维度4
char a3[] = "C++";                 //维度4
const char a4[6] = "Daniel";       //错误,没有空间可存放空字符


从右向左,从内向外


int *ptrs[10];             //ptrs是含有10个整型指针的数组
int &refs[10] = arr;       //错误:不存在引用的数组
int (*Parray)[10] = &arr;  //Parray指向一个含有10个整型的数组
int (&arrRef)[10] = arr;   //arrRef引用一个含有10个整型的数组
int *(&array)[10]  = ptrs;  //array是数组的引用,该数组含有10个指针


使用数组下标的时候,通常将其定义为size_t类型,是无符号整型


指针也是迭代器


c++11新标准引入了begin和end函数


int ia[] = {0,1,2,3,4,5,6,7,8,9};
int *beg = begin(ia);  //指向ia首元素的指针
int *last = end(ia);   //指向arr尾元素的下一个位置的指针


auto n = end(arr) - begin(arr);  //n的值是5,即arr元素个数   是ptrdiff_t类型  距离可能为负,所以是带符号类型


size_t和ptrdiff_t都是定义在cstddef头文件中的机器相关的类型。


C风格字符串


定义在cstring头文件中



与旧代码的接口


不能用string对象直接初始化指向字符的指针,为了完成功能,string专门提供了一个名为c_str的成员函数


char *str = s;  //错误:不能用string对象初始化char*
const char *str = s.c_str();  //正确


不允许用数组初始化数组


不允许用vector对象初始化数组


可以使用数组初始化vector对象


int int_arr[] = {0,1,2,3,4,5};
vector<int> ivec(begin(int_arr), end(int_arr));
vector<int> subVec(int_arr+1, int_arr+4);   //用数组的一部分初始化vector


多维数组


int ia[3][4];
size_t cnt = 0;
for (auto& row : ia) {
  for (auto& col : row) {
    col = cnt;
    ++cnt;
  }
}


for (const auto& row : ia) {   //&不能少
  for (auto col : row) {
    cout << col << endl;
  }
}


指针和多维数组


for(auto p = ia; p != ia + 3; ++p){      //p的类型是指向4个整数的数组的指针
    for(auto q = *p; q != *p +4; ++q){   //q的类型是指向整数的指针
        cout << *q << ' ';
    }
    cout << endl;
}


for(auto p = begin(ia); p != end(ia); ++p){
    for(auto q = begin(*p); q != end(*p); ++q){
        cout << *q << ' ';
    }
    cout << endl;
}


using int_array = int[4];
typedef int int_array[4];    //两句功能一样
for(int_array *p = ia; p != ia + 3; ++p){
    for(int *q = *p; q != *p +4; ++q){
        cout << *q << ' ';
    }
    cout << endl;
}
目录
相关文章
|
4月前
|
搜索推荐 编译器 C语言
【C++核心】特殊的元素集合-数组与字符串详解
这篇文章详细讲解了C++中数组和字符串的基本概念、操作和应用,包括一维数组、二维数组的定义和使用,以及C风格字符串和C++字符串类的对比。
106 4
|
3月前
|
缓存 网络协议 API
C/C++ StringToAddress(字符串转 boost::asio::ip::address)
通过上述步骤和示例代码,你可以轻松地在C++项目中实现从字符串到 `boost::asio::ip::address`的转换,从而充分利用Boost.Asio库进行网络编程。
100 0
|
3月前
|
编译器 C语言 C++
C/C++数字与字符串互相转换
C/C++数字与字符串互相转换
|
4月前
|
C++
C++(十一)对象数组
本文介绍了C++中对象数组的使用方法及其注意事项。通过示例展示了如何定义和初始化对象数组,并解释了栈对象数组与堆对象数组在初始化时的区别。重点强调了构造器设计时应考虑无参构造器的重要性,以及在需要进一步初始化的情况下采用二段式初始化策略的应用场景。
|
4月前
|
C++
HTML+JavaScript构建一个将C/C++定义的ANSI字符串转换为MASM32定义的DWUniCode字符串的工具
HTML+JavaScript构建一个将C/C++定义的ANSI字符串转换为MASM32定义的DWUniCode字符串的工具
|
4月前
|
存储 C++
C++(五)String 字符串类
本文档详细介绍了C++中的`string`类,包括定义、初始化、字符串比较及数值与字符串之间的转换方法。`string`类简化了字符串处理,提供了丰富的功能如字符串查找、比较、拼接和替换等。文档通过示例代码展示了如何使用这些功能,并介绍了如何将数值转换为字符串以及反之亦然的方法。此外,还展示了如何使用`string`数组存储和遍历多个字符串。
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
63 2
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
114 5
|
2月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
116 4
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
156 4