Некорректное закрытие QSqlDatabase, при использовании QSqlTableModel |
Здравствуйте, гость ( Вход | Регистрация )
Некорректное закрытие QSqlDatabase, при использовании QSqlTableModel |
Гость_silver_* |
4.8.2008, 18:27
Сообщение
#1
|
Гости |
Здравствуйте!
Использую в простеньком приложении QSqlTableModel для отображения и редактирования таблицы. При закрытии программы выдаёт предупреждение
Пробовал в деструкторе окна, где используется модель, делать вещи вроде
не помогает Оно, вроде бы, и нестрашно, но как-то неправильно. Кто-нибудь сталкивался с таким? |
|
|
Litkevich Yuriy |
4.8.2008, 18:39
Сообщение
#2
|
разработчик РЭА Группа: Сомодератор Сообщений: 9669 Регистрация: 9.1.2008 Из: Тюмень Пользователь №: 64 Спасибо сказали: 807 раз(а) Репутация: 94 |
|
|
|
silver |
4.8.2008, 21:45
Сообщение
#3
|
Новичок Группа: Новичок Сообщений: 8 Регистрация: 4.8.2008 Из: Харьков Пользователь №: 250 Спасибо сказали: 1 раз(а) Репутация: 0 |
Всё, разобрался!
Дело было в том, что я использовал не только QSqlTableModel, но и фильтрующую QSortFilterProxyModel. Так вот последняя при удалении не освобождает модель-источник! Что в общем-то и логично. Всё решается либо явным удалением модели-источника, либо указанием для неё в качестве родителя прокси-модели:
Причём именно в таком порядке, т.к. удаление модели-источника раньше прокси-модели недопустимо! Пишу так много, потому что полдня потратил на такую глупость, может кому пригодится... Сообщение отредактировал silver - 4.8.2008, 22:26 |
|
|
Litkevich Yuriy |
4.8.2008, 22:18
Сообщение
#4
|
разработчик РЭА Группа: Сомодератор Сообщений: 9669 Регистрация: 9.1.2008 Из: Тюмень Пользователь №: 64 Спасибо сказали: 807 раз(а) Репутация: 94 |
вообще это известное предупреждение и видно его только в консоли.
Удалять соединение нет нужды - вот от правная точка! |
|
|
silver |
4.8.2008, 22:30
Сообщение
#5
|
Новичок Группа: Новичок Сообщений: 8 Регистрация: 4.8.2008 Из: Харьков Пользователь №: 250 Спасибо сказали: 1 раз(а) Репутация: 0 |
|
|
|
ViGOur |
5.8.2008, 11:52
Сообщение
#6
|
Мастер Группа: Модератор Сообщений: 3296 Регистрация: 9.10.2007 Из: Москва Пользователь №: 4 Спасибо сказали: 231 раз(а) Репутация: 40 |
|
|
|
acen83 |
6.8.2008, 8:28
Сообщение
#7
|
Студент Группа: Участник Сообщений: 23 Регистрация: 11.2.2008 Пользователь №: 91 Спасибо сказали: 0 раз(а) Репутация: 0 |
QSqlDataBase* bd=QSqlDataBase::addConnection("QMYSQL");
// настройка базы, логина и т.п. bd->open(); // запросы bd->close(); затем в другом месте программы повторяя этот код появляется предупреждение QSqlDatabasePrivate::removeDatabase: connection 'qt_sql_default_connection' is still in use, all queries will cease to work. Что делать? Сообщение отредактировал acen83 - 6.8.2008, 8:30 |
|
|
ViGOur |
6.8.2008, 9:07
Сообщение
#8
|
Мастер Группа: Модератор Сообщений: 3296 Регистрация: 9.10.2007 Из: Москва Пользователь №: 4 Спасибо сказали: 231 раз(а) Репутация: 40 |
Читаем описание QSqlDataBase::addConnection:
Цитата Adds a database to the list of database connections using the driver type and the connection name connectionName. If there already exists a database connection called connectionName, that connection is removed. По идее QSqlDataBase* bd=QSqlDataBase::addConnection("QMYSQL"); нужно вызывать один раз и bd желательно сделать глобальным или синглтоном.Если же ты хочешь создать две копии QSqlDataBase, то используй второй параметр addConnection для их именования... |
|
|
Litkevich Yuriy |
6.8.2008, 12:58
Сообщение
#9
|
разработчик РЭА Группа: Сомодератор Сообщений: 9669 Регистрация: 9.1.2008 Из: Тюмень Пользователь №: 64 Спасибо сказали: 807 раз(а) Репутация: 94 |
Что делать? если тебе надо использовать уже созданое тобой соединение то используй так: QSqlDatabase app_db = QSqlDatabase::database(); Перевод из SVN-хранилища: Цитата Подробное описание Класс QSqlDatabase предоставляет подключение к базе данных. Класс QSqlDatabase предоставляет абстрактный интерфейс для доступа к базе данных. Он использует конкретный QSqlDriver базы данных для доступа и манипуляции данными. В следующем коде показано как установить соединение:
После создания объекта QSqlDatabase, устанавливаем параметры соединения с помощью setDatabaseName(), setUserName(), setPassword(), setHostName(), setPort(), и setConnectOptions(). Только после установки параметров вызывается open() для открытия соединения. Соединение определеное выше — безымянное соединение. Это соединение по умолчанию и доступ к нему может быть получен позже используя database():
Чтобы сделать програмирование более удобным, QSqlDatabase реализован как класс-значение (англ. "value class"). Любые изменения сделаные в соединении с базой данных через один объект QSqlDatabase будут влиять на другие объекты QSqlDatabase представляющие это же соединение. Вызовите cloneDatabase(), если вы хотите создать независимое соединение с базой данных на основе существующего. Если вы нуждаетесь во множестве соединений с базами данных одновременно, определите произвольное имя в addDatabase() и database(). Вызовите removeDatabase(), чтобы удалить соединение. QSqlDatabase выведет предупреждение, если вы попытаетесь удалить соединение, указанное в других объектах QSqlDatabase. Используйте contains(), чтобы видеть если заданное имя соединения есть в списке соединений. Как только соединение установлено вы можете посмотреть, какие таблицы база данных предоставляет с помощью tables(), найти первичный индекс для таблицы с помощью primaryIndex(), получить мета-информацию о полях таблицы (например, их имена) с помощью record() и выполнить запрос с помощью exec(). Если транзакции поддерживаются, вы можете использовать transaction(),чтобы начать транзакцию, и затем commit() или rollback(), чтобы завершить ее. Вы можете узнать поддерживается ли транзакция используя QSqlDriver::hasFeature(). При использовании транзакции вы должны начать транзакцию, прежде чем создадите свой запрос. Если произошла ошибка, она может быть получена с помощью lastError(). Имена SQL драйверов доступны из drivers(), вы можете проверить доступность драйвера с помощью isDriverAvailable(). Если вы создали свой собственный драйвер, вы можете зарегистрировать его с помощью registerSqlDriver(). Смотрите также QSqlDriver, QSqlQuery, Модуль QtSql и Потоки и Модуль QtSql. Цитата QSqlDatabase QSqlDatabase::addDatabase ( const QString & type, const QString & connectionName = QLatin1String( defaultConnection ) ) [static] Добавляет базу данных в список соединений баз данных используя драйвер type и имя соединения connectionName. Если уже существует соединение с базой данных называющееся connectionName, этосоединение удаляется. Соединения с базой данных в дальнейшем именуется connectionName. Вновь добавленое соединение будет возвращено. Если параметр connectionName не определен, внови добавленное соединение становится для приложения соединением поумолчанию, и последующие вызовы database() без параметра вернут ссылку на него. Если connectionName задан, используйте database(connectionName), чтобы вернуть указатель на соединение. Предупреждение: Если вы добавите базу данных с тем же именем как и у существующей базы данных, старая база данных будет заменена новой. Это происходит автоматически, если вы вызываете эту функцию больше, чем один раз без указания connectionName. Чтобы использовать соединение, вам нужно настроить его, например, вызвав некоторые или все из фунций setDatabaseName(), setUserName(), setPassword(), setHostName(), setPort(), и setConnectOptions(), а затем вам нужно открыть соединение с помощью open(). Замечание: Эта функция потокобезопасная. Смотрите также database(), removeDatabase(), и Драйвера баз данных SQL. acen83, silver, почитайте вот эту тему Базы данных |
|
|
silver |
6.8.2008, 14:43
Сообщение
#10
|
Новичок Группа: Новичок Сообщений: 8 Регистрация: 4.8.2008 Из: Харьков Пользователь №: 250 Спасибо сказали: 1 раз(а) Репутация: 0 |
Что делать? Скорее всего, у тебя остаются висеть незакрытые запросы. Причём не обязательно именно QSqlQuery, но и в виде QSqlQueryModel, QSqlTableModel и так далее по иерархии. Убедись, что к моменту закрытия соединения, все твои модели и запросы либо выходять из области видимости (если создавались на стеке):
либо удаляются оператором delete или родителем (если создал их как QSqlQuery *query = new QSqlQuery). В моём случае проблема была именно в этом. |
|
|
Текстовая версия | Сейчас: 13.1.2025, 18:55 |