Как рассчитать процент с помощью оператора SQL - PullRequest
144 голосов
/ 21 апреля 2009

У меня есть таблица SQL Server, которая содержит пользователей и их оценки. Для простоты, давайте просто скажем, что есть 2 столбца - name & grade. Таким образом, типичный ряд будет Имя: «Джон Доу», класс: «А».

Я ищу один оператор SQL, который найдет проценты всех возможных ответов. (A, B, C и т. Д.) Кроме того, есть ли способ сделать это без определения всех возможных ответов (открытое текстовое поле - пользователи могут ввести «пройти / не пройти», «нет» и т. Д.) 1005 *

Конечный результат, который я ищу: A: 5%, B: 15%, C: 40% и т. Д. *

Ответы [ 11 ]

196 голосов
/ 21 апреля 2009

Я проверил следующее, и это работает. Ответ от gordyii был близок, но умножил 100 в неправильном месте и пропустил несколько скобок.

Select Grade, (Count(Grade)* 100 / (Select Count(*) From MyTable)) as Score
From MyTable
Group By Grade
178 голосов
/ 01 мая 2011
  1. Наиболее эффективный (с использованием over ()).

    select Grade, count(*) * 100.0 / sum(count(*)) over()
    from MyTable
    group by Grade
    
  2. Универсальный (любая версия SQL).

    select Rate, count(*) * 100.0 / (select count(*) from MyTable)
    from MyTable
    group by Rate;
    
  3. С CTE, наименее эффективным.

    with t(Rate, RateCount) 
    as 
    ( 
        select Rate, count(*) 
        from MyTable
        group by Rate
    )
    select Rate, RateCount * 100.0/(select sum(RateCount) from t)
    from t;
    
35 голосов
/ 01 мая 2009

Вместо того, чтобы использовать отдельный CTE для получения суммы, вы можете использовать оконную функцию без предложения "partition by".

Если вы используете:

count(*)

, чтобы получить счет для группы, вы можете использовать:

sum(count(*)) over ()

чтобы получить общее количество.

Например:

select Grade, 100. * count(*) / sum(count(*)) over ()
from table
group by Grade;

В моем опыте это работает быстрее, но я думаю, что в некоторых случаях он может использовать временную таблицу (я видел «Рабочий стол» при работе с «Установить статистику io on»).

EDIT: Я не уверен, что мой пример запроса - то, что вы ищете, я просто иллюстрировал, как работают оконные функции.

9 голосов
/ 21 апреля 2009

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

    Select Grade, CountofGrade / sum(CountofGrade) *100 
    from
    (
    Select Grade, Count(*) as CountofGrade
    From Grades
    Group By Grade) as sub
    Group by Grade

Вы должны указать систему, которую используете.

9 голосов
/ 21 апреля 2009

Вы должны рассчитать общее количество оценок Если это SQL 2005, вы можете использовать CTE

    WITH Tot(Total) (
    SELECT COUNT(*) FROM table
    )
    SELECT Grade, COUNT(*) / Total * 100
--, CONVERT(VARCHAR, COUNT(*) / Total * 100) + '%'  -- With percentage sign
--, CONVERT(VARCHAR, ROUND(COUNT(*) / Total * 100, -2)) + '%'  -- With Round
    FROM table
    GROUP BY Grade
7 голосов
/ 24 мая 2016

Я просто использую это, когда когда-либо мне нужно отработать процент ..

ROUND(CAST((Numerator * 100.0 / Denominator) AS FLOAT), 2) AS Percentage

Обратите внимание, что 100.0 возвращает десятичные дроби, а само по себе 100 округляет результат до ближайшего целого числа, даже с функцией ROUND ()!

5 голосов
/ 21 апреля 2009

Я считаю, что это общее решение, хотя я тестировал его с использованием IBM Informix Dynamic Server 11.50.FC3. Следующий запрос:

SELECT grade,
       ROUND(100.0 * grade_sum / (SELECT COUNT(*) FROM grades), 2) AS pct_of_grades
    FROM (SELECT grade, COUNT(*) AS grade_sum
            FROM grades
            GROUP BY grade
         )
    ORDER BY grade;

