Do roztworu surowego SQL, Utworzyłem szorstką replikację problemu na ustawieniach ideone here
danych:
create table content_a(id int, user_id int, content varchar(20));
create table content_b(id int, user_id int, content varchar(20));
create table content_c(id int, user_id int, content varchar(20));
create table voting(user_id int, content_id int, content_type_id int, vote int);
create table users(id int, name varchar(20));
insert into content_a values(1,1,'aaaa');
insert into content_a values(2,1,'bbbb');
insert into content_a values(3,1,'cccc');
insert into content_b values(1,2,'dddd');
insert into content_b values(2,2,'eeee');
insert into content_b values(3,2,'ffff');
insert into content_c values(1,1,'gggg');
insert into content_c values(2,2,'hhhh');
insert into content_c values(3,3,'iiii');
insert into users values(1, 'first');
insert into users values(2, 'second');
insert into users values(3, 'third');
insert into users values(4, 'voteonly');
-- user 1 net votes (2)
insert into voting values (1, 1, 1, 1);
insert into voting values (2, 3, 1, -1);
insert into voting values (3, 1, 1, 1);
insert into voting values (4, 2, 1, 1);
-- user 2 net votes (3)
insert into voting values (1, 2, 2, 1);
insert into voting values (1, 1, 2, 1);
insert into voting values (2, 3, 2, -1);
insert into voting values (4, 2, 2, 1);
insert into voting values (4, 2, 3, 1);
-- user 3 net votes (-1)
insert into voting values (2, 3, 3, -1);
ja w zasadzie przyjąć, że content_a ma typ 1, content_b ma typ 2 i content_c ma typ 3. Używanie surowego SQL, nie wydaje się być t wo oczywiste podejścia. Pierwszym z nich jest połączenie całej zawartości, a następnie dołączenie jej do użytkowników i tabel głosowania. Przetestowałem to podejście poniżej.
select users.*, sum(voting.vote)
from users,
voting, (
SELECT id, 1 AS content_type_id, user_id
FROM content_a
UNION
SELECT id, 2 AS content_type_id, user_id
FROM content_b
UNION
SELECT id, 3 AS content_type_id, user_id
FROM content_c) contents
where contents.user_id = users.id
and voting.content_id = contents.id
and voting.content_type_id = contents.content_type_id
group by users.id
order by sum(voting.vote) desc;
Alternatywą wydaje się być zewnętrzne dołączenie tabel treści do tabel głosowania, bez kroku związkowego. To może być bardziej wydajne, ale nie byłem w stanie go przetestować, ponieważ studio graficzne ciągle przepisuje mój sql dla mnie ... Spodziewam się, że SQL będzie wyglądał mniej więcej tak (ale nie przetestowałem tego):
select users.*, sum(voting.vote)
from users, voting, content_a, content_b, content_c
where users.id = content_a.user_id (+)
and users.id = content_b.user_id (+)
and users.id = content_c.user_id (+)
and ((content_a.id = voting.content_id and voting.content_type_id = 1) OR
(content_b.id = voting.content_id and voting.content_type_id = 2) OR
(content_c.id = voting.content_id and voting.content_type_id = 3))
group by users.id
order by sum(voting.vote) desc;
Biorąc pod uwagę głosowanie w tabeli "Głosowanie", w jaki sposób można określić, do której tabeli treści się odnosi? Co się stanie, jeśli "content_id" istnieje w więcej niż jednej tabeli? – eggyal
Przepraszam, zapomniałem dodać kolumnę. – mburke13