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

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

新規登録して質問してみよう
ただいま回答率
85.48%
Python

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

Q&A

解決済

3回答

1940閲覧

Python 内包表記

O.K.

総合スコア30

Python

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

0グッド

0クリップ

投稿2017/11/17 02:31

jupyter notebookでとある数列を求めるプログラムを作りました。
for文の中にあるfor文(for j in range(compare.size + 1)...)を内包表記で簡潔に書きたいのですが、
breakの入れ方がわかりません。教えてください。

Python

1%matplotlib inline 2import math 3import matplotlib.pyplot as plt 4import numpy as np 5 6pd.options.display.max_columns = None 7pd.options.display.notebook_repr_html = True 8 9grundy_ceil = np.array([0]) 10grundy_ceil_1p = np.array([0]) 11 12for n in range(1, 101): 13 compare = np.array([]) 14 15 compare = np.append(compare, [grundy_ceil[n - i] for i in range(1, math.ceil(n /2 + 1))]) 16 17 for j in range(compare.size + 1): 18 if not j in compare: 19 grundy_ceil = np.append(grundy_ceil, j) 20 break 21 22df = pd.DataFrame(grundy_ceil).T

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答3

0

可能です。リスト内包表記だけをつかって同じように動作するコードが以下です。

python

1import math 2import numpy as np 3import pandas as pd 4 5b = np.array([0]) 6for n in range(1, 101): 7 a = np.array([]) 8 a = b[:-(math.ceil(n/2)+1):-1] 9 b = np.append(b, [x for x in range(len(a)+1) if x not in a]) 10df = pd.DataFrame(b).T

ただし、上記のリスト内包表記は元のコードよりも実行速度が10~20%ほど遅いです。

速度重視ならmkgreiさんのコードと上記を組み合わせた以下のコードのほうが10~20%ほど速いようです。

Python

1import math 2import numpy as np 3import pandas as pd 4from itertools import count 5 6b = [0] 7for n in range(1, 101): 8 a = b[:-(math.ceil(n/2)+1):-1] 9 b.append(next(i for i in count() if i not in a)) 10df = pd.DataFrame(b).T

101ぐらいでは実感できないですが1001を超えてくると差がはっきりとわかります。

投稿2017/11/17 06:10

yag1kaz

総合スコア253

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

mkgrei

2017/11/17 06:17

next()を使うと見やすくなるだけでした。 速度は実はあまり変わりません。 cProfileを使って調べると比較がボトルネックだったので… がんばったのに、悲しい… 早くするのならaをset()にすることみたいですね。 そうすればnot inがO(1)になるので。 ご指摘の通りデータサイズも重要ですね。 101個程度なら何をやっても気にならないです。
O.K.

2017/11/17 09:35

回答ありがとうございます。 a = b[:-(math.ceil(n/2)+1):-1] のところはどういう文法になっているのでしょうか?
yag1kaz

2017/11/17 10:03 編集

a = b[:-(math.ceil(n/2)+1):-1] で処理している内容は 『配列bの要素を、後ろから数えてmath.ceil(n/2)個までの要素を、逆順に取り出してaに代入せよ』 ということになります。 後ろから数えてX個目までを取り出す処理は、 list[:-X+1:-1] と書くことで実現できますのでそれを利用しています。 蛇足) 括弧を余分につけて list[:-(X+1):-1] と書いてもおなじですが、こう書くとサルが2匹寝転んで縦につながっているみたいでちょっと面白いですねw
mkgrei

2017/11/17 10:08

その視点は遊び心があっていいですね。←蛇足について
O.K.

2017/11/17 10:19

なるほどわかりました。 ありがとうございます。
guest

0

今回の目的を果たすためであれば、条件を満たす最初のものを取り出すことができるはずです。
こんなやつ。
ここを参考にしています。

python

1from itertools import count 2a = [grundy_ceil[n - i] for i in range(1, math.ceil(n /2 + 1))] 3j = next(i for i in count() if i not in a)

コードの速さが気になるのであれば、numpy.arrayへのキャストなくすべきでしょう。
また検索があるので、listやnp.arrayではなくsetを使うとだいぶ早くなります。

python

