草庐IT

【C++】 STL详解

顾城沐心 2023-04-23 原文

目录

一:泛型编程          

二:什么是STL

三:STL发展

四:STL组件

五:容器

六:类型成员

七:适配器

八:迭代器

九:算法

十:顺序容器

十一:向量 vector

十二:双端队列 queue

十三:列表 list

十四:关联容器

十五:map

十六:multimap

十七:set

十八:multiset

十九:迭代器

二十:函数对象

二十一:已经集成的函数对象

二十二:自定义函数对象

二十三:标准C++库中的算法

二十四:STL算法的头文件

二十五:标准函数

二十六:泛型算法例子

二十七:自定义函数作为算法参数                


一:泛型编程          

1. 将程序写得尽可能通用  

2. 将算法从特定的数据结构中抽象出来,成为通用的

3. C++的模板为泛型程序设计奠定了关键的基础

二:什么是STL

STL(Standard Template Library),即标准模板库,是一个高效的C++程序库

    被容纳于C++标准程序库(C++ Standard Library)中,是ANSI/ISO C++标准中最新的也是极具革命性的一部分

    包含了诸多在计算机科学领域里常用的基本数据结构和基本算法。为广大C++程序员们提供了一个可扩展的应用框架,高度体现了软件的可复用性

从逻辑层次来看,在STL中体现了泛型化程序设计的思想(generic programming)

     在这种思想里,大部分基本算法被抽象,被泛化,独立于与之对应的数据结构,用于以相同或相近的方式处理各种不同情形

从实现层次看,整个STL是以一种类型参数化(type parameterized)的方式实现的

    基于模板(template)

三:STL发展

1971 : David R. Musser 开始倡导Generic Programming 概念

1979 : Alexander Stepanov 创造STL

1987 : Alex 和Musser 开发出一套Ada library

???? : Alex 先后在AT&T 及HP实验室以C 及C++实验大量的体系结构和算法形式。

1992 : Meng Lee 加入称为另一位主要贡献者

1993/11 : Alex 于ANSI/ISO C++ 会议展示

1994 夏: STL 被纳入C++标准

四:STL组件

Container(容器) 各种基本数据结构

Adapter(适配器) 可改变containers或function object接口的一种组件

Algorithm(算法) 各种基本算法如sort、search…等

Iterator(迭代器)* 连接containers和algorithms

Function object(函数对象) *

Allocator(分配器)*

五:容器

容器类是容纳、包含一组元素或元素集合的对象

异类容器类与同类容器类

顺序容器与关联容器

七种基本容器:向量(vector)、双端队列(deque)、链表(list)、集合(set)、多重集合(multiset)、映射(map)和多重映射(multimap)

标准容器的成员绝大部分都具有共同的名称

六:类型成员

七:适配器

适配器是一种接口类

    为已有的类提供新的接口

    目的是简化、约束、使之安全、隐藏或者改变被修改类提供的服务集合

三种类型的适配器:

    容器适配器:用来扩展7种基本容器,它们和顺序容器相结合构成栈、队列和优先队列容器

    迭代器适配器

    函数对象适配器 

八:迭代器

迭代器是面向对象版本的指针,它们提供了访问容器、序列中每个元素的方法

九:算法

C++标准模板库中包括70多个算法

    其中包括查找算法,排序算法,消除算法,记数算法,比较算法,变换算法,置换算法和容器管理等等

这些算法的一个最重要的特性就是它们的统一性,并且可以广泛用于不同的对象和内置的数据类型

十:顺序容器

顺序容器的接口

插入方法

push_front(),push_back(),insert(),运算符“=”

删除方法

pop() ,erase(),clear()

迭代访问方法

使用迭代器

其他顺序容器访问方法(不修改访问方法)

front(),back(),下标[]运算符

十一:向量 vector

1. 向量属于顺序容器,用于容纳不定长线性序列(即线性群体),提供对序列的快速随机访问(也称直接访问)

2. 数据结构很像一个数组,所以与其他容器相比,vector 能非常方便和高效访问单个元素,支持随机访问迭代子

3. 向量是动态结构,它的大小不固定,可以在程序运行时增加或减少

    与数组不同,向量的内存用尽时,向量自动分配更大的连续内存区,将原先的元素复制到新的内存区,并释放旧的内存区;这是向量类的优点

头文件:#include <vector>

vector 基本操作

(1)头文件

#include<vector>

(2)创建vector对象

vector<int> vec;

