[解決!Python]リスト(配列)をソートしたり、逆順にしたりするには(sort/reverseメソッド、sorted/reversed関数)解決!Python

sortメソッドやsorted関数でリスト(配列)の要素をソートしたり、reverseメソッドやreversed関数で要素を逆順に並べたりする方法を紹介する。

» 2020年12月22日 05時00分 公開
[かわさきしんじDeep Insider編集部]

この記事は会員限定です。会員登録(無料)すると全てご覧いただけます。

「解決!Python」のインデックス

連載目次

# リストの要素をソートする(インプレース)
mylist = [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
mylist.sort()
print(mylist)  # [0, 1, 2, 5, 6, 7, 9, 15, 16, 18]

# リストの要素をソートした新しいリストを作成する
mylist = [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
newlist = sorted(mylist)
print('mylist:', mylist)  # mylist: [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
print('newlist:', newlist)  # newlist: [0, 1, 2, 5, 6, 7, 9, 15, 16, 18]

# リストの要素をソートする(インプレース/逆順)
mylist = [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
mylist.sort(reverse=True)
print(mylist)  # [18, 16, 15, 9, 7, 6, 5, 2, 1, 0]

# ソートに使用するキーを決定する関数を指定
mylist = [-1.7, 0.1, 4.5, -1.5, -4.6]
mylist.sort(key=abs)  # 絶対値順でソート
print(mylist)  # [0.1, -1.5, -1.7, 4.5, -4.6]

# ラムダ式を使って、リストのリストをソートする
mylist = [[0, 1, 2], [1, 2, 0], [2, 0, 1]]
mylist.sort(key=lambda x: x[1])  # 1番目の要素をキーとしてリストをソート
print(mylist)  # [[2, 0, 1], [0, 1, 2], [1, 2, 0]]

# keyキーワード引数とreverseキーワード引数を使用
mylist = [-1.7, 0.1, 4.5, -1.5, -4.6]
mylist.sort(key=abs, reverse=True# 絶対値を基に降順でソート
print(mylist)  # [-4.6, 4.5, -1.7, -1.5, 0.1]

# 要素を逆順に並べ替える(インプレース)
mylist = list(range(5))  # [0, 1, 2, 3, 4]
mylist.reverse()
print(mylist)  # [4, 3, 2, 1, 0]

# 要素を逆順に並べ替えた新しいイテレータを作成する
mylist = list(range(5))  # [0, 1, 2, 3, 4]
iterator = reversed(mylist)  # reversed関数の戻り値はイテレータ
print(iterator)  # <list_reverseiterator object at ...>
newlist = list(iterator)  # イテレータからリストを作成
print('mylist:', mylist)  # mylist: [0, 1, 2, 3, 4]
print('newlist:', newlist)  # newlist: [4, 3, 2, 1, 0]


sortメソッドとsorted関数

 リスト(配列)の要素をソートするには、リストが持つsortメソッドかPythonに組み込みのsorted関数を使う。前者はリストをインプレースに変更する。つまり、元のリストの要素が直接変更される。後者は、リストの要素をソートした結果の新しいリストを作成する。

 以下に例を示す。

# リストの要素をソートする(インプレース)
mylist = [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
mylist.sort()
print(mylist)  # [0, 1, 2, 5, 6, 7, 9, 15, 16, 18]

# リストの要素をソートした新しいリストを作成する
mylist = [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
newlist = sorted(mylist)
print('mylist:', mylist)  # mylist: [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
print('newlist:', newlist)  # newlist: [0, 1, 2, 5, 6, 7, 9, 15, 16, 18]


 1つ目の例ではリストmylistのsortメソッドを呼び出している。このため、リストの要素がインプレースで並べ替えられている。対して、2つ目の例ではsorted関数にリストmylistを渡しているので、元のリストでは要素は以前のままで、sortedメソッドの戻り値であるnewlistでは要素がソートされている。

 sortメソッドやsorted関数を使ってソートを行う際には、内部的には<演算子を使ってそれぞれの要素の大小関係が比較されている。このため、リストに含まれている全ての要素同士をこの演算子で比較できる必要がある。例えば、整数と浮動小数点数の2種類の値だけがリストの要素であれば、問題なくソートできる。

mylist = [3.0, 0, 1.1, 2]
mylist.sort()  # 整数と浮動小数点数の比較はサポートされている
print(mylist)  # [0, 1.1, 2, 3.0]


 これに対して、リストに文字列と整数値が含まれているというときには例外となる。

mylist = [3, 0, '1', 2]
mylist.sort()  # TypeError:整数と文字列の比較はサポートされていない


 このようなときには、何らかの形で要素の型を互換性のあるものに統一できるのであれば、以下で紹介するkeyキーワード引数に型変換するための関数を指定すれば、ソートすることは可能だ(そうした行為に意味があるかどうかは時と場合によるだろう)。

mylist = [3, 0, '1', 2]
mylist.sort(key=int)  # 文字列を整数値に。「key=str」も同様
print(mylist)  # [0, '1', 2, 3]


 sortメソッドとsorted関数にはreverseとkeyという2つのキーワード引数がある。

 reverseキーワード引数はソートを昇順/降順に行うかどうかを指定するものだ。これをTrueに指定すると、ソートの結果は通常とは逆の順序(降順)になる。

 以下に例を示す。

mylist = [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
mylist.sort(reverse=True)
print(mylist)  # [18, 16, 15, 9, 7, 6, 5, 2, 1, 0]

mylist = [18, 0, 16, 6, 15, 7, 9, 1, 2, 5]
newlist = sorted(mylist, reverse=True)
print(newlist)  # [18, 16, 15, 9, 7, 6, 5, 2, 1, 0]


 先ほども出てきたもう1つのkeyキーワード引数は、要素を比較する際のキーとなる値を決定するのに使う関数を指定する。この関数は1つの引数を受け取り、何らかの値を返すものである必要がある。

 以下に例を示す。

mylist = [-1.7, 0.1, 4.5, -1.5, -4.6]
mylist.sort(key=abs)  # 絶対値順でソート
print(mylist)  # [0.1, -1.5, -1.7, 4.5, -4.6]

mylist = [-1.7, 0.1, 4.5, -1.5, -4.6]
newlist = sorted(mylist, key=abs)
print(newlist)  # [0.1, -1.5, -1.7, 4.5, -4.6]


 この例では、正負の浮動小数点数値がリストの要素となっている。そして、keyキーワード引数にPythonに組み込みのabs関数を指定して、sortメソッドを呼び出している。そのためここでは、リストの各要素をabs関数に渡した結果、つまり各要素の絶対値を基に要素の並べ替えが行われる。

 このように既存の関数(や自作の関数)を指定することもできるが、ラムダ式を使うことも可能だ。例として、リストを要素とするリストで、要素となっているリストをソートすることを考える。keyキーワード引数を指定しなければ、リストとリストが<演算子で比較されるが、簡単なラムダ式を書くだけで、要素となっているリストの何番目かの要素を基にソートを行える。

mylist = [['isshiki', 175, 68], ['kawasaki', 172, 80], ['endo', 180, 75]]
mylist.sort()  # キー関数を指定せずにソート
print(mylist)  # [['endo', 180, 75], ['isshiki', 175, 68], ['kawasaki', 172, 80]]

mylist.sort(key=lambda x: x[1])  # 1番目の要素をキーとしてリストをソート
print(mylist)  # [['kawasaki', 172, 80], ['isshiki', 175, 68], ['endo', 180, 75]]

newlist = sorted(mylist, key=lambda x: x[2])  # 2番目の要素をキーとしてソート
print(newlist)  # [['isshiki', 175, 68], ['endo', 180, 75], ['kawasaki', 172, 80]]


 ここでは[名前, 身長, 体重]という要素で構成されるリストを要素とするリストがある。キー関数を指定せずにソートを行ったのが一番上の例だ。次の例では、keyキーワード引数に「lambda x: x[1]」を指定してソートをしているので、身長の低い順にリストがソートされている。最後の例ではkeyキーワード引数に「lambda x: x[2]」を指定しているので、体重が少ない順にソートが行われている。

 最後に、2つのキーワード引数を使った例も示しておく。説明は不要だろう。

mylist = [-1.7, 0.1, 4.5, -1.5, -4.6]
mylist.sort(key=abs, reverse=True# 絶対値を基に降順でソート
print(mylist)  # [-4.6, 4.5, -1.7, -1.5, 0.1]


 なお、sorted関数にはリストだけではなく、任意の反復可能オブジェクトを与えられる(Pythonが標準で提供している反復可能オブジェクトで、sortメソッドが定義されているのはlist型のみ)。

t = ('foo', 'bar', 'baz')
mylist = sorted(t)  # タプルの要素をソート(戻り値はリストになる)
print(mylist)  # ['bar', 'baz', 'foo']

s = 'python'
mylist = sorted(s)  # 文字列も反復可能オブジェクト
print(mylist)  # ['h', 'n', 'o', 'p', 't', 'y']

t.sort()  # AttributeError:タプルにsortメソッドはない(タプルは変更できない)
s.sort()  # AttributeError:文字列にsortメソッドはない(文字列は変更できない)


reverseメソッドとreversed関数

 リストのreverseメソッドは、その要素をインプレースで逆順に並べ替えるものだ。

mylist = list(range(5))  # [0, 1, 2, 3, 4]
print(mylist)  # [0, 1, 2, 3, 4]
mylist.reverse()
print(mylist)  # [4, 3, 2, 1, 0]


 sortメソッドと同様に、reverseメソッドは元のリストの要素を並べ替える。そうではなく、要素を逆順にした新しいリストが必要であれば、Pythonに組み込みのreversed関数を使用する。ただし、reversed関数はリストではなくイテレータを戻り値とする点には注意すること。

mylist = list(range(5))  # [0, 1, 2, 3, 4]
list_iter = reversed(mylist)  # reversed関数の戻り値はイテレータ
print(list_iter)  # <list_reverseiterator object at 0x...>
newlist = list(list_iter)  # リストにするにはlist関数を使う
print(newlist)  # [4, 3, 2, 1, 0]


 この例では、reversed関数に元のリストを渡して、要素を逆順に列挙するイテレータを取得した後、それをlist関数に渡して要素が逆順に並んだリストを手に入れている。

「解決!Python」のインデックス

解決!Python

Copyright© Digital Advantage Corp. All Rights Reserved.

スポンサーからのお知らせPR

注目のテーマ

Microsoft & Windows最前線2025
AI for エンジニアリング
ローコード/ノーコード セントラル by @IT - ITエンジニアがビジネスの中心で活躍する組織へ
Cloud Native Central by @IT - スケーラブルな能力を組織に
システム開発ノウハウ 【発注ナビ】PR
あなたにおすすめの記事PR

RSSについて

アイティメディアIDについて

メールマガジン登録

@ITのメールマガジンは、 もちろん、すべて無料です。ぜひメールマガジンをご購読ください。