分类目录归档:数据结构

C++ STL容器——set

      容器用于存储数据,在STL中,set是一个类似数学中”集合”的容器,可以进行交集并集的计算,且容器中不允许出现重复元素,set在日常使用中经常见到,值得注意的是,set内部使用二叉树存储数据,运算速度很快。

参考手册:http://www.cplusplus.com/reference/set/set/

      set的声明如下,

template < class T,                        // set::key_type/value_type
           class Compare = less<T>,        // set::key_compare/value_compare
           class Alloc = allocator<T>      // set::allocator_type
           > class set;

      第一个参数T是容器类型,第二个和第三个参数都提供的默认值,其中第二个参数是 比较器,用于给容器中元素进行大小排序,默认为less<T>,也可以自己指定,第三个参数是分配器,该参数指定使用哪个分配器对象管理内存,如果省略该模板参数的值,则容器模板将默认使用allocator<T>,这个类使用new和delete

set容器的常用方法:

初始化:

set容器常用的初始化方法为:

默认初始化

使用区间初始化

使用复制构造函数初始化

#include <iostream>
#include <string>
#include <set>
#include <cmath>
 
struct Point { double x, y; };
struct PointCmp {
    bool operator()(const Point& lhs, const Point& rhs) const { 
        return std::hypot(lhs.x, lhs.y) < std::hypot(rhs.x, rhs.y); 
    }
};
 
int main()
{
  // (1) 默认初始化器!!常用
  std::set<std::string> a;
  a.insert("cat");
  a.insert("dog");
  a.insert("horse");
  for(auto& str: a) std::cout << str << ' ';
  std::cout << '\n';
 
  // (2) 迭代器初始化器!!常用//使用区间进行初始化,也可以用数组的区间进行初始化。
  std::set<std::string> b(a.find("dog"), a.end());
  for(auto& str: b) std::cout << str << ' ';
  std::cout << '\n';
 
  // (3) 复制构造函数
  std::set<std::string> c(a);
  c.insert("another horse");
  for(auto& str: c) std::cout << str << ' ';
  std::cout << '\n';
 
  // (4) 移动构造函数
  std::set<std::string> d(std::move(a));
  for(auto& str: d) std::cout << str << ' ';
  std::cout << '\n';
  std::cout << "moved-from set is ";
  for(auto& str: a) std::cout << str << ' ';
  std::cout << '\n';
 
  // (5) initializer_list 构造函数
  std::set<std::string> e {"one", "two", "three", "five", "eight"};
  for(auto& str: e) std::cout << str << ' ';
  std::cout << '\n';
 
  // 自定义比较器
  std::set<Point, PointCmp> z = {{2, 5}, {3, 4}, {1, 1}};
  z.insert({1, -1}); // 这会失败,因为 1,-1 的长度等于 1,1
  for(auto& p: z) std::cout << '(' << p.x << ',' << p.y << ") ";
  std::cout << '\n';
}




//输出:

cat dog horse 
dog horse 
another horse cat dog horse 
cat dog horse 
moved-from set is 
eight five one three two 
(1,1) (3,4) (2,5)

insert():插入元素

常用insert(A,B)//A,B分别是两个迭代器,或两个数组中的指针,把[A,B)区间的元素插入容器。

insert(E)//直接插入一个元素

 

erase():删除操作:

iterator  erase (const_iterator position);//删除迭代器指向位置的元素
size_type erase (const value_type& val);//删除这个元素
iterator  erase (const_iterator first, const_iterator last);//删除这个区间的元素

find():查找元素

如果找到了,则返回指向这个元素的迭代器,如果没有找到,则返回一个指向最后一个元素尾部的迭代器.   (end()的返回值,这个迭代器指向最后一个元素的下一个元素,所以无效)

// set::find
#include <iostream>
#include <set>

int main ()
{
  std::set<int> myset;
  std::set<int>::iterator it;

  // set some initial values:
  for (int i=1; i<=5; i++) myset.insert(i*10);    // set: 10 20 30 40 50

  it=myset.find(20);
  myset.erase (it);
  myset.erase (myset.find(40));

  std::cout << "myset contains:";
  for (it=myset.begin(); it!=myset.end(); ++it)
    std::cout << ' ' << *it;
  std::cout << '\n';

  return 0;
}

 


    begin()//返回一个指向元素开头的迭代器(ps:关于迭代器,我们可以把它当作指针)

    end()//返回一个指向最后一个元素尾部的迭代器

// set::begin/end
#include <iostream>
#include <set>

int main ()
{
  int myints[] = {75,23,65,42,13};
  std::set<int> myset (myints,myints+5);//使用数组初始化set容器中的元素

  std::cout << "myset contains:";
  for (std::set<int>::iterator it=myset.begin(); it!=myset.end(); ++it)
    std::cout << ' ' << *it;

  std::cout << '\n';

  return 0;
}

