I have an hash and i want to sort based on the keys with upper case words appearing just before the lowercase words.
Example:
JANE
jane
JIM
Use a custom sort which first compares the items based on their lowercased representations (so that all variations of "jane" appear before variations of "jim"), then resolves ties by doing a default ASCII comparison (where uppercase comes before lowercase):
perl -e 'print join "\n", sort { lc $a cmp lc $b || $a cmp $b } qw( jim JANE jane JIM )'
Output:
JANE
jane
JIM
jim
Although it may seem like overkill for this operation, the standard Unicode::Collate and Unicode::Collate::Locale modules are made for this sort of thing. They also sort non-ASCII data alphabetically, which the normal sort
will not do.
use utf8;
@names = qw[ jim JANE jane JIM josé josie Mary María mark ];
@sorts = sort @names;
That gives you the sort order of
JANE JIM Mary María jane jim josie josé mark
which nobody wants. This is much better:
use utf8;
use Unicode::Collate;
@names = qw[ jim JANE jane JIM josé josie Mary María mark ];
$coll = new Unicode::Collate;
@sorts = $coll->sort(@names);
That gives you
jane JANE jim JIM josé josie María mark Mary
If you want uppercase before lowercase, specify that this way:
use utf8;
use Unicode::Collate;
@names = qw[ jim JANE jane JIM josé josie Mary María mark ];
$coll = new Unicode::Collate upper_before_lower => 1;
@sorts = $coll->sort(@names);
print "@sorts\n";
which yields:
JANE jane JIM jim josé josie María mark Mary
You can use collation objects’ cmp
method on a pair of strings in the customary fashion, like
#!/usr/bin/env perl
use 5.10.1;
use strict;
use autodie;
use warnings qw[ FATAL all ];
use utf8;
use open qw[ :std IO :utf8 ];
use Unicode::Collate;
my @names = qw[ fum fee fie foe ];
my $coll = Unicode::Collate->new;
my @sorts = $coll->sort(@names);
say "@names => @sorts\n";
for (
my($a, $b) = splice @names, 0, 2;
2 == grep {defined} $a, $b;
($a, $b) = ($b, shift @names)
)
{
given ($coll->cmp($a, $b)) {
when (-1) { say "$a < $b" }
when ( 0) { say "$a = $b" }
when (+1) { say "$a > $b" }
default { die "NOT REACHED" }
}
}
which produces:
fum fee fie foe => fee fie foe fum
fum > fee
fee < fie
fie < foe
Now consider a list of words like this:
sát sot sät sét sæt ssét sat tot ßet SET set seat ſAT ſet saet SSET
If you run the default sort on that, you get the virtually useless:
SET SSET saet sat seat set sot ssét sát sät sæt sét tot ßet ſAT ſet
And a case-sensitive sort is really no better:
use utf8;
@names = qw[ sát sot sät sét sæt ssét sat tot ßet SET set seat ſAT ſet saet SSET ];
@sorts = sort {
lc $a cmp lc $b
||
$a cmp $b
} @names;
print "@sorts\n";
producing the still stupid-and-wrong:
saet sat seat SET set sot SSET ssét sát sät sæt sét tot ßet ſAT ſet
But here it is with a standard Unicode sort:
use utf8;
use Unicode::Collate;
@names = qw[ sát sot sät sét sæt ssét sat tot ßet SET set seat ſAT ſet saet SSET ];
$coll = new Unicode::Collate upper_before_lower => 1;
@sorts = $coll->sort(@names);
print "@sorts\n";
producing the ‘correcter’ (read: infinitely preferable) version of:
saet sæt sät sat sát ſAT seat SET set sét ſet sot SSET ssét ßet tot
The Unicode::Collate module is pretty fast, so you should not hestitate to use it on your route character sorting needs. But sometimes that just isn’t enough. That’s because different languages have different ideas of alphabets.
BTW, those are also good examples why “ever hardcoding [a-z]
into your program is always wrong, sometimes.” It’s full of idiotic and even insulting assumptions. Note that all but the last three of these are actually considered Latin alphabets! That’s the same script as we use in English. In representing English text, I’ve variously had to deal with learnèd, Æneid, poſt, Laȝamon, résumé, 1ˢᵗ, MᶜKinley, Van Dijke, Cañon City Colorado, œnology, Dzur, rôle, ⅷ, première, Bjørn, naïve, coöperate, façade, café, Merððyn, archæology, and even tschüß. Repeat the mantra: “Hardcoding [a-z]
into your program is always wrong, sometimes.” Just Say No!
The Unicode::Collate::Locale module handles local sorting conventions. Just as English phonebooks and bookshelves have special ways of sorting names so that it doesn’t metter whether you’ve spelt something McBride or MacBride, the German-speaking world sorts their names such that Händel and Haendel are the same. That’s why without diacritics, one must obligatorily write über‑ as ueber‑ and Übermensch as Uebermensch. A locale sort knows to do this:
use utf8;
use Unicode::Collate::Locale;
@names = qw[ sát sot sät sét sæt ssét sat tot ßet SET set seat ſAT ſet saet SSET ];
$coll = new Unicode::Collate::Locale::
locale => de__phonebook,
upper_before_lower => 1,
;
@sorts = $coll->sort(@names);
print "@sorts\n";
now produces
saet sæt sät sat sát ſAT seat SET set sét ſet sot SSET ssét ßet tot
It’s remarkable how different from one’s own other countries’ locale conventions can be. In the Spanish locale ("es"), ñ is a letter that comes after n and before o. That means that the correct sort of
raña rastrillo radio rana rápido ráfaga ranúnculo
is
radio ráfaga rana raña ranúnculo rápido rastrillo
Say those all really fast with a fully-rolled rr to loosen your tongue. :)
The "es__traditional" locale is a little different; historically, chocolate came after color in the Spanish dictionary, unlike the way it works in Enlgish. That’s because ch came after c and before d, while ll came after l and before m. That means that this sequence:
lástima laña llama ligante
cidra caliente color chocolate con churros
pero pera Perú perro periglo peste
sorts to
caliente cidra color con chocolate churros
laña lástima ligante llama
pera periglo pero perro Perú peste
To get the keys in order, apply sort
with a custom sort function on the keys of the hash.
my %hash = ( JANE => 1, jane => 2, JIM => 3, jim => 4 );
my @sorted_keys = sort {
lc $a cmp lc $b
|| $a cmp $b
} keys %hash;
This custom sort function compares strings first as if they were of the same case, and if equal, takes case into account.
Try:
@list = ("jane","JIM","JANE","jim");
print sort { uc $a cmp uc $b or $a cmp $b } @list;