1import math 2import pandas as pd 3from itertools import count 4grundy_ceil = [0] 5 6for n in range(1, 101): 7 a = set([grundy_ceil[n - i] for i in range(1, math.ceil(n /2 + 1))]) 8 grundy_ceil.append(next(i for i in count() if i not in a)) 9 10df = pd.DataFrame(grundy_ceil).T

追記:完全に趣味ですが。ここを参考に。
yag1kazさんのlistからの取り出し方も取り入れました。
sliceをリスト内包表記で書くと遅くなるのは盲点でした…

python

1from timeit import timeit 2from itertools import count 3import math 4import pandas as pd 5import numpy as np 6 7pd.options.display.max_columns = None 8pd.options.display.notebook_repr_html = True 9 10def f0(): 11 grundy_ceil = np.array([0]) 12 #grundy_ceil_1p = np.array([0]) 13 for n in range(1, 101): 14 compare = np.array([]) 15 compare = np.append(compare, [grundy_ceil[n - i] for i in range(1, math.ceil(n /2 + 1))]) 16 for j in range(compare.size + 1): 17 if not j in compare: 18 grundy_ceil = np.append(grundy_ceil, j) 19 break 20 df = pd.DataFrame(grundy_ceil).T 21 return df 22 23def f1(): 24 grundy_ceil = [0] 25 for n in range(1, 101): 26 a = [grundy_ceil[n - i] for i in range(1, math.ceil(n /2 + 1))] 27 j = next(i for i in count() if i not in a) 28 grundy_ceil.append(j) 29 df = pd.DataFrame(grundy_ceil).T 30 return df 31 32def f2(): 33 grundy_ceil = [0] 34 for n in range(1, 101): 35 a = np.array([grundy_ceil[n - i] for i in range(1, math.ceil(n /2 + 1))]) 36 j = next(i for i in count() if i not in a) 37 grundy_ceil.append(j) 38 df = pd.DataFrame(grundy_ceil).T 39 return df 40 41def f3(): 42 grundy_ceil = [0] 43 for n in range(1, 101): 44 a = set([grundy_ceil[n - i] for i in range(1, math.ceil(n /2 + 1))]) 45 j = next(i for i in count() if i not in a) 46 grundy_ceil.append(j) 47 df = pd.DataFrame(grundy_ceil).T 48 return df 49 50def f4(): 51 grundy_ceil = [0] 52 for n in range(1, 101): 53 a = set(grundy_ceil[:-math.ceil(n/2+1):-1]) 54 j = next(i for i in count() if i not in a) 55 grundy_ceil.append(j) 56 df = pd.DataFrame(grundy_ceil).T 57 return df 58 59a = f0() 60b = f1() 61c = f2() 62d = f3() 63e = f4() 64print('f1', np.allclose(a[0].values, b[0].values)) 65print('f2', np.allclose(a[0].values, c[0].values)) 66print('f3', np.allclose(a[0].values, d[0].values)) 67print('f4', np.allclose(a[0].values, e[0].values)) 68''' 69一応同じ結果になることを確認 70f1 True 71f2 True 72f3 True 73f4 True 74''' 75n = 1000 76print('Original {}'.format(timeit(f0, number=n))) 77print('list {}'.format(timeit(f1, number=n))) 78print('np.array {}'.format(timeit(f2, number=n))) 79print('set {}'.format(timeit(f3, number=n))) 80print('slice+set {}'.format(timeit(f4, number=n))) 81''' 82Original 9.65767323700129 83list 1.1692692330107093 84np.array 7.38920716199209 85set 0.9272018460032996 86slice+set 0.6420557640085462 87'''

投稿2017/11/17 03:55

編集2017/11/17 06:35
mkgrei

総合スコア8560

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

O.K.

2017/11/17 09:36

回答ありがとうございます。 setを使うというのは盲点でした。
guest

0

ベストアンサー

Python内包表記はbreak(ループ中断)をサポートしません。(for文では何か問題があるのでしょうか?)

一応、禁じ手的なハック も存在するようですが、全くおすすめしません。

投稿2017/11/17 03:35

yohhoy

総合スコア6191

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

O.K.

2017/11/17 03:37

そうでしたか。ありがとうございます。 できるだけ短く書きたかったので。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問