質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.49%

  • Python

    12220questions

    Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

  • Python 3.x

    10228questions

    Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

pythonの辞書型で順番のキーと要素をスライシングで出力したい

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 652

tenjin

score 230

 前提・実現したいこと

pythonの辞書型で特定の番号のキーと要素を出力しようとしています。
リストの場合は、以下のようにスライシングが可能ですが、
辞書ではキーを指定して要素を出力するのが一般的であるため、
スライシングを用いた方法がみつかりませんでした。

a = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
a1 = a[2:4]     # 2番目から3番目: [2, 3]
a2 = a[2:]      # 2番目から最後: [2, 3, 4, 5, 6, 7, 8, 9]
a3 = a[:3]      # 最初から3番目: [0, 1, 2, 3]

 発生している問題・エラーメッセージ

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: unhashable type: 'slice'

 該当のソースコード

>>> dict = { 'a' : ['apple', 'assimulation'], 'b': ['banana', 'bag'], 'c':['cap', 'cat']}
>>> dict[:1]


ほしい出力

{ 'a' : ['apple', 'assimulation'], 'b': ['banana', 'bag']}

 補足情報(FW/ツールのバージョンなど)

python3.6

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • fuzzball

    2018/09/10 16:43

    で、結局直さないんですね。

    キャンセル

  • tenjin

    2018/09/10 16:59

    a3 = a[:3] # 最初から3番目: [0, 1, 2, 3]の方は修正いたしました。

    キャンセル

  • fuzzball

    2018/09/10 17:03

    なんで合ってる方を修正するかな‥。a[:3] が [0, 1, 2, 3] になりますか?

    キャンセル

回答 2

+4

必要ならこんなクラスを作ってみたり。

from collections import OrderedDict

class SliceableDict(OrderedDict):
    def __getitem__(self, arg):
        if not isinstance(arg, slice):
            return super().__getitem__(arg)

        return SliceableDict(
            list(self.items())[arg]
        )

    def __str__(self):
        return ', '.join(
            '{}: {}'.format(key, self[key]) for key in self
        )


dct = SliceableDict()
dct['a'] = ['apple', 'assimulation']
dct['b'] = ['banana', 'bag']
dct['c'] = ['cap', 'cat']

print(dct)
print(dct[:2])

実行結果 Wandbox

a: ['apple', 'assimulation'], b: ['banana', 'bag'], c: ['cap', 'cat']
a: ['apple', 'assimulation'], b: ['banana', 'bag']

どちらにせよ『辞書らしい』使い方にはならないので、
本当にスライスに依る切り出しが必要かどうかはしっかり検討した方が良いです。

コメントを受けて

以下のように一般的な辞書宣言はできないという理解で正しいでしょうか。

 Python3.6なら

キーワード引数の挿入順序なら既に保証されているので、次のように書けます。

dct = SliceableDict(
    a=['apple', 'assimulation'],
    b=['banana', 'bag'],
    c=['cap', 'cat']
)

 Python3.7、あるいはCPython実装の3.6では

辞書表現の評価順序が以前のバージョンから保証されているほか、
組み込み辞書の挿入順序が新たに保証されているので、次のように書けます。

dct = SliceableDict({
    'a': ['apple', 'assimulation'],
    'b': ['banana', 'bag'],
    'c': ['cap', 'cat']
})

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/09/10 15:59

    ご回答いただきましてありがとうございます。
    ご提示いただいた例では以下のように一般的な辞書宣言はできないという理解で正しいでしょうか。
    dct = { 'a' : ['apple', 'assimulation'], 'b': ['banana', 'bag'], 'c':['cap', 'cat']}

    キャンセル

  • 2018/09/10 16:01

    Python3.7なら、言語仕様で順序が保証されます。
    3.6でも、CPythonなら順序が保証されます。

    キャンセル

  • 2018/09/10 16:41

    ありがとうございました。

    キャンセル

checkベストアンサー

+2

そもそも辞書には「要素の順番」はないのが原則です。

>>> d = {"abcdef"[x]:x for x in range(6)}
>>> d
{'a': 0, 'c': 2, 'b': 1, 'e': 4, 'd': 3, 'f': 5}

特定のキーだけ残した新たな辞書を作りたい、という要求であれば、たとえば以下のように辞書内包表記を用いて行えます。

>>> d = { 'a' : ['apple', 'assimulation'], 'b': ['banana', 'bag'], 'c':['cap', 'cat']}
>>> {k:v for k,v in d.items() if k in {"a", "b"}}
{'a': ['apple', 'assimulation'], 'b': ['banana', 'bag']}

dictという変数名はdictクラスを上書きしてしまうのでやめましょう。

>>> dict
<class 'dict'>
>>> dict = { 'a' : ['apple', 'assimulation'], 'b': ['banana', 'bag'], 'c':['cap', 'cat']}
>>> dict
{'a': ['apple', 'assimulation'], 'b': ['banana', 'bag'], 'c': ['cap', 'cat']}

 追記

CPython3.6(最も一般的に用いられているpythonの実装)では処理系の仕様、Python3.7以降では言語仕様により辞書のキー順序が保持されます。

よって次のようなコードが成立するはずです。

>>> d = { 'a' : ['apple', 'assimulation'], 'b': ['banana', 'bag'], 'c':['cap', 'cat']}
>>> {k:d[k] for k in list(d.keys())[:2]}
{'a': ['apple', 'assimulation'], 'b': ['banana', 'bag']}

古めのpythonとコード互換性がないことに留意の上これを使う、というのは一つの選択肢です。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/09/10 15:43

    ご回答いただきましてありがとうございます。
    キー値が全てわかればいいのですが、キー値を全て参照できないため、番号で0~99番のようにスライシングしたかったのですが、辞書型のデータセットでは無理でしょうか。

    キャンセル

  • 2018/09/10 15:51

    基本的に無理です。
    OrderedDictを使うというのが一つの手でしょうか。
    https://docs.python.jp/3/library/collections.html#ordereddict-objects

    >>> from collections import OrderedDict
    >>> od1 = OrderedDict([('a',['apple', 'assimulation']), ('b',['banana', 'bag']), ('c',['cap', 'cat'])])
    >>> od1
    OrderedDict([('a', ['apple', 'assimulation']), ('b', ['banana', 'bag']), ('c', ['cap', 'cat'])])
    >>> od2 = OrderedDict(list(od.items())[:2])
    >>> od2
    OrderedDict([('a', ['apple', 'assimulation']), ('b', ['banana', 'bag'])])

    ちょっと冗長だし、効率も悪いですが、やればできます。とはいえ、ここまで面倒なことをするなら、他の方法を考えた方が良いかもしれません。
    あと、辞書からOrderedDictに変換するというヘマをしないように注意が必要(辞書にした時点で順番の情報は失われます・・・)。

    キャンセル

  • 2018/09/10 15:54

    1個目の例の実行結果はPython 3.5環境ですか?

    キャンセル

  • 2018/09/10 15:55

    python3.5環境です。3.6以降で順序が保持されるようになっていることは承知の上

    キャンセル

  • 2018/09/10 15:56

    あれ、じゃあこれで良いんですかね?
    >>> d = {"abcdef"[x]:x for x in range(6)}
    >>> {k:d[k] for k in list(d.keys())[:3]}
    {'a': 0, 'b': 1, 'c': 2}

    キャンセル

  • 2018/09/10 16:16

    なるほどなるほど、ありがとうございます。

    キャンセル

同じタグがついた質問を見る

  • Python

    12220questions

    Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

  • Python 3.x

    10228questions

    Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。