依据类的成员来对类进行排序的泛型方法

类别:编程语言 点击:0 评论:0 推荐:

//最近在工作中,遇到这样一个问题:
//一个类中有十几个成员数据,我需要根据不同的成员提供不同的排序方法. 如下:

class Cell
{
public:
    string name;
    string time;
    int   tchDrop;
    double traffic;
    // more …
};

//最近正在学习STL,所以我想结合所学的知识,设计出一种通用的方法,只要提
//供类的成员指针和排序准则(即函数对象),就可
//以同STL算法搭配,对类成员进行比较排序,比较等.

template<typename T,typename C,typename Pred=less<T> >
class Comp_Mem_Data
              :public binary_function<C,C,bool>
{
public:
 Comp_Mem_Data( T (C::*pmd) , Pred pred=Pred() )
        //Pred类采用默认构造函数,并不一定都适用。要注意。 
  :m_pmd(pmd),m_pred(pred)
 {};

 //比较 两个类对象 的成员
 bool operator() (const C& a , const C& b) const
 {
  return m_pred( a.*m_pmd , b.*m_pmd );
 };

private:
 T  (C::*m_pmd); //类成员指针
 Pred m_pred;   //比较函数
};

//仿照标准库中的 bind1st,bind2nd 的做法,提供一个方便的调用形式
template<typename T,typename C>
Comp_Mem_Data<T,C,less<T> >  comper_mem_data( T (C::*pmd) )
{
 return Comp_Mem_Data<T,C,less<T> >(pmd,less<T>() );
}
template<typename T,typename C,typename Pred>
Comp_Mem_Data<T,C,Pred>  comper_mem_data( T (C::*pmd) ,Pred pred )
{
 return Comp_Mem_Data<T,C,Pred>(pmd,pred);
}

//以下就可以开始比较了.
// sort(vCell.begin(),vCell.end(), comper_mem_data( &Cell::name) );
//sort(vCell.begin(),vCell.end(), comper_mem_data( &Cell::traffic , greater<double>() );
//等等

//以下是测试程序 在BCB5.5 、 MSVC6.0 中测试通过

#include <stdlib.h>
#include <stdio.h>
#include <time.h>
#include <iostream>
#include <string>
#include <algorithm>
#include <vector>
#include <functional>
using namespace std;

inline ostream& operator << (ostream& out,const Cell& cell)
{
 out<<cell.name<<" | "
  <<cell.time<<" | "
  <<cell.tchDrop<<" | "
  <<cell.traffic<<" | "
  // more …
  <<endl;
 return out;
}

template<typename Container>
void output(Container cont) 
{
 cout<<"cell name | time | tchDrop | traffic"<<endl;
 for(int j=0;j<cont.size();++j)
  cout<<cont[j];
}

int main()
{
 vector<Cell> vCell;
 Cell  cl;
 srand( (unsigned)time( NULL ) );
 for(int i=0;i<10; ++i)
 {
  cl.name=static_cast<char>(i+'A');
  cl.time=static_cast<char>(rand()%10+'1');
  cl.tchDrop=rand()%100;
  cl.traffic=rand()*100.0/RAND_MAX;
  vCell.push_back(cl);
 }

 sort(vCell.begin(),vCell.end(), comper_mem_data(&Cell::tchDrop) );
 output(vCell);
 sort(vCell.begin(),vCell.end(), comper_mem_data(&Cell::traffic, greater<double>() ) );
 output(vCell);

 vector<Cell>::const_iterator iter=find_if( vCell.begin() , vCell.end() ,
         bind2nd(comper_mem_data(&Cell::name , equal_to<string>() ),cl )
 );
 cout<<((iter!=vCell.end() ) ? ("cl is find") : ("cl isn't find") )<<endl;

 char lc;
 cin>>lc;
 return 0;
}

//程序的一个输出如下:
/*
cell name | time | tchDrop | traffic
A | 3 | 2 | 47.0168 |
C | 1 | 2 | 66.5151 |
D | 5 | 19 | 86.5017 |
F | 9 | 23 | 11.1087 |
E | 5 | 78 | 75.7897 |
G | 7 | 84 | 54.2253 |
H | 7 | 88 | 61.1774 |
I | 1 | 90 | 59.2578 |
J | 3 | 94 | 89.9625 |
B | 6 | 99 | 89.2514 |
cell name | time | tchDrop | traffic
J | 3 | 94 | 89.9625 |
B | 6 | 99 | 89.2514 |
D | 5 | 19 | 86.5017 |
E | 5 | 78 | 75.7897 |
C | 1 | 2 | 66.5151 |
H | 7 | 88 | 61.1774 |
I | 1 | 90 | 59.2578 |
G | 7 | 84 | 54.2253 |
A | 3 | 2 | 47.0168 |
F | 9 | 23 | 11.1087 |
cl is find
//*/

//用同样的方法,还可以制作出比较两个类成员函数返回值的泛型类,
// 类数据成员 与 同类型的常数参数相比较的泛型类来。
//此处不一一列出。

//以上不当之处,还请各位大虾不吝指教。

本文地址:http://com.8s8s.com/it/it29148.htm