Як повернути частину масиву в Ruby?


125

Маючи список у Python, я можу повернути його частину за допомогою наступного коду:

foo = [1,2,3,4,5,6]
bar = [10,20,30,40,50,60]
half = len(foo) / 2
foobar = foo[:half] + bar[half:]

Оскільки Рубі все робить у масивах, мені цікаво, чи є щось подібне до цього.

Відповіді:


192

Так, у Ruby дуже схожий синтаксис нарізки масивів як Python. Ось riдокументація для методу індексу масиву:

--------------------------------------------------------------- Array#[]
     array[index]                -> obj      or nil
     array[start, length]        -> an_array or nil
     array[range]                -> an_array or nil
     array.slice(index)          -> obj      or nil
     array.slice(start, length)  -> an_array or nil
     array.slice(range)          -> an_array or nil
------------------------------------------------------------------------
     Element Reference---Returns the element at index, or returns a 
     subarray starting at start and continuing for length elements, or 
     returns a subarray specified by range. Negative indices count 
     backward from the end of the array (-1 is the last element). 
     Returns nil if the index (or starting index) are out of range.

        a = [ "a", "b", "c", "d", "e" ]
        a[2] +  a[0] + a[1]    #=> "cab"
        a[6]                   #=> nil
        a[1, 2]                #=> [ "b", "c" ]
        a[1..3]                #=> [ "b", "c", "d" ]
        a[4..7]                #=> [ "e" ]
        a[6..10]               #=> nil
        a[-3, 3]               #=> [ "c", "d", "e" ]
        # special cases
        a[5]                   #=> nil
        a[6, 1]                #=> nil
        a[5, 1]                #=> []
        a[5..10]               #=> []

5
чому [5, 1] ​​відрізняється від [6, 1]?
дертоні


25
a[2..-1]дістатися від 3-го елемента до останнього. a[2...-1]щоб дістатися від 3-го елемента до другого останнього елемента.
Розумний програміст

1
@Rafeh ура, цікавилось, як працює цей мотлох, -1 зробив трюк
Бен Сінклер

Синтаксис @CleverProgrammer набагато ближче до Python, ніж прийнята відповідь. Я люблю Рубі, але мушу сказати, що синтаксис Python коротший і чіткіший. Як бонус можна вказати крок:range(10)[:5:-1]
Ерік Дюмініл


17

Ви можете використовувати для цього фрагмент () :

>> foo = [1,2,3,4,5,6]
=> [1, 2, 3, 4, 5, 6]
>> bar = [10,20,30,40,50,60]
=> [10, 20, 30, 40, 50, 60]
>> half = foo.length / 2
=> 3
>> foobar = foo.slice(0, half) + bar.slice(half, foo.length)
=> [1, 2, 3, 40, 50, 60]

До речі, наскільки мені відомо, «списки» Python - це просто ефективно реалізовані масиви, що динамічно зростають. Вставлення на початку знаходиться в O (n), вставлення в кінці амортизується O (1), випадковий доступ - O (1).


Ти маєш на увазі використовувати масив смуг у другому фрагменті?
Самуїл

FYI: slice!()не змінює масив на місці, скоріше, він "Видаляє елемент (и), заданий індексом (необов'язково до елементів довжини) або діапазоном." per ruby-doc.org/core-2.2.3/Array.html#method-i-slice-21
Джошуа Пінтер

7

інший спосіб - використовувати метод діапазону

foo = [1,2,3,4,5,6]
bar = [10,20,30,40,50,60]
a = foo[0...3]
b = bar[3...6]

print a + b 
=> [1, 2, 3, 40, 50 , 60]

3

Ruby 2.6 Безмежні / нескінченні діапазони

(..1)
# or
(...1)

(1..)
# or
(1...)

[1,2,3,4,5,6][..3]
=> [1, 2, 3, 4]

[1,2,3,4,5,6][...3]
 => [1, 2, 3]

ROLES = %w[superadmin manager admin contact user]
ROLES[ROLES.index('admin')..]
=> ["admin", "contact", "user"]

2

Мені подобаються діапазони для цього:

def first_half(list)
  list[0...(list.length / 2)]
end

def last_half(list)
  list[(list.length / 2)..list.length]
end

Однак будьте дуже уважні до того, включена кінцева точка у ваш діапазон. Це стає критичним у списку непарної довжини, де вам потрібно вибрати, де ви збираєтесь пробити середину. Інакше ви закінчите подвійний підрахунок середнього елемента.

Наведений вище приклад послідовно ставить середній елемент в останню половину.


Ви можете використовувати, (list.length / 2.0).ceilщоб послідовно розміщувати середній елемент у першій половині, якщо це потрібно.
Сораші
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.