Подтвердить что ты не робот

Вычитание одной строки данных из другого в SQL

Я столкнулся с некоторым SQL, где у меня есть несколько строк данных, и я хочу вычесть строку из предыдущей строки и повторить ее до конца.

Итак, вот таблица:

CREATE TABLE foo (
  id,
  length
)
INSERT INTO foo (id,length) VALUES(1,1090)
INSERT INTO foo (id,length) VALUES(2,888)
INSERT INTO foo (id,length) VALUES(3,545)
INSERT INTO foo (id,length) VALUES(4,434)
INSERT INTO foo (id,length) VALUES(5,45)

Я хочу, чтобы результаты отображали третий столбец, называемый разницей, который является одной строкой, вычитаемой из следующей ниже, с окончательной вычитанием строки с нуля.

+------+------------------------+
| id   |length |  difference  |
+------+------------------------+
|    1 | 1090  |  202         |
|    2 |  888  |  343         |
|    3 |  545  |  111         |
|    4 |  434  |  389         |
|    5 |   45  |   45         |

Я пробовал присоединиться к себе, но я не совсем уверен, как ограничить результаты, а не прокручивать сам. Я не могу зависеть от того, что значение id будет последовательным для заданного набора результатов, поэтому я не использую это значение. Я мог бы расширить схему, чтобы включить какое-то последовательное значение.

Вот что я пробовал:

SELECT id, f.length, f2.length, (f.length - f2.length) AS difference
FROM foo f, foo f2

Спасибо за помощь.

4b9b3361

Ответ 1

Это может помочь вам (несколько).


select a.id, a.length, 
coalesce(a.length - 
    (select b.length from foo b where b.id = a.id + 1), a.length) as diff
from foo a

Ответ 2

Yipee!!! это делает трюк:

SELECT  f.id, f.length, 
    (f.length - ISNULL(f2.length,0)) AS diff
FROM foo f
LEFT OUTER JOIN foo f2
ON  f2.id = (f.id +1)

Пожалуйста, проверьте также и другие случаи, он работает для значений, которые вы опубликовали! Обратите внимание, что это для SQL Server 2005

Ответ 3

Значит, они просто упорядочены по наименьшему?

SELECT f.id, f.length, (f.length - ISNULL(t.length, 0)) AS difference
FROM foo AS f
LEFT JOIN (
    SELECT f1.id
        ,MAX(f2.length) as length
    FROM foo AS f1
    INNER JOIN foo AS f2
        ON f1.length > f2.length
    GROUP BY f1.id
) AS t -- this is the triangle
    ON t.id = f.id

Вы можете использовать COALESCE (или IFNULL) вместо ISNULL для MySQL.

Ответ 4

Что-то вроде этого:

SELECT T2.ID, T2.[Length], T2.[Length]-T1.[Length] AS 'Difference'
FROM Foo AS T1 RIGHT OUTER JOIN Foo AS T2 ON ( T1.ID = (T2.ID-1) )
ORDER BY T1.ID

Ответ 5

edit: исправлено при повторном чтении Q (неправильно понято)

SELECT f.id, 
       f2.id, 
       f.length, 
       f2.length, 
       (f.length -f2.length) AS difference
FROM foo f, 
     foo f2 
where f2.id = f.id+1

id был неоднозначным

edit: note: проверено в mysql 5.0

Ответ 6

Select f1.id, f1.seqnum, f2.seqnum, f1.length, f2.length, f1.length-f2.length 

From (

Select Id, length, row_number(order by length) 'seqnum'
From
foo

) f1

Inner join (

Select 
Id, length, row_number(order by length) 'seqnum' from foo union select 0, 0, 0

) f2 

On f1.seqnum = f2.seqnum + 1

Order by f1.length desc

Ответ 7

У меня была эта проблема, и было интересно посмотреть на ваши решения. Мне странно, что такая проблема нормальной жизни настолько сложна в SQL. Поскольку мне нужны только значения в отчете, я выбрал совершенно другое решение. Я запускаю Ruby on Rails в качестве передней части моей базы данных sqlite3 и просто сделал вычитание в виде следующим образом:

В вашем рубиновом контроллере есть объектная переменная @foo, которая содержит строки, возвращаемые вашим запросом.

В представлении просто сделайте

<table border=1>
  <tr>
    <th>id</th>
    <th>length</th>
    <th>difference</th>
  </tr>

<% for i in [email protected] do %>
  <tr>
    <td><%=h @foo[i].id %></td>
    <td><%=h @foo[i].length %></td>
    <td><% if ([email protected]) then %>
        <%=  @foo[i].length %>
      <% else %>
        <%= @foo[i+1].length.to_i - @foo[i].length.to_i %>
      <% end %>
    </td>
  </tr>
<% end %>
</table>

Кажется, он более надежный, чем SQL-решения.