Rails: предотвратить деление на ноль с помощью postgres - PullRequest
1 голос
/ 01 марта 2011

У меня есть приложение rails с базой данных postgre, и я пытаюсь получить некоторые показатели со следующим утверждением

Negotiation.find(:all, :conditions => conditions_hash('negotiation'), :select => 'sum(payment_term * amount_after_annualised_base) / sum(amount_after_annualised_base) as pt').first

как я могу убедиться, что сумма (amount_after_annualised_base) не равна 0 (таким образом, не выдает ошибку «деление на ноль»)?

Я нашел это, но не уверен, что это правильный / лучший / самый эффективный способ.

Negotiation.find(:all, :conditions => conditions_hash('negotiation'), :select => 'sum(payment_term * amount_after_annualised_base) / (CASE sum(amount_after_annualised_base) WHEN 0 THEN NULL ELSE sum(amount_after_annualised_base) END) as pt').first

спасибо! Pierre

Ответы [ 3 ]

4 голосов
/ 01 марта 2011

используйте nullif :

Negotiation.find(:all, :conditions => conditions_hash('negotiation'), :select => 'sum(payment_term * amount_after_annualised_base) / nullif(sum(amount_after_annualised_base),0) as pt').first

, что является кратким сокращением для вашего решения:

Negotiation.find(:all, :conditions => conditions_hash('negotiation'), :select => 'sum(payment_term * amount_after_annualised_base) / (CASE sum(amount_after_annualised_base) WHEN 0 THEN NULL ELSE sum(amount_after_annualised_base) END) as pt').first
3 голосов
/ 01 марта 2011

Как насчет:

Negotiation.find(:all, 
                 :conditions => conditions_hash('negotiation'), 
                 :select => 'sum(amount_after_annualised_base) sum_amount,
                             sum(payment_term * amount_after_annualised_base) / greatest(1,sum_amount) as pt').first

и проверьте свой код, если sum_amount равно нулю.

2 голосов
/ 01 марта 2011

Я думаю, вам нужно написать что-то вроде

Negotiation.find(
  :all, :conditions => conditions_hash('negotiation'), 
  :select => 'CASE sum(amount_after_annualised_base) WHEN 0 THEN NULL ELSE sum(payment_term * amount_after_annualised_base) / sum(amount_after_annualised_base) END as pt').first

, который возвращает NULL, если sum(amount_after_annualised_base) равен нулю, а в противном случае выполняется запрошенное деление.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...