Does Haskell have List Slices (i.e. Python)?

后端 未结 12 995
南方客
南方客 2020-12-05 02:15

Does Haskell have similar syntactic sugar to Python List Slices?

For instance in Python:

x = [\'a\',\'b\',\'c\',\'d\']
x[1:3] 

give

相关标签:
12条回答
  • 2020-12-05 02:27

    Python slices also support step:

    >>> range(10)[::2]
    [0, 2, 4, 6, 8]
    >>> range(10)[2:8:2]
    [2, 4, 6]
    

    So inspired by Dan Burton's dropping every Nth element I implemented a slice with step. It works on infinite lists!

    takeStep :: Int -> [a] -> [a]
    takeStep _ [] = []
    takeStep n (x:xs) = x : takeStep n (drop (n-1) xs)
    
    slice :: Int -> Int -> Int -> [a] -> [a]
    slice start stop step = takeStep step . take (stop - start) . drop start
    

    However, Python also supports negative start and stop (it counts from end of list) and negative step (it reverses the list, stop becomes start and vice versa, and steps thru the list).

    from pprint import pprint # enter all of this into Python interpreter
    pprint([range(10)[ 2: 6],     # [2, 3, 4, 5]
            range(10)[ 6: 2:-1],  # [6, 5, 4, 3]
            range(10)[ 6: 2:-2],  # [6, 4]      
            range(10)[-8: 6],     # [2, 3, 4, 5]
            range(10)[ 2:-4],     # [2, 3, 4, 5]
            range(10)[-8:-4],     # [2, 3, 4, 5]
            range(10)[ 6:-8:-1],  # [6, 5, 4, 3]
            range(10)[-4: 2:-1],  # [6, 5, 4, 3]
            range(10)[-4:-8:-1]]) # [6, 5, 4, 3]]
    

    How do I implement that in Haskell? I need to reverse the list if the step is negative, start counting start and stop from the end of the list if these are negative, and keep in mind that the resulting list should contain elements with indexes start <= k < stop (with positive step) or start >= k > stop (with negative step).

    takeStep :: Int -> [a] -> [a]
    takeStep _ [] = []
    takeStep n (x:xs)
      | n >= 0 = x : takeStep n (drop (n-1) xs)
      | otherwise = takeStep (-n) (reverse xs)
    
    slice :: Int -> Int -> Int -> [a] -> [a]
    slice a e d xs = z . y . x $ xs -- a:start, e:stop, d:step
      where a' = if a >= 0 then a else (length xs + a)
            e' = if e >= 0 then e else (length xs + e)
            x = if d >= 0 then drop a' else drop e'
            y = if d >= 0 then take (e'-a') else take (a'-e'+1)
            z = takeStep d
    
    test :: IO () -- slice works exactly in both languages
    test = forM_ t (putStrLn . show)
      where xs = [0..9]
            t = [slice   2   6   1  xs, -- [2, 3, 4, 5]
                 slice   6   2 (-1) xs, -- [6, 5, 4, 3]
                 slice   6   2 (-2) xs, -- [6, 4]
                 slice (-8)  6   1  xs, -- [2, 3, 4, 5]
                 slice   2 (-4)  1  xs, -- [2, 3, 4, 5]
                 slice (-8)(-4)  1  xs, -- [2, 3, 4, 5]
                 slice   6 (-8)(-1) xs, -- [6, 5, 4, 3]
                 slice (-4)  2 (-1) xs, -- [6, 5, 4, 3]
                 slice (-4)(-8)(-1) xs] -- [6, 5, 4, 3]
    

    The algorithm still works with infinite lists given positive arguments, but with negative step it returns an empty list (theoretically, it still could return a reversed sublist) and with negative start or stop it enters an infinite loop. So be careful with negative arguments.

    0 讨论(0)
  • 2020-12-05 02:27

    Another way to do this is with the function splitAt from Data.List -- I find it makes it a little easier to read and understand than using take and drop -- but that's just personal preference:

    import Data.List
    slice :: Int -> Int -> [a] -> [a]
    slice start stop xs = fst $ splitAt (stop - start) (snd $ splitAt start xs)
    

    For example:

    Prelude Data.List> slice 0 2 [1, 2, 3, 4, 5, 6]
    [1,2]
    Prelude Data.List> slice 0 0 [1, 2, 3, 4, 5, 6]
    []
    Prelude Data.List> slice 5 2 [1, 2, 3, 4, 5, 6]
    []
    Prelude Data.List> slice 1 4 [1, 2, 3, 4, 5, 6]
    [2,3,4]
    Prelude Data.List> slice 5 7 [1, 2, 3, 4, 5, 6]
    [6]
    Prelude Data.List> slice 6 10 [1, 2, 3, 4, 5, 6]
    []
    

    This should be equivalent to

    let slice' start stop xs = take (stop - start) $ drop start xs
    

    which will certainly be more efficient, but which I find a little more confusing than thinking about the indices where the list is split into front and back halves.

    0 讨论(0)
  • 2020-12-05 02:28
    sublist start length = take length . snd . splitAt start
    
    slice start end = snd .splitAt start . take end
    
    0 讨论(0)
  • 2020-12-05 02:29

    I don't think one is included, but you could write one fairly simply:

    slice start end = take (end - start + 1) . drop start
    

    Of course, with the precondition that start and end are in-bounds, and end >= start.

    0 讨论(0)
  • 2020-12-05 02:30

    If you are trying to match Python "lists" (which isn't a list, as others note) then you might want to use the Haskell vector package which does have a built in slice. Also, Vector can be evaluated in parallel, which I think is really cool.

    0 讨论(0)
  • 2020-12-05 02:30

    I've wrote this code that works for negative numbers as well, like Python's list slicing, except for reversing lists, which I find unrelated to list slicing:

    slice :: Int -> Int -> [a] -> [a]
    
    slice 0 x arr
      | x < 0 = slice 0 ((length arr)+(x)) arr
      | x == (length arr) = arr
      | otherwise = slice 0 (x) (init arr)
    
    slice x y arr
      | x < 0 = slice ((length arr)+x) y arr 
      | y < 0 = slice x ((length arr)+y) arr
      | otherwise = slice (x-1) (y-1) (tail arr)
    
    main = do
      print(slice (-3) (-1) [3, 4, 29, 4, 6]) -- [29,4]
      print(slice (2) (-1) [35, 345, 23, 24, 69, 2, 34, 523]) -- [23,24,69,32,34]
      print(slice 2 5 [34, 5, 5, 3, 43, 4, 23] ) -- [5,3,43]
    
    
    
    0 讨论(0)
提交回复
热议问题