Форум программистов, компьютерный форум, киберфорум
PHP: базы данных
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
-28 / 0 / 0
Регистрация: 26.03.2025
Сообщений: 168

Вытащить по 1 записи от пользователя, сортировка по времени

11.04.2025, 17:13. Показов 1094. Ответов 6
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Есть таблица пользователей, есть таблица новостей.

Как вытащить 10 записей, по одной на 1 пользователя?

Допустим, есть 5 пользователей, каждый добавил по 10 новостей.

Как добиться того, чтобы первые 5 записи - последние новости от каждого пользователя, далее с 5-10, так же по одной записи от одного пользователя.
Но вытащить предпоследнюю запись от каждого пользователя.
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
11.04.2025, 17:13
Ответы с готовыми решениями:

Как вытащить последнюю запись пользователя из базы по его id с датой
У меня в базе поле id, поле сообщения, и дата сообщения как вытащить последнюю запись по дате...

Вытащить временные метки из текста
00:00 США — страна мигрантов 00:01:17 Новый поток российских мигрантов 00:03:45 Как пересечь...

Почему при записи даты и времени из формы в бд, к времени прибавляется час?
Здравствуйте, при записи даты и времени из формы в бд, к времени прибавляется час, почему так...

6
133 / 57 / 2
Регистрация: 23.11.2024
Сообщений: 686
11.04.2025, 17:39
SQL
1
2
3
4
5
6
7
8
9
SELECT n.user_id, n.created_at, n.message
FROM news n
JOIN (
    SELECT user_id, MAX(created_at) AS max_created_at
    FROM news
    GROUP BY user_id
) AS latest_news ON n.user_id = latest_news.user_id AND n.created_at = latest_news.max_created_at
ORDER BY n.created_at DESC
FETCH FIRST 10 ROWS ONLY;  -- Ограничиваем результат до 10 записей
Если пользователь может запостить несколько сообщений с одинаковым временем, то (он сам себе злобный буратино)
надо будет использовать "Общие табличные выражения"
Common Table Expressions (CTE) были введены в стандарт SQL в SQL:1999.

SQL
1
2
3
4
5
6
7
8
9
10
11
12
WITH RankedNews AS (
    SELECT 
        n.*, 
        ROW_NUMBER() OVER (PARTITION BY n.user_id ORDER BY n.created_at DESC) AS rn
    FROM 
        news n
)
SELECT * 
FROM RankedNews
WHERE rn IN (1, 2)  -- Получаем последние и предпоследние новости от каждого пользователя
ORDER BY user_id, rn
FETCH FIRST 10 ROWS ONLY;  -- Ограничиваем результат до 10 записей
Если предпоследние новости не надо, то
WHERE rn IN (1)
1
-28 / 0 / 0
Регистрация: 26.03.2025
Сообщений: 168
11.04.2025, 21:19  [ТС]
Не совсем то.

Выполняем запрос, берем последние 10 записей.

На выходе должно быть 10 записей, 1 пользователь - 1 запись его, последняя.

Затем берем последующие 10 записей.

Все тоже самое, 1 запись - пользователь.

Пример:

Code
1
2
3
4
5
6
7
id   title         user   date
1    news 1      1      6
2    news 2      3      5
3    news 3      1      4
4    news 4      11     3
5    news 5      77     2
6    news 6      12     1
Необходимо вытащить 3 последние записи

Code
1
2
3
4
id   title         user   date
1    news 1      1      6
2    news 2      3      5
4    news 4      11     3
2 записи с 1/3 от пользователя 1, игнорируем вторую запись пользователя, поскольку уже есть, берем новость следующего пользователя.

Делаем второй запрос limit 3,6, с 3 записи по 6, на выходе необходимо получить

Code
1
2
3
4
id   title         user   date
3    news 3      1      4
4    news 4      11     3
5    news 5      77     2
Берем вторую запись 1 пользователя, поскольку опубликвана позже всех. Если бы у пользователя была только одна запись, сдвинули просто при втором запросе.
0
133 / 57 / 2
Регистрация: 23.11.2024
Сообщений: 686
11.04.2025, 21:40
Если сделать постраничный просмотр, то вместо
FETCH FIRST 10 ROWS ONLY;
пишешь
OFFSET 0 ROWS FETCH NEXT 10 ROWS ONLY;
OFFSET 10 ROWS FETCH NEXT 10 ROWS ONLY;
OFFSET 20 ROWS FETCH NEXT 10 ROWS ONLY;
и т.д.

Я не понял, что ты хочешь увидеть внутри запроса.

SQL
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
WITH RankedMessages AS (
    SELECT 
        user_id,
        created_at,
        message,
        ROW_NUMBER() OVER (PARTITION BY user_id ORDER BY created_at) AS rn
    FROM 
        messages
)
SELECT 
    user_id,
    created_at,
    message