(3)尾部插入数字

vec.push_back(a);

(4)使用下标访问元素

cout<<vec[0]<<endl;记住下标是从0开始的

(5)使用迭代器访问元素

vector<int>::iterator it;

for(it=vec.begin();it!=vec.end();it++)

cout<<*it<<endl;

(6)插入元素

vec.insert(vec.begin()+i,a);在第i+1个元素前面插入a;

(7)删除元素   

vec.erase(vec.begin()+2);删除第3个元素

vec.erase(vec.begin()+i,vec.end()+j);删除区间[i,j-1];区间从0开始

(8)向量大小

vec.size();

vec.resize;改变大小

(9)清空

vec.clear();

vector,使用示例如下

#include <iostream>
#include <iomanip>
#include <vector>	//包含向量容器头文件
using namespace std ;
void main(){    
	vector<int>  A(10);	  //创建vector对象
	int n;	
	int primecount = 0, i, j;
	cout<<"Enter a value>=2 as upper limit: ";
	cin >> n;
	A[primecount++] = 2;//下标法访问元素
	for(i = 3; i < n; i++){ 
		if (primecount == A.size())	
			A.resize(primecount + 10); //改变容器大小
        if (i % 2 == 0)
			continue;
		j = 3;
 		while (j <= i/2 && i % j != 0)
			j += 2;        
		if (j > i/2) A[primecount++] = i;
	}
	for (i = 0; i<primecount; i++){//输出质数
		cout<<setw(5)<<A[i]; 
		if ((i+1) % 10 == 0) //每输出10个数换行一次
			cout << endl;
	} 
	cout<<endl;
}

十二:双端队列 queue

双端队列是一种放松了访问权限的队列

元素可以从队列的两端入队和出队,也支持通过下标操作符“[]”进行直接访问

与向量的对比:

    功能上:和向量没有多少区别,

    性能上:在双端队列起点上的插入和删除操作快

头文件:#include <deque>

十三:列表 list

链表主要用于存放双向链表,可以从任意一端开始遍历。链表还提供了拼接(splice)操作,将一个序列中的元素从插入到另一个序列中

对比:

元素的插入和删除操作对 list 而言尤为高效

与 vector 和 deque 相比,对元素的下标访问操作的低效是不能容忍的,因此 list 不提供这类操作

头文件:#include <list>

列表,使用示例如下

#include <iostream>
#include <list>
using namespace std ;
int main(){
	list<int> Link;	//构造一个列表用于存放整数链表
	int i, key, item;    
	for(i=0;i < 10;i++)// 输入10个整数依次向表头插入{
		cin>>item;
		Link.push_front(item);
	}
	cout<<“List: ”; // 输出链表
	list<int>::iterator p=Link.begin();
    while(p!=Link.end()){ //输出各节点数据,直到链表尾
		cout <<*p << "  ";
		p++;  //使P指向下一个节点
	}
	cout << endl;
	cout << "请输入一个需要删除的整数: ";
	cin >> key;
	Link.remove(key);   
	cout << "List: "; // 输出链表
	p=Link.begin();	// 使P重新指向表头
	while(p!=Link.end()){ 
		cout <<*p << "  ";
		p++; // 使P指向下一个节点
	}
	cout << endl;
}

十四:关联容器

通过保存在数据项中的索引项,尽可能快速检索数据项

STL标准库中只包含有序关联容器set、multiset、map、multimap

      set, multiset:数据项就是索引项; multiset允许出现重复的索引项

      map, multimap:数据项是由索引项和其他某种类型的数据组成的一对数据; multimap允许出现重复的索引项

十五:map

1. 增加和删除节点对迭代器的影响很小。对于迭代器来说,可以修改实值,而不能修改key

2. 自动建立Key - value的对应。key 和 value可以是任意你需要的类型

3. 根据key值快速查找记录,查找的复杂度基本是Log(N),如果有1000个记录,最多查找10次,1,000,000个记录,最多查找20次

map的构造函数

1. 使用map得包含map类所在的头文件
     #include <map>

2. map对象是模板类,需要关键字和存储对象两个模板参数:
      map<int, string> personnel;//用int作为索引,存储string对象

map的成员函数

1. map类已经对[]操作符进行了重载

2. 插入2时,先在enumMap中查找主键为2的项,没发现,然后将一个新的对象插入enumMap,键是2,值是一个空字符串,插入完成后,将字符串赋为“Two”;

