Как вернуть часть массива в 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:]

Поскольку Ruby все делает в массивах, мне интересно, есть ли что-то подобное.

Кристиан Штаде-Шульдт
источник

Ответы:

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]?
dertoni 01
3
@dertoni: stackoverflow.com/questions/3219229/…
michelpm 07
25
a[2..-1]чтобы перейти от 3-го элемента к последнему. a[2...-1]чтобы перейти от 3-го элемента ко второму последнему элементу.
Умный программист,
1
@Rafeh приветствует, интересно, как работает эта хлам, -1 сделал свое дело
Бен Синклер
Синтаксис @CleverProgrammer намного ближе к Python, чем принятый ответ. Я люблю Ruby, но должен сказать, что синтаксис Python короче и понятнее. В качестве бонуса можно указать шаг:range(10)[:5:-1]
Эрик Думинил
27

Если вы хотите разделить / вырезать массив по индексу i,

arr = arr.drop(i)

> arr = [1,2,3,4,5]
 => [1, 2, 3, 4, 5] 
> arr.drop(2)
 => [3, 4, 5] 
Ленин Радж Раджасекаран
источник
17

Для этого вы можете использовать slice () :

>> 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).

Manuel
источник
Вы хотите использовать массив столбцов во втором срезе?
Samuel
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]
user3449311
источник
3

Ruby 2.6 Beginless / Endless Ranges

(..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"]
PGill
источник
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для последовательного размещения среднего элемента в первой половине, если вам это нужно.
Сораши