-
Notifications
You must be signed in to change notification settings - Fork 50
/
Copy pathC++封装MySQL数据库连接器类和消息管理类.cpp
360 lines (336 loc) · 11.6 KB
/
C++封装MySQL数据库连接器类和消息管理类.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
#include <iostream>
#include <fstream>
#include <string>
#include <vector>
#include <ctime>
#include <stdexcept>
#include <unordered_map>
#include <windows.h> //必须在包含mysql.h之前包含windows.h
#include <mysql.h>
using std::string;
using std::vector;
using std::endl;
///数据库连接器类,负责连接数据库,并封装基本的数据库相关操作
class DB_Connector {
protected:
MYSQL* mysql;
public:
static string host_, user_, passwd_, db_;
public:
inline DB_Connector();
inline virtual ~DB_Connector();
//执行sql语句,成功则返回0,失败返回非0
inline int query(const string& sql);
//返回上一次的错误提示
inline string error();
//获取结果集,将其保存到二维string数组中
inline vector<vector<string> > getResult();
};
DB_Connector::DB_Connector()
{
mysql = mysql_init(NULL);
if (mysql_real_connect(mysql, host_.c_str(), user_.c_str(),
passwd_.c_str(), db_.c_str(), 0, 0, 0) == 0) {
string err = "Failed to connect database server: " + error();
throw std::logic_error(err);
}
query("set names 'GBK'");//设置字符集,防止中文出现乱码
}
DB_Connector::~DB_Connector()
{
mysql_close(mysql);
}
//执行sql语句,成功则返回0,失败返回非0
int DB_Connector::query(const string& sql)
{
return mysql_query(mysql, sql.c_str());
}
//返回上一次的错误提示
string DB_Connector::error()
{
return mysql_error(mysql);
}
//获取结果集,将其保存到二维string数组中
vector<vector<string> > DB_Connector::getResult()
{
vector<vector<string> >result;
MYSQL_RES* res = mysql_store_result(mysql);
if (res == NULL)
return result;
unsigned num = mysql_num_fields(res); //结果集的列数
for (MYSQL_ROW row; (row = mysql_fetch_row(res)) != NULL;) {
result.push_back(vector<string>(num));
for (size_t cur = result.size() - 1, i = 0; i < num; ++i) {
result[cur][i] = row[i];
}
}
mysql_free_result(res); //释放结果集资源
return result;
}
///消息管理类
class DB_Msg {
protected:
DB_Connector db_conn; //数据库连接器对象
const string tbName; //要操作的表名
std::ofstream fsLog; //日志记录文件流
public:
inline DB_Msg(const string& _tableName, const string& _logFileName);
inline virtual ~DB_Msg();
//添加消息
inline void push(const string& to, const string& from, const string& msg);
//获取执行指定sql语句后的结果
inline vector<vector<string> > getBySql(const string& sql);
//根据指定sql语句删除相应记录
inline void removeBySql(const string& sql);
//获取当前日期和时间,用于更新日志
inline static string getTime();
protected:
//创建存储聊天记录的表
inline void createTable();
};
DB_Msg::DB_Msg(const string& _tableName, const string& _logFileName)
: tbName(_tableName)
{
if (_logFileName != "") {
fsLog.open(_logFileName, std::ios::app);
}
createTable();
}
DB_Msg::~DB_Msg()
{
if (fsLog) {
fsLog.close();
}
}
//添加消息
void DB_Msg::push(const string& to, const string& from, const string& msg)
{
string sql = "insert into " + tbName + "(toUser,fromUser,msg) values('"
+ to + "','" + from + "','" + msg + "')";
if (fsLog) {
fsLog << getTime() << ">>>execute: " + sql << endl;
}
if (db_conn.query(sql) != 0 && fsLog) { //执行sql语句失败返回非0
fsLog << "Failed to insert: " << db_conn.error() << endl;
}
}
//获取执行指定sql语句后的结果
vector<vector<string> > DB_Msg::getBySql(const string& sql)
{
vector<vector<string> >result;
if (fsLog) {
fsLog << getTime() << ">>>execute: " + sql << endl;
}
if (db_conn.query(sql) == 0) { //执行成功则把结果输出
result = db_conn.getResult();
} else if (fsLog) {
fsLog << "Failed to search: " << db_conn.error() << endl;
}
return result;
}
//根据指定sql语句删除相应记录
void DB_Msg::removeBySql(const string& sql)
{
if (fsLog) {
fsLog << getTime() << ">>>execute: " << sql << endl;
}
if (db_conn.query(sql) != 0 && fsLog) {
fsLog << "Failed to delete: " << db_conn.error() << endl;
}
}
//获取当前日期和时间,用于更新日志
inline string DB_Msg::getTime()
{
char time_buf[64];
time_t now_time = time(NULL);
strftime(time_buf, 64, "%Y-%m-%d %H:%M:%S ", localtime(&now_time));
return time_buf;
}
//创建存储聊天记录的表
void DB_Msg::createTable()
{
db_conn.query("show tables like '" + tbName + "'");
auto res = db_conn.getResult();
if (res.size() > 0 && res[0][0] == tbName) {
return;
}
string sql = "CREATE TABLE `" + tbName + "` ("
"`id` tinyint(4) NOT NULL AUTO_INCREMENT,"
"`toUser` varchar(20) NOT NULL,"
"`fromUser` varchar(20) NOT NULL,"
"`time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,"
"`msg` text NOT NULL,"
"PRIMARY KEY (`id`)"
") ENGINE=InnoDB DEFAULT CHARSET=utf8;";
if (fsLog) {
fsLog << getTime() << ">>>execute: " << sql << endl;
}
if (db_conn.query(sql) != 0 && fsLog) {
fsLog << "Failed to create: " << db_conn.error() << endl;
}
}
///离线消息管理类 - 仅用于服务器端
class DB_OfflineMsg : public DB_Msg {
public:
DB_OfflineMsg(const string& _tableName, const string _logFileName = "")
: DB_Msg(_tableName, _logFileName)
{}
//从离线消息表中得到所有发送给user的消息,提取完后从数据库删除
inline vector<vector<string> > pop(const string& user);
};
//从离线消息表中得到所有发送给user的消息,提取完后从数据库删除
vector<vector<string> > DB_OfflineMsg::pop(const string& user)
{
string sql = "select fromUser,time,msg from " + tbName
+ " where toUser='" + user + "'";
vector<vector<string> >result = getBySql(sql);
//删除发给user的所有消息
if (result.size() > 0) {
sql = "delete from " + tbName + " where toUser='" + user + "'";
removeBySql(sql);
}
return result;
}
///聊天记录消息管理类
class DB_ChatLogMsg : public DB_Msg {
public:
//用户名标识用户,日志文件名为空时不建立前缀用于区分是在客户端还是服务器端
inline DB_ChatLogMsg(const string& _userName, const string _logFileName = "",
const string& _prefix = "")
: DB_Msg(_prefix + "chat_log_" + _userName, _logFileName), userName(_userName)
{}
//添加聊天记录消息
inline void push(const string& userOthers, const string& msg, bool isReceived = 1);
//添加收到的带时间记录的离线消息
inline void pushOffline(const string& from, const string& msg, const string& _time);
//获取消息user发来的或发给user的所有消息
inline vector<vector<string> > get(const string& user);
//在该用户的聊天记录中查找与withUser的包含str的消息记录
inline vector<vector<string> > find(const string& str, const string& withUser = "*");
//根据传递的用户名删除与对应用户的聊天记录,传入"*"则删除所有记录
inline void remove(const string& user);
//获取与当前用户有聊天记录的用户列表
inline vector<string> getUserWithChatLog();
//获取当前用户的用户名
inline const string& getUserName();
protected:
string userName; //该用户的用户名
};
//添加聊天记录消息
void DB_ChatLogMsg::push(const string& userOthers, const string& msg, bool isReceived)
{
if (isReceived) { //收到消息
DB_Msg::push(userName, userOthers, msg);
} else { //发送消息
DB_Msg::push(userOthers, userName, msg);
}
}
//添加收到的带时间记录的离线消息
void DB_ChatLogMsg::pushOffline(const string& from, const string& msg,
const string& _time)
{
string sql = "insert into " + tbName + "(toUser,fromUser,time,msg) values('"
+ userName + "','" + from + "','" + _time + "','" + msg + "')";
if (fsLog) {
fsLog << getTime() << ">>>execute: " + sql << endl;
}
if (db_conn.query(sql) != 0 && fsLog) { //执行sql语句失败返回非0
fsLog << "Failed to insert: " << db_conn.error() << endl;
}
}
//获取消息user发来的或发给user的所有消息
vector<vector<string> > DB_ChatLogMsg::get(const string& user)
{
string sql = "select fromUser,time,msg from " + tbName +
" where toUser='" + user + "' or fromUser='" + user + "'";
return getBySql(sql);
}
//在该用户的聊天记录中查找与withUser的包含str的消息记录
vector<vector<string> > DB_ChatLogMsg::find(const string& str, const string& withUser)
{
string sql = "select fromUser,toUser,time,msg from " + tbName + " where ";
if (withUser != "*") {
sql += "(toUser='" + withUser + "' or fromUser='" + withUser + "') and";
}
sql += " msg like '%" + str + "%'";
return getBySql(sql);
}
//根据传递的用户名删除与用户user的聊天记录,传入"*"则删除所有记录
void DB_ChatLogMsg::remove(const string& user)
{
string sql = "delete from " + tbName;
if (user != "*") {
sql += " where toUser='" + user + "' or fromUser='" + user + "'";
}
removeBySql(sql);
}
//获取与当前用户有聊天记录的用户列表
vector<string> DB_ChatLogMsg::getUserWithChatLog()
{
string sql = "select distinct toUser from " + tbName;
auto users = getBySql(sql);
std::unordered_map<string, bool> userList;
for (auto &elem : users) {
userList[elem[0]];
}
sql = "select distinct fromUser from " + tbName;
users = getBySql(sql);
for (auto &elem : users) {
userList[elem[0]];
}
userList.erase(userList.find(userName)); //删除用户列表中的自己
//将结果保存到vector<string>中返回
vector<string> res;
res.reserve(userList.size());
for (auto &it : userList) {
res.push_back(it.first);
}
return res;
}
//获取当前用户的用户名
inline const string& DB_ChatLogMsg::getUserName()
{
return userName;
}
using std::cout;
//可直接输出查询结果
inline std::ostream& operator<<(std::ostream& out, const vector<vector<string> >& res)
{
if (res.size() == 0) {
cout << "暂无内容可显示!" << endl;
return out;
}
for (auto &it : res) { //显示提取出的离线消息内容
for (auto &elem : it) {
cout << elem << " ";
}
cout << endl;
}
return out;
}
string DB_Connector::host_ = "localhost";
string DB_Connector::user_ = "root";
string DB_Connector::passwd_ = "123456";
string DB_Connector::db_ = "mfc_qq";
using namespace std;
int main()
{
DB_OfflineMsg off_msg("offline_msg", "offileMsg.log");
off_msg.push("to", "from", "hey"); //存入一条离线消息
auto res = off_msg.pop("to"); //读取发给该用户的全部离线消息
cout << res << endl;
DB_ChatLogMsg clMsg("Bob", "chat_log_Bob.log");
clMsg.push("Miranda-lym", "hello", 0);
clMsg.push("Miranda-lym", "hi!", 1);
clMsg.push("Miranda-lym", "this a test case!", 0);
clMsg.push("Miranda-lym", "soga,I get it!", 1);
clMsg.push("John", "what are you doing!", 1);
cout << clMsg.get("Miranda-lym") << endl;
cout << "find 'hi':\n" << clMsg.find("hi") << endl;
auto userList = clMsg.getUserWithChatLog();
cout << "users that have chated with " <<clMsg.getUserName()<<endl;
for (auto &elem : userList)
cout << elem << endl;
cin.get();
return 0;
}