//输出:myset contains: 13 23 42 65 75

rbegin()/rend():返回反向迭代器

// set::rbegin/rend
#include <iostream>
#include <set>

int main ()
{
  int myints[] = {21,64,17,78,49};
  std::set<int> myset (myints,myints+5);

  std::set<int>::reverse_iterator rit;

  std::cout << "myset contains:";
  for (rit=myset.rbegin(); rit != myset.rend(); ++rit)
    std::cout << ' ' << *rit;

  std::cout << '\n';

  return 0;
}

//输出:myset contains: 78 64 49 21 17 
//参数中默认的比较器是从小到大排列的,所以这里反向输出的容器元素的值

cbegin()/cend():返回一个const的迭代器

crbegin()/crend():返回const的反向迭代器


empty()//测试容器是否为空,为空则返回true.

size()//返回容器有多少元素.

max_size()//返回容器存储元素的最大值.

 



最后介绍一下set在集合意义上的操作

set_union()//求并集

set_intersection//交集

set_difference//差集

这三个函数的接口是相同的,他们都有5个参数,分别是:第一个集合的区间,第二个集合的区间,输出迭代器,这个迭代器指出要将操作的结果输出到哪里。

今天写累了,明天再写

待更新

 

 

使用栈(stack)简单地判断表达式合法性

      栈Stack是一种常见的数据结构,常用于函数调用,表达式处理等方面,栈结构就像是一个井,push操作就是把东西扔进井里面,pop操作就是把东西捞上来,很显然,后面仍下去的东西会先被捞上来,所以我们说Stack结构是一种先进后出,后进先出的结构。

      大家可能注意到,我们写代码的时候,比较智能的编辑器都会有一些错误提示,如果我们写了“(”而没有对应的”)“,那么就会看到提示,所以这是咋做到的?其实是使用了Stack栈结构。下面我们看一下使用Stack进行验证的规则:

      做一个空栈,读取字符串,依次对每个字符进行处理,如果不是(){}等符号字符,则跳过,如果是一个起始符号例如”(“,”{“,则把字符仍入栈中,如果字符是一个结束符合,那么当栈空时报错,当栈非空时,弹出顶部元素,验证是否与当前处理的字符匹配,若匹配,处理下一个,若不匹配,产生错误。最后当字符结束时,查看栈是否为空,如果栈中依然有元素,则报错。

      看下面的代码(很简单的,耐心看一下哈):

#include <iostream>
#include <stack>
using namespace std;
bool isTrueExpression(char exp[]);//主要的验证函数
bool isMarch(char,char);
int main(){
    char exp1[]="int main(void){for(){}return 0;}";//三个待验证的表达式
    char exp2[]="int main(void){for({)}return 0;}";
    char exp3[]="int main(void{for(){}return 0;}(";
    if(isTrueExpression(exp1))  cout<<"Exp1:True"<<endl;
    if(isTrueExpression(exp2))  cout<<"Exp2:True"<<endl;
    if(isTrueExpression(exp3))  cout<<"Exp3:True"<<endl;
    return 0;
}
bool isTrueExpression(char exp[]){
    stack<char> sta;
    for(;(*exp)!='\0';){//依次对字符进行处理,直到结尾
        //cout<<"handle:"<<*exp<<endl;
        if((*exp)=='('|(*exp)=='{'|(*exp)=='['){//如果字符是( { [等起始符合,那么扔入栈中
            sta.push(*exp);
            //cout<<"push:"<<*exp<<endl;
        }else if((*exp)==')'|(*exp)=='}'|(*exp)==']'){//如果是) } ]等符合,那么继续进行验证
            if((int)sta.size()==0){//如果栈是空的,返回false
                //cout<<"return F size==0"<<endl;
                return false;
            }else if(!isMarch(sta.top(),*exp)){
             //如果盏不空,取栈顶元素进行匹配,如果匹配失败,返回false
                return false;
            }else{//匹配成功,则把栈顶元素弹出
                //cout<<"pop:"<<*exp<<endl;
                sta.pop();
            }
        }
        if(*++exp=='\0'&&(int)sta.size()!=0){//验证是否到达结尾,
            //如果到达结尾,验证Stack是否为空,如果栈非空,则返回false
            return false;
        }
    }
    return true;
}
bool isMarch(char c1,char c2){
    //cout<<"比较"<<c1<<"和"<<c2<<endl;
    if(c1=='('){
        if(c2==')') return true;
    }else if(c1=='{'){
        if(c2=='}') return true;
    }else if(c1=='['){
        if(c2==']') return true;
    }
    //cout<<"不匹配"<<endl;
    return false;
}

以上代码输出为:

Exp1:True