在 C++ 中对共享指针的向量进行排序

Sorting Vectors Of Shared Pointers in C++

假设我们有一个简单的 class:

class Data {
private:
   std::string m_name;
   unsigned m_id;

   Data( const std::string& name, unsigned id ) :
   m_name( name ), m_id( id ) {}

   void setName( const std::string& name ) { m_name = name; }
   void setId( unsigned id ) { m_id = id; }

   std::string getName() const { return m_name; }
   unsigned    getId() const { return m_id; }
}; // Data

现在假设我们有另一个 class 在从文件中读入的智能指针容器中填充和管理它们。

Database.h

#ifndef DATABASE_H
#define DATABASE_H

class Data; 

class Database {
private:
    std::string m_filename;
    std::vector<std::shared_ptr<Data>> m_vpData;

public:
    explicit Database( const std::string& filename );        

    void displayByOption( unsigned option = 0 );

private:
    void parseFile();
    void display() const;

    // Sorting Methods Used for std::sort
    bool sortByName( const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2 );
    bool sortById( const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2 );

}; // Database

#endif // DATABASE_H

我对文件数据的打开、读取和解析没有任何问题。我正在努力解决的是要传递给 std::sort 的方法。 这些是Database.cpp中的显示方法,会受到关注。

注意:我的标准 header 中确实包含 <algorithm><functional> 以及其他适当的 stl header。

Database.cpp

// -------------------------------------------------------------------------
// display() - Shouldn't be an issue, but here for reference
void Database::display() const {
    for ( unsigned u = 0; u < m_vpData.size(); u++ ) {
        std::cout << m_vpData[u]->getName() << "/t"
                  << m_vpData[u]->getId() << "/t" << std::endl;
     } 
} // display

// -------------------------------------------------------------------------
// displayByOption() - This function itself isn't an issue
// But the std::sort method is generating a compiler error
void Database::displayByOption( unsigned option ) {
    switch( option ) {
        case 1: { // Sort By Name
             std::sort( m_vpData.begin(), m_vpData.end(), sortByName );
             display();
             break;
         }
         case 2: { // Sort By Id
             std::sort( m_vpData.being(), m_vpData.end(), sortById );
             display();
             break;
         } 
         default : { // Unsorted - Order Read In By File
             display();
         }
     }
} // displayByOption

// -------------------------------------------------------------------------
// sortByName()
bool Database::sortByName( const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2 ) {
     return d1->getName() < d2->getName();
 } // sortByName

 // ------------------------------------------------------------------------
 // sortById()
 bool Database::sortById( const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2 ) {
     return d1->getId() < d2->getId();
 } // sortById

这是生成的编译器错误

1>------ Build started: Project: FileParse, Configuration: Debug Win32 ------
1>  CharacterDatabase.cpp
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(66): error C3867: 'CharacterDatabase::sortByName': non-standard syntax; use '&' to create a pointer to member
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(66): error C2672: 'std::sort': no matching overloaded function found
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(66): error C2780: 'void std::sort(_RanIt,_RanIt)': expects 2 arguments - 3 provided
1>  c:\program files (x86)\microsoft visual studio 14.0\vc\include\algorithm(3209): note: see declaration of 'std::sort'
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(71): error C3867: 'CharacterDatabase::sortByHealth': non-standard syntax; use '&' to create a pointer to member
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(71): error C2672: 'std::sort': no matching overloaded function found
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(71): error C2780: 'void std::sort(_RanIt,_RanIt)': expects 2 arguments - 3 provided
1>  c:\program files (x86)\microsoft visual studio 14.0\vc\include\algorithm(3209): note: see declaration of 'std::sort'
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(76): error C3867: 'CharacterDatabase::sortByExperience': non-standard syntax; use '&' to create a pointer to member
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(76): error C2672: 'std::sort': no matching overloaded function found
1>c:\users\skilz80\documents\visual studio 2015\projects\fileparse\fileparse\characterdatabase.cpp(76): error C2780: 'void std::sort(_RanIt,_RanIt)': expects 2 arguments - 3 provided
1>  c:\program files (x86)\microsoft visual studio 14.0\vc\include\algorithm(3209): note: see declaration of 'std::sort'
========== Build: 0 succeeded, 1 failed, 0 up-to-date, 0 skipped ==========

我通常很擅长找出编译器错误,但这一次让我难住了。我已经使用直接存储到容器中的基本 class 类型完成了类似这样的基本排序方法,但这是我第一次尝试在 object 的共享指针存储在向量。

我不确定的是一些事情。

任何想法都会非常有帮助!

是什么导致此文件无法编译并生成所提供的错误消息?唯一的区别是我发布的内容和编辑器中的内容之间的 class 名称。另外,对属于 class object 的 vector<shared_ptr<Object>> 进行排序的最简单和最有效的方法是什么?

最讽刺的是,我已经在网上搜索了几个小时以找到正确解决我的问题的方法,所以我决定在这里询问社区。在等待答案的过程中,不久之后,我找到了解决问题的可能方法。至于现在,我能够编译和构建解决方案,但我还没有测试它是否按预期工作。这是我找到的解决编译问题的方法。

我将我的数据库 class 中的 3 个排序函数声明和定义从上面的先前签名更改为以下内容:

static bool sortBy ( std::shared_ptr<Data> d1, std::shared_ptr<Data> d2 );

然后我将它们从私人部分移到了 public 部分。我将函数设为 public 和静态,我从参数中删除了 const 引用。现在我的源文件可以编译,并且可以零错误地构建。现在测试我的 class.

只是时间问题

编辑

在阅读了社区成员对我上述问题的评论后,我认为主要问题是函数不是静态的。所以现在我有了方法调用的静态存储;我将尝试查看是否可以将 const 引用添加回函数签名。

问题是您不能简单地将非静态成员函数作为 std::sort 谓词的一部分传递。

根据您现在的代码,最简单的解决方案是创建 Database::sortyByNameDatabase::sortById static 成员函数。

static bool sortByName( const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2 );

static bool sortById( const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2 );

请注意,还有各种其他解决方案,例如使用 boost::bind、函数对象/仿函数,以及使用 C++11、lambda。我不会讨论它们,只是为了给你其他选择。

您不能在std::sort()中使用非静态成员函数,请使您的函数成为非成员函数或使它们成为 静态成员 (正如您已经想到的那样)。带有非成员排序函数的代码实现。

Database.h

#ifndef DATABASE_H
#define DATABASE_H

#include "Data.h"

class Data;

class Database
{
private:
    std::string m_filename;
    std::vector<std::shared_ptr<Data>> m_vpData;

public:
    explicit Database(const std::string& filename);

    void displayByOption(unsigned option = 0);

private:
    void parseFile();
    void display() const;

    // Sorting Methods Used for std::sort


}; // Database

bool sortByName(const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2);
bool sortById(const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2);

#endif // DATABASE_H

非成员排序函数实现(通知删除 Database::

// -------------------------------------------------------------------------
// sortByName()
bool sortByName(const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2)
{
    return d1->getName() < d2->getName();
} // sortByName

// ------------------------------------------------------------------------
// sortById()
bool sortById(const std::shared_ptr<Data>& d1, const std::shared_ptr<Data>& d2)
{
    return d1->getId() < d2->getId();
} // sortById

一个example如何使用std::sort