3. 但是该方法会将每个值都赋为缺省值,然后再赋为显示的值,如果元素是类对象,则开销比较大。可以用以下insert()来避免开销

4. 下标操作符给出了获得一个值的最简单方法:

     CString tmp = enumMap[2];

     但是,只有当map中有这个键的实例时才对,否则会自动插入一个实例,值为初始化值

5. 我们可以使用Find()和Count()方法来发现一个键是否存在

6. 查找map中是否包含某个关键字条目用find()方法,传入的参数是要查找的key

7. 通过map对象的方法获取的iterator数据类型是一个std::pair对象,包括两个数据 iterator->first 和 iterator->second 分别代表关键字和存储的数据

从map中删除元素

1. 移除某个map中某个条目用erase()

2. 该成员方法的定义如下

iterator erase(iterator it);              //通过一个条目对象删除
iterator erase(iterator first, iterator last);//删除一个范围

size_type erase(const Key& key);   //通过关键字删除

例如:

enumMap.erase(1);//删掉关键字“1”对应的条目
enumMap.erase(enumMap.begin());//删掉第一个条目
enumMap.erase(enumMap.begin(), enumMap.begin() + 2);//删掉起始的两个条目

3. clear()就相当于

 enumMap.erase(enumMap.begin(), enumMap.end()); 

map,使用示例如下

#include <map>
#include <iostream>
#include <string>
using namespace std;
void main(){
	map< string, string > trans_map;
	typedef map< string, string >::value_type valType;
	trans_map.insert( valType( "001", "grateful" ));
	trans_map.insert( valType( "002", "them" ));
	trans_map.insert( valType( "003", "because" ));
	trans_map.insert( valType( "004", "no" ));
	trans_map.insert( valType( "005", "says" ));
	trans_map.insert( valType( "006", "thanks" ));
	trans_map.insert( valType( "007", "was" ));
    trans_map.insert( valType( "008", "suppose" ));	
	map< string,string >::iterator it;
	cout << "Here is our transformation map: \n\n";
	for(it=trans_map.begin();it!=trans_map.end();++it)
		cout<<"key: "<<(*it).first<<"\t"<<"value: " <<(*it).second<<"\n";
	cout<<"Find Key:005"<<endl;
	it=trans_map.find("105");
	if (it==trans_map.end()){
		cout<<"not found"<<endl;
	}
	else{
		cout<<"key: "<<(*it).first <<"\t"<<"value: " <<(*it).second<<"\n";
	}
}

十六:multimap

multimap 除了元素对的关键字不是唯一外,与 map 相似

头文件:#include <map>

十七:set

set 可以被视为只有关键字而没有相关的元素值的 map,因此 set 的用户接口也发生了微小的变化:成员类型中没有:

     typedef Key value_type;

     typedef Cmp value_compare

     操作中没有元素的下标访问操作

头文件:#include <set>

十八:multiset

multiset 除了关键字不是唯一外,与 set 相似

头文件:#include <set>

十九:迭代器

迭代器是面向对象版本的指针

     指针可以指向内存中的一个地址

     迭代器可以指向容器中的一个位置

STL的每一个容器类模版中,都定义了一组对应的迭代器类。使用迭代器,算法函数可以访问容器中指定位置的元素,而无需关心元素的具体类型

迭代器的类型

1. 输入迭代器

可以用来从序列中读取数据

2. 输出迭代器

允许向序列中写入数据

3. 前向迭代器

既是输入迭代器又是输出迭代器,并且可以对序列进行单向的遍历

4. 双向迭代器

与前向迭代器相似,但是在两个方向上都可以对数据遍历

5. 随机访问迭代器

也是双向迭代器,但能够在序列中的任意两个位置之间进行跳转

迭代器的类型表

迭代器,使用示例

#include <list>
#include <iostream>
using namespace std;
int main(){
	int i,key;
	list<int> intList;	
	list<int>::iterator it;
	cout<<"input 5 digit:";
	for(i=0;i<5;i++){
		cin>>key;
		intList.push_front(key);
	}	
    cout<<"data list:"<<endl; 	
	it=intList.end();
	while(1){
		cout.width(6);
		cout<<*(--it);
		if(it==intList.begin())
			break;
	}
	return 0;
}

二十:函数对象

1. 一个行为类似函数的对象,它可以没有参数,也可以带有若干参数,其功能是获取一个值,或者改变操作的状态

2. 任何普通的函数和任何重载了调用运算符operator()的类的对象都满足函数对象的特征

