多人操作sqlite3数据库冲突问题解决方法
问题描述:sqlite3数据放置在某一台电脑的某个共享文件夹下,操作数据库的应用程序安装在同一局域网下的很多台电脑上,由于存在多人同时使用该应用程序,所以存在多人同时操作数据库的情况。经过测试发现,最常见的情况是当两人或者多人往数据库中写入数据时,只有其中一个写入成功,其他数据都写入失败。
解决方案分析:
由于本人编写程序未MFC应用程序,所以尝试使用windows互斥量mutex,具体的使用方法如下:
bool CMFCApplication2Dlg::Lock()
{
m_pMutex = CreateMutex(NULL, false, L"txt_mutex");
if (NULL == m_pMutex)
{
return false;
}
DWORD nRet = WaitForSingleObject(m_pMutex, INFINITE);
if (nRet != WAIT_OBJECT_0)
{
return false;
}
return true;
}
bool CMFCApplication2Dlg::UnLock()
{
return ReleaseMutex(&m_pMutex);
}
在某用户开始进行写入操作时,先调用Lock()获取mutxt,写入完成之后调用UnLock()释放mutxt。然并卵,该方法并不奏效。(可能由于本人对windows多线程/多进程编程这一块太过生疏,所以无法利用这方面的知识来解决这个问题,如果有大神知道解决方法,求不吝赐教)
所以经过一番思考之后,决定使用在共享盘的那台电脑上跑一个服务端小程序来防止数据库的操作冲突。
具体实现方法如下:
思路分析:在服务端每收到一个客户端的连接请求之后,都创建一个新的线程来处理相应的操作,新的线程不断的去获取客户端发来的消息,当客户端发来的消息是”开始操作”时,线程将尝试获取互斥量mutxt,获取成功之后将给客户端发送回复消息,当该线程接收到”操作结束”的消息时,线程将释放互斥量mutxt,并且会断开该客户端与服务端的socket连接。
服务端代码:
#include <WinSock2.h>
#include <Windows.h>
#include <iostream>
#include <string>
#pragma comment(lib, "ws2_32.lib")
using namespace std; static HANDLE m_mutex = INVALID_HANDLE_VALUE; DWORD WINAPI AnswerThread(LPVOID lparam)
{
SOCKET ClientSocket = (SOCKET)(LPVOID)lparam;
int bytesRecv;
while ()
{
bytesRecv = SOCKET_ERROR;
char sendbuff[] = "ok";
char recvbuf[] = "";
for (int i = ; i<(int)strlen(recvbuf); i++)
{
recvbuf[i] = '\0';
}
while (bytesRecv == SOCKET_ERROR)
{
bytesRecv = recv(ClientSocket, recvbuf, sizeof(recvbuf), );
}
string recved = recvbuf;
if (recved == "op_begin")
{
WaitForSingleObject(m_mutex, INFINITE);
cout << "op_begin" << endl;
send(ClientSocket, sendbuff, sizeof(sendbuff), );
}
if (recved == "op_end")
{
cout << "op_end" << endl;
ReleaseMutex(&m_mutex);
closesocket(ClientSocket);
return ;
}
}
return ;
}
int main()
{
WSADATA wsaData;
int iRet = WSAStartup(MAKEWORD(, ), &wsaData);
if (iRet != NO_ERROR)
printf("Error at WSAStartup()\n"); SOCKET m_socket;
m_socket = socket(AF_INET, SOCK_STREAM, );
if (m_socket == INVALID_SOCKET)
{
printf("Error at socket():%ld\n", WSAGetLastError());
WSACleanup();
return ;
}
SOCKADDR_IN service;
service.sin_family = AF_INET;
service.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
service.sin_port = htons(); if (bind(m_socket, (SOCKADDR*)&service, sizeof(service)) == SOCKET_ERROR)
{
printf("bind() failed.\n");
closesocket(m_socket);
return ;
}
else
printf("bind ok.\n"); if (listen(m_socket, ) == SOCKET_ERROR)
printf("Error listening on socket.\n");
else
printf("listening ok.\n"); SOCKET AcceptSocket;
printf("waiting for a client to connect...\n");
m_mutex = CreateMutex(NULL, FALSE, L"Mutex");
if (!m_mutex)
{
cout << "Failed to CreateMutex !" << endl;
return ;
}
int count = ;
while ()
{
AcceptSocket = SOCKET_ERROR;
while (AcceptSocket == SOCKET_ERROR)
{
AcceptSocket = accept(m_socket, NULL, NULL);
}
count++;
printf("client num %d connected.\n", count); DWORD dwThreadId;
HANDLE hThread; hThread = CreateThread(NULL, NULL, AnswerThread, (LPVOID)AcceptSocket, , &dwThreadId);
if (hThread == NULL)
{
printf("CreatThread AnswerThread() failed.\n");
}
else
{
printf("create thread %d ok.\n", count);
}
CloseHandle(hThread);
}
closesocket(m_socket);
WSACleanup();
}
客户端代码:(进入数据库操作前)
WSADATA wsaData;
if (WSAStartup(MAKEWORD(, ), &wsaData) != )
{
TRACE("Failed to load Winsock");
return;
}
string txtPath = save_path + "\\ip.txt";
ifstream infile(txtPath);
string ip;
getline(infile, ip);
infile.close();
SOCKADDR_IN addrSrv;
addrSrv.sin_family = AF_INET;
addrSrv.sin_port = htons();
addrSrv.sin_addr.S_un.S_addr = inet_addr(ip.c_str());
sockClient = socket(AF_INET, SOCK_STREAM, );
if (SOCKET_ERROR == sockClient){
TRACE("Socket() error:%d", WSAGetLastError());
return;
}
if (connect(sockClient, (SOCKADDR*)&addrSrv, sizeof(addrSrv)) == INVALID_SOCKET){
TRACE("Connect failed:%d", WSAGetLastError());
return;
}
char buff[] = "op_begin";
send(sockClient, buff, sizeof(buff), );
int bytesRecv = SOCKET_ERROR;
char recvbuf[] = "";
for (int i = ; i<(int)strlen(recvbuf); i++)
{
recvbuf[i] = '\0';
}
while (bytesRecv == SOCKET_ERROR)
{
CMessageDlg message;
message.DoModal();
bytesRecv = recv(sockClient, recvbuf, sizeof(recvbuf), );
}
客户端代码:(数据库操作完成之后)
char buff[] = "op_end"; send(sockClient, buff, sizeof(buff), ); closesocket(sockClient);
注:由于经验不足,本方法可能有很多内存释放,资源利用等细节没有考虑到,所以本方法仅供参考。另外有关socket编程部分的代码参考:
http://blog.csdn.net/chence19871/article/details/44019633
多人操作sqlite3数据库冲突问题解决方法的更多相关文章
- 使用 SQLiteManager 操作 sqlite3 数据库
SQLiteManager https://github.com/misato/SQLiteManager4iOS 本人以前从事过嵌入式开发,后来转职为iOS开发,即使如此,也绝不想去碰C语言级别的面 ...
- 《Python操作SQLite3数据库》快速上手教程
为什么使用SQLite数据库? 对于非常简单的应用而言,使用文件作为持久化存储通常就足够了,但是大多数复杂的数据驱动的应用需要全功能的关系型数据库.SQLite的目标则是介于两者之间的中小系统.它有以 ...
- (一一三)使用系统自带框架操作SQLite3数据库
系统自带的框架是基于C语言的,使用比较繁琐. 下面是使用步骤: 首先导入libsqlite3.0.dylib. ①在Document目录下打开数据库,如果没有则创建. NSString *sqlite ...
- .net操作Oracle数据库步骤及方法
1.首先安装PL/SQL Developer Oracle客户端软件 2.安装Oracle Instant Client(即时客户端) 安装与配置 配置环境变量ORAClE HOME 地址为insta ...
- 安装mysql数据库及问题解决方法
1.mysql官网下载安装包,官网地址:www.mysql.com [root@seiang software]# ll total 580020 -rw-r--r--. 1 root root 59 ...
- sqlite3 not found问题解决方法
测试发现,有些Android手机自带sqlite3命令,有些不带.对于不带sqlite3的手机,我们可以手动将sqlite3加入系统. 执行如下命令 adb remount adb push 路径/s ...
- jQuery$命名冲突问题解决方法
也许你在看此文章之前还不知道jquery有一个noConflict()东西了,它就是为了避免与其它js插件碰到相同变量的一个解决方法,利用noConflict()可以把变量存到其它指定的变量中去如,我 ...
- pip2和pip3冲突问题解决方法
python使用pip安装模块时报错:unable to create process using ' '的解决方法: 参考:http://qoogle.cn/?id=39 1.删除C:\Python ...
- Flask:操作SQLite3(0.1)
Windows 10家庭中文版,Python 3.6.4,Flask 1.0.2 本文介绍了第一次在Flask框架中操作SQLite3数据库的测试,参考了官网的文档Using SQLite 3 wit ...
随机推荐
- Windows nexus 启动失败
现象: nexus Windows系统服务安装成功,但启动失败 D:\nexus-2.10.0-02-bundle\nexus-2.10.0-02\bin>nexus.bat Usage: ne ...
- Ubuntu安装图形桌面
apt-get直接更新即可 apt-get install ubuntu-desktop
- (旧)子数涵数·DW——图文混排页面
一.首先,打开Dreamweaver,新建一个的HTML项目. 二.在设计区里,写一些文字,随便写一点(也可以在代码区中的<body>和</body>之间写). 三.插入一张图 ...
- C++ 面向对象的三个特点--多态性(一)
C++的多态性定义 所谓多态性就是不同对象收到相同的消息产生不同的动作.通俗的说,多态性是指一个名字定义不同的函数,这些函数执行不同但又类似的操作,即用同样的接口访问功能不同的函数,从而实现“一个接口 ...
- 轻量级SaaS在线作图工具(继之前介绍后完整介绍)
俗话说“一图胜千言”,在办公应用领域,流程图是一个非常好的表现企业业务流程或工作岗位规范等内容的展现形式,比如去给客户做调研,回来后都要描述出客户的关键业务流程,谁.什么时候.在什么地方.负责什么事情 ...
- 如何解决div层被flash遮盖的问题
页面构建中的Flash层会遮挡Div的问题,一般通过设置wmode="transparent" 或wmode="window"就可以解决.不过对于Flash视频 ...
- ArcObject10.1降级至10.0
最开始接触ArcGIS版本是9.3,为了需要也安装了9.2进行开发:因为自己的电脑配置较低,所以跑不起10.0中文版:毕业工作后,行业内用10.1居多(虽然10.3已出):现在10.4都要出来了:由于 ...
- 转:Web应用程序项目XX已配置为使用IIS
转:http://www.cnblogs.com/Joetao/articles/2392526.html 今天在看开源项目Umbraco是,出现一个项目加载不了,并报如下错误: Web应用程序项目U ...
- SeismicPro地震剖面显示程序
SeismicPro是一个地震剖面显示软件,可从标准SEGY地震数据体中抽取纵测线和横测线的二维剖面,并以波形.变面积和变密度等多种方式进行专业化显示,可进行一键式显示方式切换,并可进行定制开发叠加井 ...
- android 圆角按钮和按钮颜色
1. android 设置圆角按钮后,按下按钮后,还能改变按钮的颜色 <?xml version="1.0" encoding="UTF-8"?> ...