2009-08-31 14 views
9

Próbuję porównać std::string s w sposób zależny od ustawień regionalnych.Ustawienia zależne od lokalizacji dla std :: string

Dla zwykłych strun stylu C, Znalazłem strcoll, który robi dokładnie to, co chcę, po zrobieniu std::setlocale

#include <iostream> 
#include <locale> 
#include <cstring> 

bool cmp(const char* a, const char* b) 
{ 
    return strcoll(a, b) < 0; 
} 

int main() 
{ 
    const char* s1 = "z", *s2 = "å", *s3 = "ä", *s4 = "ö"; 

    std::cout << (cmp(s1,s2) && cmp(s2,s3) && cmp(s3,s4)) << "\n"; //Outputs 0 
    std::setlocale(LC_ALL, "sv_SE.UTF-8"); 
    std::cout << (cmp(s1,s2) && cmp(s2,s3) && cmp(s3,s4)) << "\n"; //Outputs 1, like it should 

    return 0; 
} 

Jednak chciałbym mieć to zachowanie dla std::string również. może po prostu przeciążać operator< coś zrobić jak

bool operator<(const std::string& a, const std::string& b) 
{ 
    return strcoll(a.c_str(), b.c_str()); 
} 

ale wtedy musiałbym się martwić o kodzie za pomocą std::less i std::string::compare, więc nie czuję się dobrze.

Czy istnieje sposób, aby tego rodzaju sortowanie działało w sposób ciągły w sposób ciągły?

Odpowiedz

7

operator() std :: locale jest dokładnie tym, czego szukasz. Aby uzyskać aktualne globalne ustawienia regionalne, użyj domyślnego konstruktora.

+0

To przydaje się. Dzięki temu standardowe zbiory działają bez wysiłku. – CAdaker

7

Biblioteka C++ udostępnia collate facet do sortowania specyficznego dla lokalizacji.

+0

Operator() w ustawieniach narodowych to najprostszy sposób, jaki znam, aby uzyskać do niego dostęp. – AProgrammer

+1

Rozumiem - nie wiedziałem tego. –

0

Po krótkim przeszukiwaniu zdałem sobie sprawę, że jednym ze sposobów, aby to zrobić, może być przeciążenie szablonu std::basic_string, aby utworzyć nową, zlokalizowaną klasę ciągów znaków.

nie ma chyba błędy gazillion w to, ale jako dowód pojęcia:

#include <iostream> 
#include <locale> 
#include <string> 

struct localed_traits: public std::char_traits<wchar_t> 
{ 
    static bool lt(wchar_t a, wchar_t b) 
    { 
     const std::collate<wchar_t>& coll = 
      std::use_facet< std::collate<wchar_t> >(std::locale()); 
     return coll.compare(&a, &a+1, &b, &b+1) < 0; 
    } 

    static int compare(const wchar_t* a, const wchar_t* b, size_t n) 
    { 
     const std::collate<wchar_t>& coll = 
      std::use_facet< std::collate<wchar_t> >(std::locale()); 
     return coll.compare(a, a+n, b, b+n); 
    } 
}; 

typedef std::basic_string<wchar_t, localed_traits> localed_string; 

int main() 
{ 
    localed_string s1 = L"z", s2 = L"å", s3 = L"ä", s4 = L"ö"; 

    std::cout << (s1 < s2 && s2 < s3 && s3 < s4) << "\n"; //Outputs 0 
    std::locale::global(std::locale("sv_SE.UTF-8")); 
    std::cout << (s1 < s2 && s2 < s3 && s3 < s4) << "\n"; //Outputs 1 

    return 0; 
} 

Howerver, to nie wydaje się działać, jeśli oprzeć go na char zamiast wchar_t i nie mam pojęcia dlaczego ...

+0

Powód char nie działa, ponieważ nie używa Unicode (jak w ".UTF-8" Prawdopodobnie używasz ISO/IEC 8859-1. –

+0

Co ma robić '& a + 1'? – 0x499602D2