3. STL中也定义了一些标准的函数对象,如果以功能划分,可以分为算术运算、关系运算、逻辑运算三大类;为了调用这些标准函数对象,需要包含头文件<functional>

二十一:已经集成的函数对象

二十二:自定义函数对象

#include <iostream>
using namespace std;
class CFunObj{
public:
	void operator()(){
		cout<<"hello,function object!"<<endl;
	}
	int operator()(int i){
		cout<<"hello, function object other!"<<endl;
		return i+1;
	}
private:
	int dat;
};

void main(){
	CFunObj fo;
	fo();
	cout<<fo(1)<<endl;
	//CFunObj()();
}	

二十三:标准C++库中的算法

1. 算法本身是一种函数模板

2. 不可变序列算法(non-mutating algorithms)

       不直接修改所操作的容器内容的算法

3. 可变序列算法(mutating algorithms)

       可以修改它们所操作的容器的元素

4. 算法部分主要由头文件<algorithm>,<numeric>和<functional>组成

二十四:STL算法的头文件

<algorithm>是所有STL头文件中最大的一个,它是由一大堆模版函数组成的,可以认为每个函数在很大程度上都是独立的,其中常用到的功能范围涉及到比较、交换、查找、遍历操作、复制、修改、移除、反转、排序、合并等等

<numeric>体积很小,只包括几个在序列上面进行简单数学运算的模板函数,包括加法和乘法在序列上的一些操作

<functional>中则定义了一些模板类,用以声明函数对象

二十五:标准函数

二十六:泛型算法例子

#include <iostream>
#include <vector>
#include <algorithm>
#include <iterator>
#include <numeric>
#include <functional>
using namespace std;
int main(){
	int ia[] = { 1, 2, 3, 4, 5, 7, 9, 11 };
	vector<int> iv(ia, ia+8);
	//累加,52
	cout<<accumulate(iv.begin(),iv.end(),10)<<endl; //相邻差值
	adjacent_difference(iv.begin(),iv.end(),iv.begin());
	//复制到ostream_iterator 去, 每列印一个元素, 即加上一个空格
    	copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
	// 1 1 1 1 1 2 2 2 	
	//计算元素值为2 的个数
	cout << count(iv.begin(), iv.end(), 2) << endl; // 3
	//计算奇数元素的个数
	cout << count_if(iv.begin(), iv.end(),
	bind2nd(modulus<int>(),2)) << endl; // 5
	//从头开始填入新值7, 填3 次
	fill_n(iv.begin(), 3, 7);
	copy(iv.begin(), iv.end(), ostream_iterator<int>(cout,“ ”));
	// 7 7 7 1 1 2 2 2
	//内积, 7*7 + 7*7 + 7*7 + 1*1 + 1*1 + 2*2 + 2*2 + 2*2
	cout << inner_product(iv.begin(), iv.end(), iv.begin(),0) << endl; 
	//161
    	//排序
	sort(iv.begin(), iv.end());
	copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
	// 1 1 2 2 2 7 7 7
	//顛倒元素次序
	reverse(iv.begin(), iv.end());
	copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
	// 7 7 7 2 2 2 1 1
	//旋转, 交换[first, middle)和[middle, last)
	rotate(iv.begin(), iv.begin()+3, iv.begin()+6);
	copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
	// 2 2 2 7 7 7 1 1
}

二十七:自定义函数作为算法参数                

#ifndef CMYCLASS_H
#define CMYCLASS_H
#include <string.h>
class CMyClass{
public:
	CMyClass();
	CMyClass(string name,int age);
	friend bool Less(const CMyClass &num,const CMyClass &myclass);
	bool operator==(const CMyClass &myclass);
	string &GetName();
	int GetAge();
private:
	string m_name;
	int m_age;};
#endif
CMyClass::CMyClass(){
	m_name="";
	m_age=12;
}
CMyClass::CMyClass(string name,int age)
{	m_name=name;
	m_age=age;
}
bool CMyClass::operator==(const CMyClass &myclass){
	return m_name==myclass.m_name;
}
string & CMyClass::GetName(){
	return m_name;}
int CMyClass::GetAge(){
	return m_age;}
