سؤال

أتساءل عما إذا كانت هناك طريقة للقيام بما يمكنني القيام به أدناه باستخدام بايثون، في روبي:

sum = reduce(lambda x, y: x + y, map(lambda x, y: x * y, weights, data))

لدي صفيفتان متساويتان في الحجم مع الأوزان والبيانات ولكن لا يمكنني العثور على وظيفة مشابهة للخريطة في روبي، لذا قلل من العمل.

هل كانت مفيدة؟

المحلول

@ ميشيل دي ماري

يمكن اختصار مثال Ruby 1.9 الخاص بك قليلاً:

weights.zip(data).map(:*).reduce(:+)

لاحظ أيضًا أنه في Ruby 1.8، إذا كنت بحاجة إلى ActiveSupport (من Rails)، يمكنك استخدام:

weights.zip(data).map(&:*).reduce(&:+)

نصائح أخرى

في روبي 1.9:

weights.zip(data).map{|a,b| a*b}.reduce(:+)

في روبي 1.8:

weights.zip(data).inject(0) {|sum,(w,d)| sum + w*d }

تقوم الدالة Array.zip بدمج المصفوفات بشكل عنصري.إنها ليست نظيفة تمامًا مثل بناء جملة بايثون، ولكن إليك طريقة واحدة يمكنك استخدامها:

weights = [1, 2, 3]
data = [4, 5, 6]
result = Array.new
a.zip(b) { |x, y| result << x * y } # For just the one operation

sum = 0
a.zip(b) { |x, y| sum += x * y } # For both operations

روبي لديه map الطريقة (ويعرف أيضا باسمال collect الطريقة)، والتي يمكن تطبيقها على أي Enumerable هدف.لو numbers عبارة عن مجموعة من الأرقام، السطر التالي في روبي:

numbers.map{|x| x + 5}

يعادل السطر التالي في بايثون:

map(lambda x: x + 5, numbers)

لمزيد من التفاصيل، انظر هنا أو هنا.

بديل للخريطة يعمل لأكثر من مصفوفتين أيضًا:

def dot(*arrays)
  arrays.transpose.map {|vals| yield vals}
end

dot(weights,data) {|a,b| a*b} 

# OR, if you have a third array

dot(weights,data,offsets) {|a,b,c| (a*b)+c}

يمكن أيضًا إضافة هذا إلى Array:

class Array
  def dot
    self.transpose.map{|vals| yield vals}
  end
end

[weights,data].dot {|a,b| a*b}

#OR

[weights,data,offsets].dot {|a,b,c| (a*b)+c}
weights = [1,2,3]
data    = [10,50,30]

require 'matrix'
Vector[*weights].inner_product Vector[*data] # => 200 
مرخصة بموجب: CC-BY-SA مع الإسناد
لا تنتمي إلى StackOverflow
scroll top