FROM 
    RankedMessages
ORDER BY 
    rn, user_id;
Здесь мы нумеруем сообщения каждого пользователя - первое самое свежее, 2 - следующее, 3 - за ним и т.д.
Затем показываем первые сообщения разных пользователей, затем вторые и т.д.
Может так оказаться, что в десяти сообщениях один пользователь повторяется, если сообщений от разных пользователей мало, меньше десяти.
1
-28 / 0 / 0
Регистрация: 26.03.2025
Сообщений: 168
11.04.2025, 22:59  [ТС]
Работает, почти у цели. Это уже лучше чем было у меня.
Code
1
2
3
4
5
6
7
title         user   date
news 1      1      6
news 2      3      5
news 3      1      4
news 4      11     3
news 5      77     2
news 6      12     1
Тут сортируем по дате, от большего к меньшему.

Допустим последние 3 записи от одного пользователя.

Необходимо на выходе получить так, чтобы не было такого, что 2 и более новости подряд от одного пользователя.

То есть, сортируем записи как обычно, от большего времени к меньшему. Но исключаем подряд записи от одного пользователя

SQL
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
WITH RankedMessages AS (
    SELECT 
    n.id,
        n.id_u,
        n.time,
        n.title,
        ROW_NUMBER() OVER (PARTITION BY n.id_u ORDER BY n.time DESC) AS rn
    FROM 
        news n
)
SELECT 
        id,
        id_u,
        TIME,
        title
FROM 
    RankedMessages
ORDER BY 
    rn, TIME DESC LIMIT {$start},".$limit
Работает, на половину. Что на выходе получаю.

Берем последнюю запись каждого пользователя, ок. Постранично работает, трудность в том, что пока не покажет, по последней записи каждого пользователя. Не пойдет на второй круг...

Допустим самая ранняя запись была добавлена 2 года назад. Только потом будим доставать 2 запись каждого пользователя.

Пытаюсь выполнить так, чтобы сортировка была по времени публикации, от большего к меньшему. Но в тоже время исключить повторения нескольких публикаций подряд от одного пользователя и сделать так, чтобы избежать таких случаев. Когда необходимо просмотреть по одной записи каждого пользователя за все время, после чего начнет выбирать свежее публикации но уже вторые.

Не знаю, как описать что необходимо. Постарался, возможно ли такого добиться.

Добавлено через 6 минут
Сортируем как обычно от большего к меньшему времени.
Исключаем повторения подряд записи от одного пользователя.

Возможно тут стоит как то указать интервал, допустим в 1 неделю или что то подобное. Чтобы вторая запись пользователя появилась раньше чем пока не вытащим все и только тогда будем брать 2 запись каждого пользователя.
0
133 / 57 / 2
Регистрация: 23.11.2024
Сообщений: 686
11.04.2025, 23:04
Попробуй в конце досортировать, не
ORDER BY rn, user_id;
а
ORDER BY rn, created_at, user_id;

на выходе получить так, чтобы не было такого, что 2 и более новости подряд от одного пользователя.
У тебя требования противоречивые. С них начни.
0
-28 / 0 / 0
Регистрация: 26.03.2025
Сообщений: 168
11.04.2025, 23:20  [ТС]
Code
1
2
3
4
   FROM 
        news n
   WHERE AND 
        n.time >= ".strtotime('today -1 month')."
Понимаю, что противоречивые.

Добавил ограничение, не ранее чем месяц назад выбирать. Более менее, но тогда ограничиваем себя во времени...

Думаю дай добавлю через
Code
1
UNION ALL
запрос но уже
Code
1
n.time <= ".strtotime('today -1 month')."
Сперва делаем круги по записям которые не страше месяца, затем круги которые страше. Что то не вышло, не понятно как тут это выполднить.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
11.04.2025, 23:20
Помогаю со студенческими работами здесь

Вытащить из бд с сортировкой
В бд есть столбцы &quot;х&quot; и &quot;у&quot; Как мне вывести, что бы последовательность строк была такой: Вывести...

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

Как вытащить из БД (MySql) логин авторизованного пользователя?
вот код входа. всем привет у меня такой вопрос вот я сделал форму регистрации и входа все...

Как вытащить данные из записей в гостевой книге и вставить их в форму?
Существует гостевая книга. Решил добавить для неё возможность редактирования сообщений. Логика у...

Как вытащить с помощью php запроса ID последней добавленной записи
Это все конечно хорошо, но если за те милисекунды которые пройдут между INSERT ... и SELECT...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
7
Ответ Создать тему
Новые блоги и статьи
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Access
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
Создание Single Page Application на фреймах
krapotkin 16.11.2025
Статья исключительно для начинающих. Подходы оригинальностью не блещут. В век Веб все очень привыкли к дизайну Single-Page-Application . Быстренько разберем подход "на фреймах". Мы делаем одну. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru