3
votes

std :: mutex en tant que membre de classe, et stocke l'objet de classe dans le conteneur

Voici un code minimal pour reproduire l'erreur.

>   C:\Program Files (x86)\Microsoft Visual
> Studio\2017\Community\VC\Tools\MSVC\14.16.27023\include\xlocale(319):
> warning C4530: C++ exception handler used, but unwind semantics are
> not enabled. Specify /EHsc    C:\Program Files (x86)\Microsoft Visual
> Studio\2017\Community\VC\Tools\MSVC\14.16.27023\include\utility(405):
> error C2440: '<function-style-cast>': cannot convert from 'initializer
> list' to '_Mypair'    C:\Program Files (x86)\Microsoft Visual
> Studio\2017\Community\VC\Tools\MSVC\14.16.27023\include\utility(405):
> note: No constructor could take the source type, or constructor
> overload resolution was ambiguous
>   ..\examples\json_object\json.cpp(16): note: see reference to function
> template instantiation 'std::pair<std::string,A>
> std::make_pair<std::string,A>(_Ty1 &&,_Ty2 &&)' being compiled            with
>           [
>               _Ty1=std::string,
>               _Ty2=A          ]

Voici l'erreur.

Compilateur d'optimisation Microsoft (R) C / C ++ version 19.16.27026.1 pour x64 Copyright (C) Microsoft Corporation. Tous droits réservés.

#include <iostream>
#include <mutex>
#include <vector>

class A {
    std::mutex mutex;
    public:
    A(){};
};
int main() 
{
    std::vector<std::pair<std::string,A>> aa;
    A a;
    //aa.push_back(std::make_pair(std::string("aa"),A()));
    //aa.push_back(std::make_pair(std::string("aa"),a));
    aa.push_back(std::make_pair(std::string("aa"),std::move(a)));    
}

Erreur similaire pour le compilateur gcc . Lorsque je supprime std :: mutex de la classe OU que je ne pousse pas l'objet sur std :: vector , alors il se compile correctement.


0 commentaires

3 Réponses :



1
votes

Comme indiqué par PW et un indice fourni par freakish, je propose la solution ci-dessous.

#include <iostream>
#include <mutex>
#include <vector>
#include <memory>

class A {
    std::mutex mutex;
    public:
    A(){};
};
int main() 
{
    std::vector<std::pair<std::string,std::shared_ptr<A>>> aa;
    A a;
    //aa.push_back(std::make_pair(std::string("aa"),A()));
    //aa.push_back(std::make_pair(std::string("aa"),a));
    aa.push_back(std::make_pair(std::string("aa"),std::make_shared<A>()));   
}

J'ai modifié mon conteneur pour stocker le pointeur intelligent de l'objet au lieu de l'objet lui-même.


1 commentaires

FWIW, vous devriez vraiment utiliser std :: unique_ptr sauf si vous avez vraiment besoin d'une propriété partagée. 90% du temps, non.



2
votes

Comme P.W. fait remarquer, std :: mutex n'est ni copiable ni déplaçable, et pour une bonne raison. L'intérêt d'avoir un mutex est de se protéger contre l'accès multithread simultané à certaines données. L'opération de déplacement elle-même doit être protégée et le mutex doit être utilisé par l'opération de déplacement.

L'exemple suivant donne à la classe des données mobiles et montre comment le mutex doit être utilisé dans une opération de déplacement (les opérations de copie seraient similaires):

#include <iostream>
#include <mutex>
#include <vector>
#include <memory>

class A {
public:
  A() {};

  // Move constructor
  A(A&& other) {
    std::lock_guard<std::mutex> guard(other.m_mutex);
    m_data = std::move(other.m_data);
  }

  // Move operator
  A& operator=(A&& other) {
    if (this == &other) return *this;

    // Lock this and other in a consistent order to prevent deadlock
    std::mutex* first;
    std::mutex* second;
    if (this < &other) {
      first = &this->m_mutex;
      second = &other.m_mutex;
    } else {
      first = &other.m_mutex;
      second = &this->m_mutex;
    }
    std::lock_guard<std::mutex> guard1(*first);
    std::lock_guard<std::mutex> guard2(*second);

    // Now both this and other are safe to access.  Do the actual data move.
    m_data = std::move(other.m_data);
    return *this;
  }

private:
  std::mutex m_mutex;
  std::unique_ptr<int> m_data;
};

int main() {
  std::vector<std::pair<std::string,A>> aa;
  A a1;
  A a2;
  a1 = std::move(a2);
  aa.emplace_back("aa", std::move(a1));
}


0 commentaires