내가 가지고 있다고 가정 해보십시오.
test = numpy.array([[1, 2], [3, 4], [5, 6]])
test[i]
배열의 i 번째 줄을 얻습니다 (예 🙂[1, 2]
. i 번째 열에 어떻게 액세스 할 수 있습니까? (예 🙂 [1, 3, 5]
. 또한 이것은 비싼 작업입니까?
답변
>>> test[:,0]
array([1, 3, 5])
비슷하게,
>>> test[1,:]
array([3, 4])
행에 액세스 할 수 있습니다. 이것은 NumPy 참조 의 1.4 (인덱싱) 섹션에서 다룹니다 . 적어도 내 경험상 이것은 빠르다. 루프에서 각 요소에 액세스하는 것보다 훨씬 빠릅니다.
답변
한 번에 둘 이상의 열에 액세스하려면 다음을 수행하십시오.
>>> test = np.arange(9).reshape((3,3))
>>> test
array([[0, 1, 2],
[3, 4, 5],
[6, 7, 8]])
>>> test[:,[0,2]]
array([[0, 2],
[3, 5],
[6, 8]])
답변
>>> test[:,0]
array([1, 3, 5])
이 명령은 행 벡터를 제공합니다. 루프 벡터를 반복하고 싶다면 괜찮습니다. 그러나 차원이 3xN 인 다른 배열로 hstack하고 싶다면
ValueError: all the input arrays must have same number of dimensions
동안
>>> test[:,[0]]
array([[1],
[3],
[5]])
연결 또는 hstack 작업을 수행 할 수 있도록 열 벡터를 제공합니다.
예 :
>>> np.hstack((test, test[:,[0]]))
array([[1, 2, 1],
[3, 4, 3],
[5, 6, 5]])
답변
행을 바꾸고 반환 할 수도 있습니다.
In [4]: test.T[0]
Out[4]: array([1, 3, 5])
답변
여러 개의 열을 얻으려면 다음을 수행하십시오.
> test[:,[0,2]]
당신은 열 0과 2를 얻을 것이다
답변
질문에 대한 답변이 있지만 뉘앙스에 대해 언급하겠습니다.
배열의 첫 번째 열에 관심이 있다고 가정 해 봅시다.
arr = numpy.array([[1, 2],
[3, 4],
[5, 6]])
다른 답변에서 이미 알고 있듯이 “행 벡터”(array of shape (3,)
) 형식으로 얻으려면 슬라이싱을 사용하십시오.
arr_c1_ref = arr[:, 1] # creates a reference to the 1st column of the arr
arr_c1_copy = arr[:, 1].copy() # creates a copy of the 1st column of the arr
배열이 다른 배열의 뷰인지 또는 사본인지 확인하려면 다음을 수행하십시오.
arr_c1_ref.base is arr # True
arr_c1_copy.base is arr # False
ndarray.base를 참조하십시오 .
둘 사이의 명백한 차이 (수정 arr_c1_ref
은에 영향을 미침 arr
) 외에도 각각을 통과하는 바이트 단계 수는 다릅니다.
arr_c1_ref.strides[0] # 8 bytes
arr_c1_copy.strides[0] # 4 bytes
보폭을 참조하십시오 . 이것이 왜 중요한가? : A
대신에 매우 큰 배열이 있다고 상상해보십시오 arr
.
A = np.random.randint(2, size=(10000,10000), dtype='int32')
A_c1_ref = A[:, 1]
A_c1_copy = A[:, 1].copy()
첫 번째 열의 모든 요소, 즉 A_c1_ref.sum()
또는 의 합을 계산하려고합니다 A_c1_copy.sum()
. 복사 된 버전을 사용하는 것이 훨씬 빠릅니다.
%timeit A_c1_ref.sum() # ~248 µs
%timeit A_c1_copy.sum() # ~12.8 µs
이것은 앞에서 언급 한 보폭이 다르기 때문입니다.
A_c1_ref.strides[0] # 40000 bytes
A_c1_copy.strides[0] # 4 bytes
열 복사본을 사용하는 것이 더 나은 것처럼 보이지만 복사본을 만드는 데 시간이 걸리고 더 많은 메모리를 사용하는 이유는 항상 사실이 아닙니다 (이 경우을 만드는 데 약 200µs가 걸렸습니다 A_c1_copy
). 그러나 처음에 사본이 필요하거나 배열의 특정 열에 대해 여러 가지 다른 작업을 수행해야하고 속도를 위해 메모리를 희생하면 사본을 만드는 것이 좋습니다.
주로 열을 다루는 데 관심이있는 경우 행 주요 ( ‘C’) 순서 대신 열-주요 ( ‘F’) 순서로 배열을 만드는 것이 좋습니다. )을 누른 다음 열을 복사하지 않고 이전과 같이 슬라이싱을 수행하십시오.
A = np.asfortranarray(A) # or np.array(A, order='F')
A_c1_ref = A[:, 1]
A_c1_ref.strides[0] # 4 bytes
%timeit A_c1_ref.sum() # ~12.6 µs vs ~248 µs
이제 열보기에서 합계 작업 (또는 다른 작업)을 수행하는 것이 훨씬 빠릅니다.
마지막으로 배열을 바꾸고 행 슬라이싱을 사용하는 것은 원래 배열의 모양과 보폭을 바꾸는 것만으로 변환이 이루어 지므로 원래 배열의 열 슬라이싱을 사용하는 것과 같습니다.
A.T[1,:].strides[0] # 40000
답변
>>> test
array([[0, 1, 2, 3, 4],
[5, 6, 7, 8, 9]])
>>> ncol = test.shape[1]
>>> ncol
5L
그런 다음이 방법으로 2-4 번째 열을 선택할 수 있습니다.
>>> test[0:, 1:(ncol - 1)]
array([[1, 2, 3],
[6, 7, 8]])