這是我的問題,我有一個sqlite
memory database
使用QSql
。我有幾個線程每個處理這個公共數據庫的一個不同的表。我用Win API
,以確保工作在不同的處理器這些線程,就像這樣:在多處理器線程中優化內存數據庫的QSql
SetThreadAffinityMask (hThread, processorMask);
如果只有線程處理一個表,它需要10秒,並使用總CPU的25%。但是當有4個線程處理4個不同的表格時,它需要將近40秒,並且只佔總CPU的35%。我認爲原因是在一個數據庫中存在某種thread-safe
同步。但由於不同的線程讀取或寫入不同的表格,線程安全性會降低我的程序。我怎樣才能優化它。
更新:最可能的原因是某些類型的鎖的Qt
和/或Sqlite 3
內減緩我的程序,所以是有可能關閉或繞過這些鎖通過預先設置。
Update2:下面是一個示例。 (也許有點長,對不起)
class MultiProcessorThread
{
public:
virtual void run();
bool start()
{
m_hThread = CreateThread (NULL, 0, MultiProcessorThread::ThreadFunc, this, CREATE_SUSPENDED, NULL);
if (m_hThread != INVALID_HANDLE_VALUE)
{
RunningThreadCount++;
m_ProcessorMask = 1 << ((RunningThreadCount - 1) % ProcessorCount);
SetThreadAffinityMask (m_hThread, m_ProcessorMask); // Make thread working on different processor
ResumeThread (m_hThread);
return true;
}
else
return false;
}
protected:
static DWORD WINAPI ThreadFunc (LPVOID in);
HANDLE m_hThread;
DWORD_PTR m_ProcessorMask;
static DWORD_PTR ProcessorCount;
static DWORD_PTR RunningThreadCount;
static DWORD_PTR GetNumCPUs();
};
DWORD_PTR MultiProcessorThread::ProcessorCount = GetNumCPUs();
DWORD_PTR MultiProcessorThread::RunningThreadCount = 0;
DWORD_PTR MultiProcessorThread::GetNumCPUs() // Get how many processors on this PC
{
SYSTEM_INFO m_si = {0};
GetSystemInfo (&m_si);
return (DWORD_PTR) m_si.dwNumberOfProcessors;
}
DWORD WINAPI MultiProcessorThread::ThreadFunc (LPVOID in)
{
static_cast<MultiProcessorThread*> (in)->run();
return 0;
}
class Run : public MultiProcessorThread
{
public:
void run()
{
int i = 0;
QString add = "insert into %1 values(1)";
add = add.arg (table);
QString sel = "select a from %1 ";
sel = sel.arg (table);
QString del = "delete from %1 where a=1";
del = del.arg (table);
while (++i) // read and write database
{
query.exec (add);
query.exec (sel);
query.exec (del);
}
}
QSqlQuery query;
QString table;
};
int main (int argc, char *argv[])
{
QCoreApplication a (argc, argv);
QSqlDatabase db = QSqlDatabase::addDatabase ("QSQLITE", "test");
db.setDatabaseName (":memory:"); // All threads working on the same memory database.
db.open();
QSqlQuery q (db), q1 (db), q2 (db);
q.exec ("create table A (a)");
q1.exec ("create table B (a)");
q2.exec ("create table C (a)"); // All threads working on different table.
Run b[3];
b[0].query = QSqlQuery (q);
b[0].table = "A";
b[1].query = QSqlQuery (q1);
b[1].table = "B";
b[2].query = QSqlQuery (q2);
b[2].table = "C";
b[0].start();
b[1].start();
b[2].start();
return a.exec();
}
這真的取決於你如何使用sqlite3。粘貼你正在用數據庫做的一些例子。 – synthesizerpatel 2012-01-24 11:30:30
在繼續進行優化之前,請看一下這個問題:http://stackoverflow.com/questions/1680249/how-to-use-sqlite-in-a-multi-threaded-application – Neox 2012-01-24 12:18:05
@Neox它會幫助一些,但我的情況有一些差異。首先,在我的程序中,我直接使用'QSql'而不是'sqlite',所以我不能選擇如何編譯sqlite3庫。第二我不需要在線程之間進行同步,因爲它們在不同的表上運行。第三我使用內存數據庫,所以我必須共享相同的連接。 – xucheng 2012-01-24 12:30:51