كيف تكتب هذا المقتطف clojure في روبي و/أو هاسكل؟
-
26-09-2019 - |
سؤال
كنت أعمل على قالب القضبان وكنت أحاول كتابة القليل من التعليمات البرمجية التي تتيح لي ملء جدول أو أعمدة متعددة من علامات UL "من أعلى إلى أسفل" و "يسار إلى يمين" عبر العديد من الأعمدة أحددها . أنا فقط أحصل على تعليق من روبي حتى لا أستطيع معرفة ذلك. أنا فضولي أيضًا بشأن إصدار Haskell الاصطلاحي لهذا المقتطف المفيد. التحسينات على نسخة clojure موضع تقدير:
(defn table [xs & {:keys [cols direction]
:or {cols 1 direction 'right}}]
(into []
(condp = direction
'down (let [c (count xs)
q (int (/ c cols))
n (if (> (mod c q) 0) (inc q) q)]
(apply map vector (partition n n (repeat nil) xs)))
'right (map vec (partition cols cols (repeat nil) xs)))))
مع هذا الجزء من الكود يمكنني بعد ذلك القيام بما يلي:
(table (range 10) :cols 3)
طباعة هذا سيبدو هكذا:
0 1 2
3 4 5
6 7 8
9
وأصعب واحد:
(table (range 10) :cols 3 :direction 'down)
يبدو هكذا:
0 4 8
1 5 9
2 6
3 7
المحلول
من المحتمل أن أكتب شيئًا كهذا في هاسكل ، باستخدام Data.List.Split
حزمة من Hackage:
import Data.List (intercalate, transpose)
import Data.List.Split (splitEvery)
data Direction = Horizontal | Vertical deriving (Eq, Read, Show)
table :: Direction -> Int -> [a] -> [[a]]
table Horizontal cols xs = splitEvery cols xs
table Vertical cols xs = let (q,r) = length xs `divMod` cols
q' = if r == 0 then q else q+1
in transpose $ table Horizontal q' xs
showTable :: Show a => [[a]] -> String
showTable = intercalate "\n" . map (intercalate "\t" . map show)
main :: IO ()
main = mapM_ putStrLn [ showTable $ table Horizontal 3 [0..9]
, "---"
, showTable $ table Vertical 3 [0..9] ]
بعض هذا ، مثل Direction
اكتب و transpose
خدعة ، تم اشتقاقها من إجابة Jkramer. لن أستخدم وسيطات الكلمات الرئيسية لشيء مثل هذا في Haskell (ليس لديها مثل هذه الأشياء حقًا ، ولكن يمكنك محاكاةها باستخدام السجلات كما في إجابة إدوارد كميت) ، لكنني أضع هذه الوسائط أولاً لأنها أكثر فائدة مع التطبيق الجزئي ((defaultTable = table Horizontal 1
). ال splitEvery
وظيفة فقط قطع قائمة في قوائم الحجم المناسب ؛ يجب أن تكون بقية الكود واضحة. ال table
وظيفة إرجاع قائمة القوائم ؛ للحصول على سلسلة ، showTable
تدرج الدالة علامات التبويب والخطوط الجديدة. (ال intercalate
الوظيفة تسلس قائمة القوائم ، وفصلها مع القائمة المحددة. إنه مماثل لـ Perl/Python/Ruby's join
, ، فقط للقوائم بدلا من مجرد سلاسل.)
نصائح أخرى
لا يمكنني قراءة رمز Clojure (لم أستخدم اللغة أبدًا) ، ولكن بناءً على الأمثلة ، إليك كيف سأفعل ذلك في Ruby.
def table array, cols, direction
if direction==:down
if array.size%cols != 0
array[(array.size/cols+1)*cols-1]=nil
#putting nil in the last space in the array
#also fills all of the spaces before it
end
newarray=array.each_slice(array.size/cols).to_a
table newarray.transpose.flatten(1), cols, :across
elsif direction==:across
array.each_slice(cols) do |row|
puts row.join(" ")
end
else
raise ArgumentError
end
end
هذا شيء اخترقته بسرعة في هاسكل. أنا متأكد من أنه عربات التي تجرها الدواب ويمكن تحسينها ، ولكن هذا شيء لتبدأ بـ:
import System.IO
import Data.List
data Direction = Horizontal | Vertical
main = do
putStrLn $ table [1..9] 3 Horizontal
putStrLn "---"
putStrLn $ table [1..9] 3 Vertical
table xs ncol direction =
case direction of
Horizontal -> format (rows strings ncol)
Vertical -> format (columns strings ncol)
where
format = intercalate "\n" . map (intercalate " ")
strings = map show xs
rows xs ncol =
if length xs > ncol
then take ncol xs : rows (drop ncol xs) ncol
else [xs]
columns xs = transpose . rows xs
انتاج:
1 2 3
4 5 6
7 8 9
---
1 4 7
2 5 8
3 6 9
حل روبي الخاص بي
def table(values)
elements = values[:elements]
cols = values[:cols]
rows = (elements.count / cols.to_f).ceil
erg = []
rows.times do |i|
cols.times do |j|
erg << elements[values[:direction] == 'down' ? i+(rows*j) : j+i*(rows-1)]
if erg.length == cols
yield erg
erg = []
end
end
end
yield erg
end
الاستخدام والإخراج:
table(:elements => [0,1,2,3,4,5,6,7,8,9], :cols => 3) do |h,i,j|
puts h.to_s << " " << i.to_s << " " << j.to_s
end
puts "---"
table(:elements => [0,1,2,3,4,5,6,7,8,9], :cols => 3, :direction => "down") do |h,i,j|
puts h.to_s << " " << i.to_s << " " << j.to_s
end
0 1 2
3 4 5
6 7 8
9
---
0 4 8
1 5 9
2 6
3 7
يعطي التقطيع والقطعة حل روبي مباشر:
def table(range, cols, direction=:right)
if direction == :right
range.each_slice cols
else
columns = range.each_slice((range.to_a.length - 1) / cols + 1).to_a
columns[0].zip *columns[1..-1]
end
end
puts table(0..9, 3, :down).map { |line| line.join ' ' }
import Data.Array
stride :: Int -> Int -> Int
stride count cols = ceiling (fromIntegral count / fromIntegral cols)
type Direction = Int -> Int -> Int -> Int -> Int
right :: Direction
right count cols x y = y * cols + x
down :: Direction
down count cols x y = x * stride count cols + y
data Options = Options { cols :: Int, direction :: Direction }
options :: Options
options = Options 1 right
table :: Options -> [a] -> Array (Int,Int) (Maybe a)
table (Options cols dir) xs
= listArray newRange (map f (range newRange))
where count = length xs
rows = stride count cols
newRange = ((0,0),(rows-1,cols-1))
f (y, x)
| ix < count = Just (xs !! ix)
| otherwise = Nothing
where ix = dir count cols x y
هذا يمنحنا تقريبًا تصفيريًا إلى حد ما لاستعلامك الأصلي مع الوسائط الاختيارية:
*Main> table options { cols = 3 } [1..10]
listArray ((0,0),(3,2)) [Just 1, Just 2, Just 3
,Just 4, Just 5, Just 6
,Just 7, Just 8, Just 9
,Just 10,Nothing,Nothing]
*Main> table options { direction = down, cols = 3 } [1..10]
listArray ((0,0),(3,2)) [Just 1,Just 5,Just 9
,Just 2,Just 6,Just 10
,Just 3,Just 7,Nothing
,Just 4,Just 8,Nothing]
تركت النتائج المتوسطة في شكل صفيف كما أوضحت أنك خططت لتنسيقها كجدول أو علامات UL.