Apabila menggunakan fungsi kumpulan untuk menapis set hasil, beberapa masalah dihadapi dan penyelesaian [Disyorkan : tutorial video mysql]
1. Senario aplikasi
Terdapat dua jadual
Jadual artikel (satu-ke-banyak mesej Jadual) t_posts:
oid, posts_name
Jadual mesej (jadual artikel banyak-ke-satu) t_comment:
oid, posts_id, msg_content, create_time
2. Analisis keperluan
Soal kandungan jawapan terkini setiap artikel
3.SQL writing
select tp.oid, tp.posts_name, tc.msg_content, tc.create_time from t_posts tp left join t_comment tc on tp.oid = tc.posts_id group by tp.oid having create_time = max(create_time)
Andaikan terdapat dua artikel A dan B (urutan rekod balasan dalam pangkalan data adalah konsisten dengan yang berikut)
<p>A有一个回复记录时间为: 2019-09-10 <br>A有一个回复记录时间为: 2019-09-11 <br>B有一个回复记录时间为: 2019-09-01 <br>B有一个回复记录时间为: 2019-09-09<br></p>
Jalankan sql di atas, anda akan mendapati bahawa sejumlah besar rekod hilang dalam set hasil , dan hasilnya salah Selepas menanyakan data, kami mendapati bahawa
pemilik mysql dilaksanakan selepas kumpulan mengikut, iaitu, mengumpulkan dahulu dan kemudian menapis. , tetapi kerana terdapat lebih daripada dua rekod mesej,
Jadi hasil yang ditetapkan selepas pengumpulan hanya akan mengambil mesej pertama setiap mesej sebagai maklumat rekod selepas pengumpulan Pada masa ini, jika anda menggunakan mempunyai create_time = max(create_time)
, maka max(create_time) ialah nilai maksimum kumpulan semasa Masa
ialah: 2019-09-09
, jadi sql di atas akan kehilangan set hasil
4 Ubah suai SQL
Kerana kita tahu set hasil pendua yang digabungkan selepas pengumpulan adalah yang mempunyai rownum terkecil. , bolehkah kita mengubah suai SQL seperti berikut??
select tp.oid, tp.posts_name, tc.msg_content, tc.create_time from t_posts tp left join t_comment tc on tp.oid = tc.posts_id group by tp.oid having create_time = max(create_time) -- 下面的是新增的sql order by tc.create_time desc
Selepas menjalankannya, saya dapati ia masih tidak bagus, jadi, buktikan bahawa susunan mengikut adalah selepas kumpulan oleh & mempunyai
Kemudian saya memikirkannya, bolehkah saya menggunakan pesanan dengan terus untuk mengoptimumkan hasil terkumpul tanpa perlu?
mempunyai masa_cipta = maks(masa_buat)
select tp.oid, tp.posts_name, tc.msg_content, tc.create_time from t_posts tp left join t_comment tc on tp.oid = tc.posts_id group by tp.oid order by tc.create_time desc
Ralat set hasil tidak menjejaskan keputusan kumpulan Set hasil pendua masih digabungkan mengikut kumpulan minimum rownum, dan kemudian diisih
5 >
Oleh kerana tertib mengikut hanya boleh menjejaskan kumpulan mengikut, adakah mungkin untuk mengisih hasil yang ditetapkan sebelum kumpulan mengikut, dan kemudian mengumpulkannya?select * from ( select tp.oid, tp.posts_name, tc.msg_content, tc.create_time from t_posts tp left join t_comment tc on tp.oid = tc.posts_id order by tc.create_time desc ) t group by t.oidSaya mendapati ia masih tidak berfungsi, tetapi subkueri itu memang diisih dahuluSelepas bertanya (terangkan), saya mendapati bahawa susunan mengikut subkueri telah dioptimumkan Penyelesaian:
- Had penggunaan 99999 dalam. subquery
- Gunakan di mana keadaan dalam subquery, create_time = (pilih maks(create_time) daripada kumpulan t_comment mengikut oid)
select * from ( select tp.oid, tp.posts_name, tc.msg_content, tc.create_time from t_posts tp left join t_comment tc on tp.oid = tc.posts_id order by tc.create_time desc limit 9999 ) t group by t.oid
Selesai
Mata pengetahuan tambahan:Perbezaan antara versi mysql5.5 dan mysql 5.7: Dalam versi 5.7, jika had tidak digunakan, kumpulan mengikut akan mengoptimumkan susunan mengikut