#include "MyClass.h"
#include <iostream>
#include <algorithm>
#include <vector>
#include <functional>
#include <numeric>
using namespace std;
bool Less(const CMyClass &num,const CMyClass &myclass){
	return num.m_name<myclass.m_name;
}
void PrintMyClass(CMyClass &myclass){
	cout<<"name:"<<myclass.GetName()<<"\t age:"<<myclass.GetAge()<<endl;
}
bool SearchByName(CMyClass &myclass){
	return myclass.GetName()=="AAAAZ";
}
void  main(){
	CMyClass myclass;
	vector<CMyClass> vec;
	vec.push_back(CMyClass("AAAA",12));
	vec.push_back(CMyClass("DFKASDF",12));
	vec.push_back(CMyClass("ASDFSAFA",12));
	vec.push_back(CMyClass("Z",12));
	vec.push_back(CMyClass("AAAAZ",12));
	vec.push_back(CMyClass("DFKSADFZ",12));
	sort(vec.begin(),vec.end(),Less);
//	for (vector<CMyClass>::iterator it=vec.begin();it!=vec.end();it++){
//		cout<<it->GetName()<<endl;}
	for_each(vec.begin(),vec.end(),PrintMyClass);
	vector<CMyClass>::iterator r=find_if(vec.begin(),vec.end(),SearchByName);
	cout<<(*r).GetName()<<endl;
}

