创建于 2020-07-31,最终修订于 2020-08-06
在之前的介绍课程(零)中我们说了要用六节课来实现一个HTTP,
今天的任务是 “[易]简单的Cmake的教程,选用一个Socket库并实现一个echo”。
所有的代码都在 https://github.com/dashjay/http_demo/tree/master/1-cmake-socket-echo 中
Let’s do it
#include
int main(){
std::cout << "Hello World!" << '\n';
}
我们本可以使用如下命令进行编译并且运行
g++ -o main main.cpp
./main
Hello World!
但是我们要尝试使用一次Cmake[1]
# 指定cmake的版本,随意了,我们不会用太高级的功能
cmake_minimum_required(VERSION 3.16)
# set可以设定一个变量
set(project_name http_demo)
# 项目名称 ${xxxx} 可以导入之前的变量
project(${project_name})
set(CMAKE_CXX_COMPILER "c++") # 编译器
set(CMAKE_CXX_STANDARD_REQUIRED True)
set(CMAKE_CXX_STANDARD 17) # cpp指定标准
# 设定编译的FLAGS
set(CMAKE_CXX_FLAGS -g -W)
string(REPLACE ";" " " CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}")
message(STATUS "CMAKE_CXX_FLAGS: " "${CMAKE_CXX_FLAGS}")
# 添加一个可执行文件
add_executable(main main.cpp)
通常情况下我们不会再项目目录里执行 cmake .
,这会导致很多混乱的问题,我们通常创建一个build文件夹做这个操作
mkdir build; cd build
cmake .. && make
./main
Hello World!
build目录下会产生很多文件,通常我们不用理会他们,只需要管我们的输出文件即可。
make新手教程
当项目复杂到一定程度时,有大量自动化的操作可以节省我们的时间,例如我们可以针对Hello World输出生成一个测试,每次编写完程序,我们只需要make一下(如果cmake文件没有变化的话),然后运行make test来进行自动化测试(测试样例需要我们自己编写)。当然这个Cmake在本次项目中的作用也不是很能体现出优势,但是笔者认为仍然有必要学习这种自动化工具。
经过一些挑选,选中了一个叫Sockpp的库。这个库用法很“现代”CPP,十分舒适,有很多上古socket库,用法或者代码比较古老了,我就只推荐这个了。
安装也很简单
git clone https://github.com/fpagliughi/sockpp
cd sockpp; mkdir build; cd build
cmake .. && make && sudo make install
这样,sockpp就安装到你系统中了,确切的说是分别安装了头文件和依赖库到你系统的 include 和 lib 中。
我们在本节课中暂时只简单的使用 tcp 接收一些数据,因此我们在本节课程里只引入 sockpp 的一个头部 #include "sockpp/tcp_acceptor.h"
这样几行代码,就可以指定端口绑定,监听,等待连接的到来。
sockpp::tcp_acceptor acc(port);
if (!acc){
std::cerr << acc.last_error_str() << '\n';
exit(1)
}
使用accept函数就可以和到来的连接创建一个socket链接,
sockpp::tcp_socket sock = acc.accept();
不多废话,直接看代码
#include "sockpp/tcp_acceptor.h"
#include
void print_without_crlf(char *ptr);
int main(){
// 指定监听端口
int16_t port = 8080;
// 使用端口启动监听
sockpp::tcp_acceptor acc(port);
if (!acc){
std::cerr << acc.last_error_str() << '\n';
exit(1);
}
std::cout << "try accept a conn at port " << port << '\n'
// 接收连接请求
sockpp::tcp_socket sock = acc.accept();
// 创建缓存
char buf[1024];
// 循环接收打印并且echo
for(;;){
if(auto siz = sock.read(buf, 1024); siz > 0){
print_without_crlf(buf);
sock.write(buf, siz);
}else{
// siz < 0(断开连接)会退出
break;
}
}
std::cout << "closed" << '\n';
}
// 这个函数为什么这么写稍后解释
void print_without_crlf(char *ptr){
std::cout << "recv: ";
for(;;){
if(*ptr == '\r' || *ptr == '\n'){
break;
}
std::cout << *ptr;
ptr+=1;
}
std::cout << '\n';
}
我们首先尝试直接使用 g++ 编译这串代码
g++ -o server server.cpp
# 以下是输出
~/CLionProjects/http_demo » g++ -o server server.cpp
In file included from server.cpp:1:
In file included from /usr/local/include/sockpp/tcp_acceptor.h:48:
In file included from /usr/local/include/sockpp/acceptor.h:48:
In file included from /usr/local/include/sockpp/inet_address.h:50:
/usr/local/include/sockpp/sock_address.h:94:3: warning: 'auto'
type specifier is a C++11 extension [-Wc++11-extensions]
auto p = sockaddr_ptr();
^
/usr/local/include/sockpp/sock_address.h:116:9: error: unknown
type name 'constexpr'
static constexpr size_t MAX_SZ = sizeof(sockaddr_storage);
为什么直接编译会这样报错
我不知道 g++(clang++) 默认使用什么版本,也许是 c++11 或者 c++98 但是因为 sockpp 是一个比较新的库,我们这里推荐指定 c++11 以上 c++17 当然更好了,在 C++17 中还可以使用 if(STATEMENT;CONDITIONS){}
这种用法,美滋滋。
上方的有关 cpp 版本的描述不是十分专业,请自行查询更详细的资料
c++03 c++2a gnu++03 gnu++2a iso9899:1990
c++11 c++98 gnu++11 gnu++98 iso9899:199409
c++14 c11 gnu++14 gnu11 iso9899:1999
c++17 c89 gnu++17 gnu89 iso9899:2011
c++1y c90 gnu++1y gnu90 iso9899:2017
c++1z c99 gnu++1z gnu99
g++ -std=c++17 -o server server.cpp
# 以下是输出
Undefined symbols for architecture x86_64:
"sockpp::inet_address::create(unsigned int, unsigned short)", referenced from:
sockpp::inet_address::inet_address(unsigned short) in server-40d3ec.o
"sockpp::stream_socket::read_timeout(std::__1::chrono::duration > const&)" , referenced from:
....
在指定了版本之后,为什么还是会报错,因为sockpp不仅仅是一个头部申明,他复杂还有函数定义被编译成了动态链接库。(这里放置一个占位符,等找到教程了之后放在这里),因此我们需要添加链接库:-lsockpp
,不出意外的话,下面的命令可以帮助你成功编译我们的server。
g++ -std=c++17 -o server server.cpp -lsockpp
./server
try accept a conn at port 8080
recv: helloworld # 看下方的telnet可以找到输入
closed
print_without_crlf
是什么? [了解即可]CRLF其实是两个字符,回车换行的意思,他们分别是
(CR, ASCII 13, \r)
(LF, ASCII 10, \n)
全称是 Carriage-Return Line-Feed ,可以代表回车的东西,这里有一个Stack Overflow的提问可以帮助你思考这个东西。
如果到这里你不知道ptr是个指针,*ptr会取指针的值,那么你应该先去了解一下有关指针的内容。(占位符,我之后会自己写一篇)。
这串代码,编译成功后按照预期运行。我们即可尝试连接他,并且发送消息。在linux和macos下(好像window也有,只是默认隐藏),有一个小工具叫做 telnet
。这个工具在我们前期的开发中帮助很大,建议找出来并且使用,或者搜索下载一个类似的网络测试助手,可以发起TCP链接并发送数据的小公举。
telnet 0.0.0.0 8080
Trying 0.0.0.0...
Connected to 0.0.0.0.
Escape character is '^]'.
hello world # <- 发出去的数据
hello world # <- 收到的数据
我们在Cmake中添加一个可执行文件,向之前的FLAG添加一个-lsockpp
set(CMAKE_CXX_FLAGS -g -W -lsockpp)
...
add_executable(server server.cpp)
然后我们立刻执行
cd build
cmake .. # 每次CMakeLists.txt变化后需要重新执行
make
# 你有可能编译成功,有可能失败(在笔者这里失败了)
make
Scanning dependencies of target server
[ 25%] Building CXX object CMakeFiles/server.dir/server.cpp.o
clang: warning: -lsockpp: 'linker' input unused [-Wunused-command-line-argument]
/Users/dashjay/CLionProjects/http_demo/server.cpp:1:10: fatal error:
'sockpp/tcp_acceptor.h' file not found
#include "sockpp/tcp_acceptor.h"
推测是cmake并没有使用系统的的include目录和lib链接目录,导致的,搜索文档后手动添加这两个目录,在笔者电脑上这两个目录分别是/usr/local/include
和/usr/local/lib
,修改后的CMake文件像这样。(已经去掉注释了)
cmake_minimum_required(VERSION 3.16)
set(project_name http_demo)
project(${project_name})
set(CMAKE_CXX_COMPILER "c++")
set(CMAKE_CXX_STANDARD_REQUIRED True)
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_FLAGS -g -W -lsockpp)
string(REPLACE ";" " " CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}")
message(STATUS "CMAKE_CXX_FLAGS: " "${CMAKE_CXX_FLAGS}")
include_directories(/usr/local/include)
link_directories(/usr/local/lib)
add_executable(main main.cpp)
add_executable(server server.cpp)
再次执行cmake .. && make
后,成功编译出server
,执行,并且使用 telnet
小工具帮助我们调试。
今天我们做了以下事情:
还有什么问题可以直接到ISSUE中提问,我或者社区里的其他小伙伴会帮你解答的。
[1] Cmake可以帮助跨平台软件的安装和编译,可以通过命令生成一系列Makefile,然后进一步执行编译,测试等指令(对我们来说就这个作用,实际上功能比这个要强大的多。)