CS106L Learning notes
什么年代了还在写原始人 C++
L2-Streams
C++ 中的字符串:
string str = "Hello world";
采用下标访问,0-based.本质上是一个字符数组。
流的存在?
与 console 和 files 等等对象进行交互的方式。很酷的在于流提供了一个通用接口.
举个例子,cout
如何输出一个东西到控制台?程序中的数据对象 string
表示
stringstream
stringstream
分为两种,有 ostringstream
和 istringstream
,代表的是输出流和输入流。ostringstream
可以使用 <<
运算符向里面写入东西。也就相当于一个缓冲区。
我们可以使用 .str()
把 ostringstream
中的东西转换为字符串。
ostringstream oss("it's me");
cout << oss.str() << endl;
//it's me
oss << "888hi!";
cout << oss.str();
//888hi!e
想象一下我们有一个字符指针一直在移动,最开始构造这个流的时候,指针从起点开始,所以我们覆盖了缓冲区。如果我们想让字符指针从流末尾开始,我们可以使用 stringstream::ate
构造一个流:
ostringstream oss("it's me",stringstream::ate);
cout << oss.str() << endl;
//it's me
oss << "888hi!";
cout << oss.str();
//it's me888hi!
stream
比 buffer
更厉害的在于,你可以直接从 stream
中读入数字或者其他什么东西。比如我们可以把读入流中的 string
也自动变为 int
等类型,并且以符号作为分割,特别智能:
istringstream iss("114 yuan");
int price;string unit;
iss >> price;
iss >> unit;
cout << price << " " << unit << endl;
//57 yuan
stringstream
牛逼之处在于,他把一个字符串给划分成了若干个 tokens,比如:
16.9 \n Ounces. . \t\n\n -38271
16.9
,Ounces
,.
,-38271
就是我们分出来的四个 tokens.但是 tokens 的分割也是基于我们如何读入的,比如上面这段,如果我们用 int
类型的变量去读入,那么第一个 token 就是 16
.
我们对于每个 ostringstream
的 IO,分为两步:
- 把对应变量转换为字符串
- 插入到对应的
ostringstream
的缓冲区。
istringstream
也是类似:
- 从缓冲区中读取字符串
- 把对应字符串转化为所求类型的变量
stringstream
支持链式调用,也就是可以实现 iss >> var1 >> var2 >> ...
的使用。
刚才我们提到了流的字符指针,我们同样可以通过一些函数来知道当前指针的位置,这些东西比较底层:
功能 | istringstream |
ostringstream |
---|---|---|
获取指针位置 | oss.tellp() |
iss.tellg() |
改变指针位置 | oss.seekp(pos) |
iss.seekg(pos) |
获取偏移量 | streamoff() |
ostringstream iss("114 yuan",stringstream::ate);
cout << iss.tellp();
//8 这里设置了 stringstream::ate,所以是在末尾
ostringstream iss1("114 yuan");
cout << iss1.tellp();
//0
我们可以实现一个简单的函数,实现从字符串到数字的转换啦!
int StringToInterger(const string &s) {
int number;
istringstream iss(s);
iss >> number;
return number;
}
这个函数实现了从字符串内读入了一个数的功能,我们也可以实现从字符串自动读入一堆数,然后返回一个 vector
的功能:
vector<int> StringToVector(const string &s) {
vector <int> number;
istringstream iss(s);
int x;
while (iss >> x) {
number.push_back(x);
}
return number;
}
int main() {
vector <int> vec;
vec = StringToVector("1 2 3 4 5");
for (auto x : vec) cout << x*x << "\n";
return 0;
}
//1 4 9 16 25
状态位
反应流到底发生什么了,分成四种:
- Good bit: 准备好读入/输出了
- Fail bit: 先前操作失败,冻结此后操作
- EOF bit: 先前操作已经到达了缓冲区内容末尾
- Bad bit: 由于流缓冲区上的输入/输出操作失败而导致的错误。(比如你文件被删掉了)
状态位之间并不矛盾,比如 G 和 B 通常同时出现在类型不匹配的时候,正常读入到 EOF
也会同时出现 G 和 E.良好和失败并不对立,正如考试一样^_^
void PrintStateBits(const istringstream &s) {
cout << (s.bad() ? "B" : "-");
cout << (s.good() ? "G" : "-");
cout << (s.fail() ? "F" : "-");
cout << (s.eof() ? "E" : "-");
cout << endl;
}
int StringToInterger(const string &s) {
int number;
istringstream iss(s);
PrintStateBits(iss);
iss >> number;
PrintStateBits(iss);
return number;
}
int main() {
while (1) {
string s;
if (!getline(cin,s)) exit(1);
StringToInterger(s);
}
return 0;
}
1
-G--
---E
12 2
-G--
-G--
a
-G--
--F-
11111111111111111111111
-G--
--FE
状态位的存在帮助我们很好地去检查我们在使用流时发生的错误。
标准 iostream
以下四个流都是与 console 交互的:
cin
: 标准输入流cout
: 标准输出流(buffered)cerr
: 标准错误流(unbuffered)clog
: 标准错误流(buffered)
cin,cout
和控制台交互的原则和我们在 stringstream
大同小异,但是特点在于,如果缓冲区这时候没有东西,cin
会等待用户输入。提取器会帮助我们跳过前导空格,但是不会消耗下一个非空格字符。
注意,cin
如果读入失败了,这个时候 Fail bit 为 1,此后所有 cin
操作都倒闭了。
我们也可以使用 getline()
代替运算符,这样我们就可以忽略空格读入一整行了,getline
是以换行符为分界,每次消耗掉一个换行符。主要作用在于确保我们可以清空缓存区,不会让我们读入变得差。使用方法也很简单:
cin >> str;
//更改成
getline(cin,str);
getline
函数返回的是 bool
,指示我们的操作是否成功。基于此我们可以写出一个新的更加完备的 getInteger()
int getInterget() {
while (1) {
string line;
if (!getline(cin,line))
throw domain_error();
istringstream iss(line);
int res;char trash;
if (iss >> res && !(iss >> trash))
return res;
}
}
对于与文件交互,我们使用 ifstream
和 ofstream
int var1;
ifstream in("in.txt",ifstream::in);
ofstream out("out.txt",ofstream::out);
in >> var1;
out << var1;
这段代码实现了从 in.txt
中读入,并且输出到 out.txt
中。
现代 C++ 类型
size_t
在实际编码过程中,经常碰到对于 size_t
的 warnings。我们来深究一下这个东西。
在调用 std::.size()
的时候我们得到的是一个 size_t
类型的无符号整数,它代表一个表示大小的变量,本质上是一个无符号整型。size_t
在不同的机器上有微小不同,提供了一种可以移植的方法来声明与系统中可寻址的内存区域一致的长度。
在实际编码中,我们要小心 arr.size() - 1
这样的写法!当 size() = 0
的时候就会产生很多问题。
类型别名
我们可以使用 using a = b;
这样的语句来给 b
这个类型取一个 a
的别名。比如:
using map_iter = unordered_map <forward_list<int>,unordered_set> :: const_iterator;
map_iter begin = map1.cbegin();
最直接的应用就是 STL 库里的各种 iterator
和 reference
。
auto
自动推导,来点 C++ 笑话:
auto f(string& a) {
auto b = a;
return b;
}
比较需要注意的是,在 C++ 中尽量少用 C 风格字符串。在 lambda 函数,自动推导 iterator
,定义 templates
中我们经常用到 auto.
L3-Sequences Containers
pair
pair
是一个可以储存两个对象的元组。从 C++17 开始,C++ 本身支持了 structured bindings.我们可以通过 make_pair
自动推导一个对应的二元组。当然也可以利用 auto
自动推导:
auto p = make_pair(1.28,"abc"); //返回的是一个 pair<double,string>
auto [a,b] = p;// a 为 double 类型,b 为 string 类型。
struct
struct
本身是一个更为强大的元组,可以任意储存多种类型的对象。并且每个对象是具有名称的,不需要再用 first,second
来写了。
struct Time {
double timestamp;
};
struct Course {
Time start,end;
string name;
vector <string> students;
};
struct
是 class
的一种轻量级形式,可以说我们不涉及 private/public
的内容。
当然我们依然可以利用结构化绑定(structured bindings)来简化代码:
struct node {
int l,r;
};
node Construct_node(int l,int r) {
return node{l,r};
}
初始化
C++11 开始支持了一种统一初始化:
int main() {
vector <int> pi{3,1,4,1,5,9};
}
对于自定义的 struct
来说也很适合,你可以按照 struct
里面定义的顺序来初始化一个东西。注意对于 0 初始化来说,我们一般只认为对全局变量或者某些东西你可以依赖 C++ 进行空初始化,对于其他情况尽量不要依赖 C++.举个例子,对于 std::vector
来说,他在构造函数里有指定某个类型的默认值,所以新定义 std::vector
你可以相信!
序列容器
是容器的子集,容器又是 STL 的一方面东西。
std::vector <T>
std::deque <T>
std::list <T>
std::array <T>
std::forward_list <T>
vector
可以使用任意类型定义的线性容器。
基本方法 | std::vector<int> |
---|---|
创建新 vector | vector <int> v; |
创建新 vector,长度为 n | vector <int> v(n); |
创建新 vector,长度为 n,值为 k | vector <int> v(n,k); |
添加新元素 | v.push_back(k); |
清空 | v.clear() |
获取下标元素 | v.at(i);v[i]; |
注意,使用 at(i)
的时候可以丢出错误 out_of_range
,但是用中括号就没有了,如果超出范围实际上是一个 undefined behavior。
deque
注意到 vector
实际上不支持 push_front
操作,或者说这样做的操作效率很低。我们使用 deque
来实现一个可以在双端插入删除的东西。但是请注意,在同样操作下 deque
的效率总是比 vector
低的!
STL
先来点现代 C++ 的三体人震撼:
int main() {
int n = 15;
vector <int> vec(n);
generate(vec.begin(),vec.end(),rand);
sort(vec.begin(),vec.end());
copy(vec.begin(),vec.end(),ostream_iterator<int>(cout,"\n"));
}
这个代码实现了自动生成 n 个整数并且排序输出的功能,太无敌了。
L4-Associative Containers & Iterators
Container Adaptor
STL 库里给我们提供了很多常用数据结构的封装完毕的实现!比如 std::stack,std::queue
,这俩东西实际上是 vector,deque
只允许在某一端进行操作的实现。
因为这个原因,所以这俩都被称为容器适配器(container adaptor).就是这俩实际上的实现是把 vector,deque
给封装成了其他东西。在声明的时候如果你喜欢给他任意一个满足需求的容器都可以。
为什么要有容器适配器?这就来到了设计 C++ 的哲学之一:直接在代码中表达自己的意图。
Associative Containers
关联容器是一种没有序列概念的数据类型,数据以键值类储存。
std::map<T1,T2>
std::set<T>
std::unordered_map<T1,T2>
std::unordered_set<T>
前两个是基于顺序排序的储存,所以键值类型必须支持小于号。后两个必须实现 hash
函数。所以 map,set
在遍历连续的一段区间时显著快,后两者在随机访问中显著快。
std::map <string,int> freq;
string word;
freq[word]++;
freq.get(word);//如果不存在则会抛出错误
int a = freq[word];//如果不存在则会新建一个 (word,0) 的元组
freq.count(word);//这个返回0/1 代表是否存在
Iterators
太超模了,这才是新时代 C++!
迭代器允许我们遍历任意容器,无论其是否有序。提供了一种以线性方式遍历容器的方法。
迭代器的类型取决于使用的数据结构的类型,我们可以对一个迭代器类型进行 ++
的操作,以及使用 *
解引用。每个容器都有 .begin(),.end()
两个迭代器。比如:
set <int>::iterator iter = mySet.begin();
for (;iter != mySet.end();++iter) {
cout << *iter << "\n";
}
当然实际上由于我们现在有了 auto,我们也可以让 auto 直接推导。
本作品采用 知识共享署名-相同方式共享 4.0 国际许可协议 进行许可。