用MinGW和CMake搭建便捷的C/C++开发环境(二)

七、CMake练习
准备工作:在E盘新建一个文件夹cmakeproj,作为工程目录,
在cmakeproj文件夹中,建立src文件夹,用来存放源代码;建立build文件夹,用来构建我的应用程序;
在cmakeproj文件夹中,新建CMakeList.txt文件,我用工程根目录中的CMakeList.txt文件做以下几件事情:
1、指定CMake的最低版本
CMAKE_MINIMUM_REQUIRED(VERSION 2.8) 
2、设定项目名称

PROJECT(CMakeProj)
3、指定子目录
ADD_SUBDIRECTORY(src)
这里还应当了解两个重要的变量${PROJECT_SOURCE_DIR}和${PROJECT_BINARY_DIR},他们分别是项目源代码目录和项目输出目录,可以用MESSAGE指令输出。
MESSAGE(STATUS "源码目录:" ${PROJECT_SOURCE_DIR})
MESSAGE(STATUS "编译目录:" ${PROJECT_BINARY_DIR})

4、在src文件夹中,建立mylibs和myapps文件夹
在mylibs中,建立mylib.cpp和mylib.h文件,代码如下:
 

  
  
  
  
  1. mylib.h 
  2. //---------------------------------------------------------------------------------------------------- 
  3. #ifndef _MYLIB_H 
  4. #define _MYLIB_H 
  5.  
  6. class MyLib{ 
  7. public
  8.  void DoSomething(); 
  9. }; 
  10.  
  11. #endif 
  12. //---------------------------------------------------------------------------------------------------- 
  13. mylib.cpp 
  14. //---------------------------------------------------------------------------------------------------- 
  15. #include "mylib.h" 
  16. #include <iostream> 
  17.  
  18. void MyLib::DoSomething() { 
  19.    std::cout << "hi~! 我是一条来自MyLib中DoSomething的消息" << std::endl; 
  20. //---------------------------------------------------------------------------------------------------- 

 

在mylibs文件夹中的CMakeLists.txt文件中写CMake指令:
ADD_LIBRARY(mylibs mylib)

ADD_LIBRARY指令的第一个参数为库文件的名称,这里指定为mylibs,将来make之后,会生成一个名称为libmylibs.a的库文件,第二个参数为源码,可以只写文件名mylib,也可以写mylib.h空格mylib.cpp,如果头文件和源文件的名称一致简写为mylib就可以了。
5、在myapps文件夹中,建立myapp.cpp文件

  
  
  
  
  1. myapp.cpp 
  2.  
  3. #include <iostream> 
  4. #include "mylib.h" 
  5.  
  6. int main(int argc, const char *argv[]) 
  7.  MyLib.DoSomething(); 
  8.  return 0; 

 

在myapps文件夹中的CMakeLists.txt文件中写CMake指令:
告诉编译器共享库的位置:
INCLUDE_DIRECTORIES(${PROJECT_SOURCE_DIR}/src/mylibs)
根据myapp.cpp源文件生称名称为exename的可执行文件:exename.exe
ADD_EXECUTABLE(exename myapp.cpp)
链接可执行文件需要的动态库文件libmylibs.a
TARGET_LINK_LIBRARIES(exename mylibs)

6、构建
在build文件夹中:cmd:cmake –G”MinGW Makefiles” ..  在build文件夹中已经生成了我需要的Makefie,
接下来cmd:mingw32-make
在build文件夹的src文件夹中的myapps文件夹中,成功的生成了名称为exename.exe的可执行文件,在build文件夹的src文件夹中的mylibs文件夹中生成了libmylib.a文件。

八、CMake练习的深入
在上一步生成了exe文件后,还是有很多不如意的地方,例如exe文件生成的目录并不是我想要的等等等等。还需要对我的构建进行改进。
1、改变目标二进制的输出目录
在工程根目录中的CMakeList.txt文件中增加指令:
SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin)
SET(LIBRARY_OUTPUT_PATH ${PROJECT_BINARY_DIR}/lib)
2、增加版本号
SET(${PROJECT_NAME}_MAJOR_VERSION 0)
SET(${PROJECT_NAME}_MINOR_VERSION 1)
SET(${PROJECT_NAME}_PATCH_LEVEL 0)
但我设置后,在windows中并没有看到效果。去公司在linux下看看。
3、生成共享库
修改src/mylibs/中的CMakeLists.txt文件
ADD_LIBRARY(mylibs SHARED mylib)
ADD_LIBRARY的参数形式为:ADD_LIBRARY (库名称 库类型 源文件)
只需要增加参数SHARED就可以了,在make之后,build/lib/中可生成mylibs.dll文件。
如果想同时生成静态库和共享库,如何实现?

 