有关【C++】 STL详解的更多相关文章

  1. 物联网MQTT协议详解 - 2

    一、什么是MQTT协议MessageQueuingTelemetryTransport:消息队列遥测传输协议。是一种基于客户端-服务端的发布/订阅模式。与HTTP一样,基于TCP/IP协议之上的通讯协议,提供有序、无损、双向连接,由IBM(蓝色巨人)发布。原理:(1)MQTT协议身份和消息格式有三种身份:发布者(Publish)、代理(Broker)(服务器)、订阅者(Subscribe)。其中,消息的发布者和订阅者都是客户端,消息代理是服务器,消息发布者可以同时是订阅者。MQTT传输的消息分为:主题(Topic)和负载(payload)两部分Topic,可以理解为消息的类型,订阅者订阅(Su

  2. Tcl脚本入门笔记详解(一) - 2

    TCL脚本语言简介•TCL(ToolCommandLanguage)是一种解释执行的脚本语言(ScriptingLanguage),它提供了通用的编程能力:支持变量、过程和控制结构;同时TCL还拥有一个功能强大的固有的核心命令集。TCL经常被用于快速原型开发,脚本编程,GUI和测试等方面。•实际上包含了两个部分:一个语言和一个库。首先,Tcl是一种简单的脚本语言,主要使用于发布命令给一些互交程序如文本编辑器、调试器和shell。由于TCL的解释器是用C\C++语言的过程库实现的,因此在某种意义上我们又可以把TCL看作C库,这个库中有丰富的用于扩展TCL命令的C\C++过程和函数,所以,Tcl是

  3. 【详解】Docker安装Elasticsearch7.16.1集群 - 2

    开门见山|拉取镜像dockerpullelasticsearch:7.16.1|配置存放的目录#存放配置文件的文件夹mkdir-p/opt/docker/elasticsearch/node-1/config#存放数据的文件夹mkdir-p/opt/docker/elasticsearch/node-1/data#存放运行日志的文件夹mkdir-p/opt/docker/elasticsearch/node-1/log#存放IK分词插件的文件夹mkdir-p/opt/docker/elasticsearch/node-1/plugins若你使用了moba,直接右键新建即可如上图所示依次类推创建

  4. 【Elasticsearch基础】Elasticsearch索引、文档以及映射操作详解 - 2

    文章目录概念索引相关操作创建索引更新副本查看索引删除索引索引的打开与关闭收缩索引索引别名查询索引别名文档相关操作新建文档查询文档更新文档删除文档映射相关操作查询文档映射创建静态映射创建索引并添加映射概念es中有三个概念要清楚,分别为索引、映射和文档(不用死记硬背,大概有个印象就可以)索引可理解为MySQL数据库;映射可理解为MySQL的表结构;文档可理解为MySQL表中的每行数据静态映射和动态映射上面已经介绍了,映射可理解为MySQL的表结构,在MySQL中,向表中插入数据是需要先创建表结构的;但在es中不必这样,可以直接插入文档,es可以根据插入的文档(数据),动态的创建映射(表结构),这就

  5. 最强Http缓存策略之强缓存和协商缓存的详解与应用实例 - 2

    HTTP缓存是指浏览器或者代理服务器将已经请求过的资源保存到本地,以便下次请求时能够直接从缓存中获取资源,从而减少网络请求次数,提高网页的加载速度和用户体验。缓存分为强缓存和协商缓存两种模式。一.强缓存强缓存是指浏览器直接从本地缓存中获取资源,而不需要向web服务器发出网络请求。这是因为浏览器在第一次请求资源时,服务器会在响应头中添加相关缓存的响应头,以表明该资源的缓存策略。常见的强缓存响应头如下所述:Cache-ControlCache-Control响应头是用于控制强制缓存和协商缓存的缓存策略。该响应头中的指令如下:max-age:指定该资源在本地缓存的最长有效时间,以秒为单位。例如:Ca

  6. IDEA 2022 创建 Spring Boot 项目详解 - 2

    如何用IDEA2022创建并初始化一个SpringBoot项目?目录如何用IDEA2022创建并初始化一个SpringBoot项目?0. 环境说明1.  创建SpringBoot项目 2.编写初始化代码0. 环境说明IDEA2022.3.1JDK1.8SpringBoot1.  创建SpringBoot项目        打开IDEA,选择NewProject创建项目。        填写项目名称、项目构建方式、jdk版本,按需要修改项目文件路径等信息。        选择springboot版本以及需要的包,此处只选择了springweb。        此处需特别注意,若你使用的是jdk1

  7. 详解Unity中的粒子系统Particle System (二) - 2

    前言上一篇我们简要讲述了粒子系统是什么,如何添加,以及基本模块的介绍,以及对于曲线和颜色编辑器的讲解。从本篇开始,我们将按照模块结构讲解下去,本篇主要讲粒子系统的主模块,该模块主要是控制粒子的初始状态和全局属性的,以下是关于该模块的介绍,请大家指正。目录前言本系列提要一、粒子系统主模块1.阅读前注意事项2.参考图3.参数讲解DurationLoopingPrewarmStartDelayStartLifetimeStartSpeed3DStartSizeStartSize3DStartRotationStartRotationFlipRotationStartColorGravityModif

  8. VMware虚拟机与本地主机进行磁盘共享(详解) - 2

    VMware虚拟机与本地主机进行磁盘共享前提虚拟机版本为Windows10(专业版,不是可能有问题)本地主机为家庭版或学生版(此版本会有问题,但有替代方式)最好是专业版VMware操作1.关闭防火墙,全部关闭。2.打开电脑属性3.点击共享-》高级共享-》权限4.如果没有everyone,就添加权限选择完全控制,然后应用确定。5.打开cmd输入lusrmgr.msc(只有专业版可以打开)如果不是专业版,可以跳过这一步。点击用户-》administrator密码要复杂密码,否则不行。推荐admaiN@1234类型的密码。设置完密码,点击属性,将禁用解开。6.如果虚拟机的windows不是专业版,可

  9. ElasticSearch之 ik分词器详解 - 2

    IK分词器本文分为简介、安装、使用三个角度进行讲解。简介倒排索引众所周知,ES是一个及其强大的搜索引擎,那么它为什么搜索效率极高呢,当然和他的存储方式脱离不了关系,ES采取的是倒排索引,就是反向索引;常见索引结构几乎都是通过key找value,例如Map;倒排索引的优势就是有效利用Value,将多个含有相同Value的值存储至同一位置。分词器为了配合倒排索引,分词器也就诞生了,只有合理的利用Value,才会让倒排索引更加高效,如果一整个Value不进行任何操作直接进行存储,那么Value和key毫无区别。分词器Analyzer通常会对Value进行操作:一、字符过滤,过滤掉html标签;二、分

  10. Educational Codeforces Round 146 (Rated for Div. 2)(B,E详解) - 2

    题外话:抑郁场,开局一小时只出A,死活想不来B,最后因为D题出锅ura才保住可怜的分。但咱本来就写不到DB-LongLegs(数论)本题题解法一学自同样抑郁的知乎作者幽血魅影的题解,有讲解原理。法二来着知乎巨佬cup-pyy(大佬说《不难发现》呜呜)题意三种操作:向上走mmm步向右走mmm步给自己一次走的步数加111,即使得m=m+1m=m+1m=m+1问从(0,0)(0,0)(0,0)走到(a,b)(a,b)(a,b)的最小操作次数,值得注意的是操作三不可逆。解析假设我们最终一步的大小增长到mmm,那么在这个过程中我能以[1,m][1,m][1,m](当步数增长到该数时)之间的任何数字向上或

随机推荐