дает следующий вывод на тестовые данные, показанные ниже горизонтального правила. Функция ROUND может быть специфичной для СУБД, а остальное (вероятно) - нет. (Обратите внимание, что я изменил 100 на 100,0, чтобы гарантировать, что вычисление происходит с использованием нецелого числа - DECIMAL, NUMERIC - арифметика; см. Комментарии и спасибо Thunder.)

grade  pct_of_grades
CHAR(1) DECIMAL(32,2)
A       32.26
B       16.13
C       12.90
D       12.90
E       9.68
F       16.13

CREATE TABLE grades
(
    id VARCHAR(10) NOT NULL,
    grade CHAR(1) NOT NULL CHECK (grade MATCHES '[ABCDEF]')
);

INSERT INTO grades VALUES('1001', 'A');
INSERT INTO grades VALUES('1002', 'B');
INSERT INTO grades VALUES('1003', 'F');
INSERT INTO grades VALUES('1004', 'C');
INSERT INTO grades VALUES('1005', 'D');
INSERT INTO grades VALUES('1006', 'A');
INSERT INTO grades VALUES('1007', 'F');
INSERT INTO grades VALUES('1008', 'C');
INSERT INTO grades VALUES('1009', 'A');
INSERT INTO grades VALUES('1010', 'E');
INSERT INTO grades VALUES('1001', 'A');
INSERT INTO grades VALUES('1012', 'F');
INSERT INTO grades VALUES('1013', 'D');
INSERT INTO grades VALUES('1014', 'B');
INSERT INTO grades VALUES('1015', 'E');
INSERT INTO grades VALUES('1016', 'A');
INSERT INTO grades VALUES('1017', 'F');
INSERT INTO grades VALUES('1018', 'B');
INSERT INTO grades VALUES('1019', 'C');
INSERT INTO grades VALUES('1020', 'A');
INSERT INTO grades VALUES('1021', 'A');
INSERT INTO grades VALUES('1022', 'E');
INSERT INTO grades VALUES('1023', 'D');
INSERT INTO grades VALUES('1024', 'B');
INSERT INTO grades VALUES('1025', 'A');
INSERT INTO grades VALUES('1026', 'A');
INSERT INTO grades VALUES('1027', 'D');
INSERT INTO grades VALUES('1028', 'B');
INSERT INTO grades VALUES('1029', 'A');
INSERT INTO grades VALUES('1030', 'C');
INSERT INTO grades VALUES('1031', 'F');
5 голосов
/ 21 апреля 2009

Следующее должно работать

ID - Key
Grade - A,B,C,D...

EDIT: переместил * 100 и добавил 1.0, чтобы убедиться, что он не выполняет целочисленное деление

Select 
   Grade, Count(ID) * 100.0 / ((Select Count(ID) From MyTable) * 1.0)
From MyTable
Group By Grade
4 голосов
/ 05 сентября 2012
SELECT Grade, GradeCount / SUM(GradeCount)
FROM (SELECT Grade, COUNT(*) As GradeCount
      FROM myTable
      GROUP BY Grade) Grades
3 голосов
/ 21 апреля 2009

Вы можете использовать подвыбор в запросе от (не проверено и не уверен, что быстрее):

SELECT Grade, COUNT(*) / TotalRows
FROM (SELECT Grade, COUNT(*) As TotalRows
      FROM myTable) Grades
GROUP BY Grade, TotalRows

Или

SELECT Grade, SUM(PartialCount)
FROM (SELECT Grade, 1/COUNT(*) AS PartialCount
      FROM myTable) Grades
GROUP BY Grade

Или

SELECT Grade, GradeCount / SUM(GradeCount)
FROM (SELECT Grade, COUNT(*) As GradeCount
      FROM myTable
      GROUP BY Grade) Grades

Вы также можете использовать хранимую процедуру (извинения за синтаксис Firebird):

SELECT COUNT(*)
FROM myTable
INTO :TotalCount;

FOR SELECT Grade, COUNT(*)
FROM myTable
GROUP BY Grade
INTO :Grade, :GradeCount
DO
BEGIN
    Percent = :GradeCount / :TotalCount;
    SUSPEND;
END
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...