九、使用其他库文件
建立一个目录,用来存放一些常用的库(e:/cpplibs)。
(一)sqlite
从最简单的sqlite开始。在E盘建目录cpplibs,在cpplibs文件夹中建立sqlite目录,在sqlite中建立include和lib文件夹,分别存放sqlite3.h文件和libsqlite3.a文件。
在myapps文件夹中的CMakeLists.txt文件中增加:
INCLUDE_DIRECTORIES(d:/cpplibs/sqlite/include)
LINK_DIRECTORIES(d:/cpplibs/sqlite/lib)
TARGET_LINK_LIBRARIES(exename sqlite3 mylibs)
告诉编译器到哪里去找sqlite的头文件和库,连接sqlite3的库文件。
运行cmake和mingw32-make,sqlite3的应用程序可以正常运行了。但是在上面到处写类似d:/cpplibs/sqlite/include这样的路径,如果cpplibs名称改变了,那就太致命了。在项目根目录的CMakeLists.txt文件中定义一个变量:
SET(SQLITE3_BASE d:/cpplibs/sqlite)
将上面两行CMake指令修改为:
INCLUDE_DIRECTORIES(${SQLITE3_BASE}/include)
LINK_DIRECTORIES(${SQLITE3_BASE}/lib)
这样看起来好多了。再次编译时,只要修改一下变量SQLITE3_BASE的值就可以了。

(二)MySQL++
1、定义两个变量MYSQL_BASE和MYSQLPP_BASE,分别存放mysql和mysql++的位置,在myapps文件夹中的CMakeLists.txt文件中增加:
SET(MYSQL_BASE "C:/Program Files/MySQL/MySQL Server 5.1")
SET(MYSQLPP_BASE "E:/MySQL++")
因为CMakeLists.txt的语法是以空格区分参数是否结束的,所以路径加了引号,在CMake的语法中,加不加引号其实都没有关系,但如果值中有空格或分号“;”,还是需要加引号。
2、源码目录(这里是myapps)的CMakeLists.txt文件修改为:

#INCLUDE_DIRECTORIES
#告诉编译器共享库的位置:
#sqlite mysql mysql++
INCLUDE_DIRECTORIES("${SQLITE3_BASE}/include" "${MYSQL_BASE}/include" "${MYSQLPP_BASE}/include")

#LINK_DIRECTORIES
#告诉外部依赖库的搜索路径
#sqlite mysql mysql++
LINK_DIRECTORIES("${SQLITE3_BASE}/lib" "${MYSQLPP_BASE}/lib" "${MYSQL_BASE}/lib")

最后别忘了告诉编译器mysqlpp的依赖项
TARGET_LINK_LIBRARIES(${PROJECT_NAME} "${MYSQLPP_BASE}/lib/libmysqlpp.a")

开始写一些代码试试看,已经可以很顺利的make了。

  
  
  
  
  1. #include <cstdlib> 
  2. #include <cstdio> 
  3. #include <iostream> 
  4. #include <sqlite3.h> 
  5.  
  6. #include <mysql++.h> 
  7.  
  8. #include "mylib.h" 
  9.  
  10. using namespace std; 
  11.  
  12. // sqlite callback 
  13. static int callback(void *NotUsed, int argc, char **argv, char **azColName){ 
  14.   int i; 
  15.   for(i=0; i<argc; i++){ 
  16.     printf("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL"); 
  17.   } 
  18.   printf("\n"); 
  19.   return 0; 
  20. // end sqlite callback 
  21.  
  22.  
  23. int main(int argc, const char *argv[]) 
  24.     // mylib 
  25.     MyLib().DoSomething(); 
  26.      
  27.     // sqlite 
  28.     sqlite3 *db; 
  29.     char *zErrMsg = 0; 
  30.     int rc; 
  31.      
  32.     char sql[1024]=""
  33.     sprintf(sql, "SELECT * FROM table1;"); 
  34.  
  35.     rc = sqlite3_open("test.db", &db); 
  36.     if (rc) 
  37.     { 
  38.        fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(db)); 
  39.        sqlite3_close(db); 
  40.        system("PAUSE"); 
  41.        exit(1); 
  42.     } 
  43.    
  44.     rc = sqlite3_exec(db, sql, callback, 0, &zErrMsg); 
  45.     if ( rc!=SQLITE_OK ) 
  46.     { 
  47.         fprintf(stderr, "SQL error: %s\n", zErrMsg); 
  48.     } 
  49.     sqlite3_close(db); 
  50.     // end sqlite 
  51.  
  52.     // mysql 
  53.     mysqlpp::Connection con(false); 
  54.  
  55.     con.set_option(new mysqlpp::SetCharsetNameOption("gbk")); 
  56.  
  57.     cout << "请输入数据库(root用户)连接密码:"
  58.     string pwd; 
  59.     getline(cin, pwd); 
  60.  
  61.     if (!con.connect("mytable""localhost""root", pwd.c_str())) 
  62.     { 
  63.         cout << "无法连接,请检查密码是否正确!" << endl; 
  64.         return -1; 
  65.     } 
  66.     else 
  67.     { 
  68.         cout << "shit.终于连上了。" << endl; 
  69.         mysqlpp::Query query = con.query("select mycol from firsttable"); 
  70.         if (mysqlpp::StoreQueryResult res = query.store()) { 
  71.             cout << "We have:" << endl; 
  72.             mysqlpp::StoreQueryResult::const_iterator it; 
  73.             for (it = res.begin(); it != res.end(); ++it) { 
  74.                 mysqlpp::Row row = *it; 
  75.                 cout << '\t' << row["mycol"] << endl; 
  76.                 // 或者使用列索引 
  77.                 //cout << '\t' << row[0] << endl; 
  78.             } 
  79.         } 
  80.         else { 
  81.             cerr << "Failed to get mycol list: " << query.error() << endl; 
  82.             return 1; 
  83.         } 
  84.     } 
  85.     // end mysql 
  86.  
  87.  
  88.     system("PAUSE"); 
  89.     return EXIT_SUCCESS; 

 

(三)wxWidgets
在myapps的CMakeLists.txt文件中:
#wxWidgets
#MinGW 对库的顺序是有要求的,这一点很重要
FIND_PACKAGE(wxWidgets REQUIRED)
INCLUDE(${wxWidgets_USE_FILE})

之后:
TARGET_LINK_LIBRARIES(${PROJECT_NAME} ${wxWidgets_LIBRARIES})

下面修改myapp.cpp的代码,写一个wxWidgets的hello world,wxFrame窗口已经呈现在眼前。

myapp.cpp的代码就是wxWidgets官方网站的helloword:

  
  
  
  
  1. /* 
  2.  * hworld.cpp 
  3.  */ 
  4.  
  5. #include "wx/wx.h"  
  6.  
  7. class MyApp: public wxApp 
  8.     virtual bool OnInit(); 
  9. }; 
  10.  
  11. class MyFrame: public wxFrame 
  12. public
  13.  
  14.     MyFrame(const wxString& title, const wxPoint& pos, const wxSize& size); 
  15.  
  16.     void OnQuit(wxCommandEvent& event); 
  17.     void OnAbout(wxCommandEvent& event); 
  18.  
  19.     DECLARE_EVENT_TABLE() 
  20. }; 
  21.  
  22. enum 
  23.     ID_Quit = 1, 
  24.     ID_About, 
  25. }; 
  26.  
  27. BEGIN_EVENT_TABLE(MyFrame, wxFrame) 
  28.     EVT_MENU(ID_Quit, MyFrame::OnQuit) 
  29.     EVT_MENU(ID_About, MyFrame::OnAbout) 
  30. END_EVENT_TABLE() 
  31.  
  32. IMPLEMENT_APP(MyApp) 
  33.  
  34. bool MyApp::OnInit() 
  35.     MyFrame *frame = new MyFrame( _("Hello World"), wxPoint(50, 50), 
  36.                                   wxSize(450,340) ); 
  37.     frame->Show(true); 
  38.     SetTopWindow(frame); 
  39.     return true
  40. }  
  41.  
  42. MyFrame::MyFrame(const wxString& title, const wxPoint& pos, const wxSize& size) 
  43. : wxFrame( NULL, -1, title, pos, size ) 
  44.     wxMenu *menuFile = new wxMenu; 
  45.  
  46.     menuFile->Append( ID_About, _("&About...") ); 
  47.     menuFile->AppendSeparator(); 
  48.     menuFile->Append( ID_Quit, _("E&xit") ); 
  49.  
  50.     wxMenuBar *menuBar = new wxMenuBar; 
  51.     menuBar->Append( menuFile, _("&File") ); 
  52.  
  53.     SetMenuBar( menuBar ); 
  54.  
  55.     CreateStatusBar(); 
  56.     SetStatusText( _("Welcome to wxWidgets!") ); 
  57.  
  58. void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event)) 
  59.     Close(TRUE); 
  60.  
  61. void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event)) 
  62.     wxMessageBox( _("This is a wxWidgets Hello world sample"), 
  63.                   _("About Hello World"), 
  64.                   wxOK | wxICON_INFORMATION, this); 

编译,运行。

 

总结:

      以前喜欢用vc,vs,但vs越来越庞大,每一次的安装都有如梦魇,相信很多朋友和我一样都有同样的感受。为了有一个轻便的环境,我逐渐转移到Devc++,再到CodeBlocks,但总是会遇到这样或那样的问题没有办法解决。我知道这并不是IDE的问题,而是我不知如何解决。
      当然我并不是反对使用IDE,我本人也很依赖IDE。人类之所以发明工具,是因为工具可以提高生产效率。上面提到的这些IDE都是极品,尤其Visual studio,是我认为最强大的开发工具,确切的说是C#最强大的开发工具,我甚至一度认为没有其他的开发工具可以超越它,即便如此,我还是想知道在vs强大的背后到底是什么?现在我似乎知道了一些:-)。

      学习和使用CMake,开始入手时比较困难,但我才经过总共不超过20个小时的学习,已经基本可以开始用CMake来完成大多数的日常工作了。但是还有一些迷惑,总不能不停的在命令行中敲命令吧,刚开始熟悉命令时多敲几次还过得去,命令已经非常熟悉了,而且使用频率也很高,不停的敲那也太麻烦了,在第三篇学习记录中解决这个问题,地址在这里:

http://www.cnblogs.com/ode/archive/2011/08/04/2152251.html

参考资料:
1、CMake文档
2、CMake实践.pdf
3、http://bbs.osgchina.org/viewthread.php?tid=1189 
   http://bbs.osgchina.org/viewthread.php?tid=1229    

作者:vincent zhang
出处:http://ode.cnblogs.com http://odevincent.blog.51cto.com
Email:wsaspx#hotmail.com
知识共享许可协议
本作品由vincent zhang创作,采用知识共享署名-非商业性使用-禁止演绎 3.0 中国大陆许可协议进行许可。

你可能感兴趣的:(C++,mysql,sqlite3,mysql++,c++连接数据库)