漫画C语言 做个聊天软件你不懂也得懂
yuyutoo 2024-12-22 21:47 4 浏览 0 评论
学完C语言做不出东西?不存在的,咱们做一个最“隐私”的聊天器,就俩人,你和我。咱们聊天的信息你知我知没别人知。
我们直接开始写代码,只要你会基础的C语言,不要担心看不懂,不懂的我帮你刨根问底,把根都挖出来嚼烂,绝对懂。
一、一个聊天软件的基础模型是怎么样的?
你是个新手的话你可能就会问,什么是模型?!听不懂,我在骗你学习。放心,我现在就告诉你什么是基础“模型”。
我们可以简单的理解“模型”指这个聊天软件基本是怎么进行通信的,常规形式是怎样的,只要清楚了这个形式流程,然后在这个流程中添加一些代码就ok了,啥都不用想。如果你还是不懂什么是“流程”,那我就跟你说这个是一个步骤,只需要懂这个步骤,我们使用代码编写这个步骤就可以完成了。
好了,现在没啥问题了吧?现在开始,第一步在一个通信中,一般有一个服务端。那什么是服务端?
1.1 什么是服务端
服务端就简单了,曾经…曾经…你去例如移动或者联通的营业挺,客服小姐姐就会对你提供服务,例如业务办理,办个卡,销个号等,那我们的服务端是用来通信的,所以这个服务端就是指等待跟我聊天的人,只要你上线了,开电脑打开软件了,连接上我的服务端了,咱们就可以聊天了。
服务端一般就是一直在这里等你上线的那个,风里雨里我在这里等你。
1.2 又不懂什么是客户端了?
不懂没关系,打游戏懂吧?你下载到你电脑你手机的就是客户端,你打个游戏如果没有服务端就不能跟人匹配,这个懂了吧?
1.3 基本的工具要拿过来吧?
还知道头文件吧?
头文件就等于是一个工具箱,需要干啥就可以使用拿头文件过来,这样就可以用里面的工具了。
那咱们做一个聊天的软件就需要一个工具箱吧,这个工具箱叫做“winsock2.h”,那怎么拿呢?都知道#include<> 吧?
那就直接把这个头文件拿过来就好了,代码就可以写成:#include<winsock2.h>。
常规的输入输出工具箱也要拿吧?所以就第一步把 stdio.h 也拿过来,所以这个服务端的第一行第二行代码就写成:
#include<stdio.h>
#include<WinSock2.h>
1.4 开始 socket 编程
不会了不会了!是不是一说 socket 你就说这是个什么鬼?
我先说一句让你懵逼的定义“socket 就是应用之间通信的端点”。懂不懂?
不懂呀,那我继续说。
socket 就是两个通信软件之间的接口,你可以当成服务端是“插座”,客户端是“插头”,一插,欧了!这样不就通电了,这样说你明白了吧?
当然这样解释比较片面,但用“抽象”的方式讲又不一定能让大家听得懂,所以你就理解成插头肯定没问题。
1.5 开始抬杠我拿三座插两座插不进!
咱们用的插头都是有标准的,你想想,没有标准怎么那么多电器都可以用常规的插头?
像这个 socket 这个通信端口,是有基于一些标准的。例如 TCP/IP这些通信协议。
好了,我说了TCP/IP可能就会有同学问,这又是什么鬼!没关系,你只需要知道这个是一个通信协议,咱们现在是用 socket 进行通信就好,知道 socket 怎么用就行,协议咱们可不需要现在搞懂,咱们只需要知道 socket 如何运用即可。
二、开始敲服务端代码
2.1 搞清楚使用 socket 进行通信的步骤
编写C语言Windows下的socket需要经过几个步骤:首先对WSAStartup 进行初始化,初始化对socket 套接字(socket也叫套接字)进行创建,随后配合绑定信息,接着进行配置信息的bind 绑定;绑定了信息后,通过该信息进行isten 监听,监听后若有链接则connect 连接,再接下来开始使用accept 接收请求,得到请求后可以选择接受recv或者send发送数据,最后closesocket 关闭 socket,WSACleanup 最终关闭。
简单点就是下面的这个流程:
不懂了?不懂就慢慢来嘛。
这是进行 socket 编程的步骤,如果你要问为什么要这样做…我只能回答你规定的流程就这样,因为你要进行通信,那肯定需要创建一个 socket ,创建完毕后那么肯定要绑定你要通信的信息,如果你不绑定你怎么知道你要跟谁说话呢?急着我收到了一个信息后就等于跟我请求通话,我同意了,咱们就开始通信了,通信肯定要发送信息,那就用send这些方法发送了,最后面说完话我就关闭这个 socket了,那你说不是吗?
还不懂?那你看下面。
2.1 第一步初始化
既然第一步是初始化,那我要初始化什么东西?
我们需要初始化一个 WSADATA 类型数据的对象。
什么鬼?又是 WSADATA 又是对象的,听不懂啊!
没关系的拉,WSADATA 其实就是一个结构体,咱们在把使用socket的工具箱 WinSock2 拿过来的时候这个 WSADATA 结构体就已经创建好了,直接使用这个结构体创建一个结构体变量就好了。
WSADATA 的作用就是用来存储初始化信息的,就像你打个游戏初始化创建一个人,这个人总得有信息吧,光头、小眼睛、腿短…对吧?
那么我们的代码就可以写成以下:
#include<stdio.h>
#include<WinSock2.h>
#include <stdlib.h>
int main(){
WSADATA wsaData;
}
接下来就可以开始初始化了,初始化 socket 有一个函数叫做 WSAStartup,既然是函数一般都有参数吧,参数有哪些呢?
这个 WSAStartup 方法需要传入一个 版本号,还有一个用于存储信息的 WSADATA 结构体。现在我们已经知道 WSADATA 的结构体就是上面这个代码创建的 wsaData 结构体变量,那么版本号又是什么?
这个版本号是说明我们使用哪个 Winsock 版本,Winsock 有一个 1.1 版本还有一个 2.2 版本。两个版本有不同,1.1 版本只支持 TCP/IP 协议,还有一个版本 2.2 支持多个协议,这个时候你懂用哪个了吧?
什么?! 还不懂? 那肯定是全都要呀!
2.2 版本兼容性之类的更好,兼容啥我们不管,反正用多的。
那直接写成 WSAStartup(2, 2, &wsaData)?
不不不,我们写法有一些不同,需要用一个函数 MAKEWORD 对版本进行生成,就像这样 WSAStartup(MAKEWORD(2, 2), &wsadata);,规定咱们使用 MAKEWORD 告诉 WSAStartup 初始化调用什么版本。
那么整个初始化的代码就如下所示咯:
#include<stdio.h>
#include<WinSock2.h>
#include <stdlib.h>
int main(){
WSADATA wsaData;
WSAStartup(MAKEWORD(2, 2), &wsadata);
}
什么?不懂 &wsadata ?来来来,我们的漫画同学告诉你是啥意思:
懂了吧?传个地址方便信息存储。
2.2 第二步创建 socket
这一步超级简单,代码就是这个:
SOCKET serverSock = socket(PF_INET, SOCK_STREAM, 0);
我知道你要骂我,写什么是什么鬼。
好了好了,首先 SOCKET 是一个socket的类型,还记得 int a 吧?int 是一个类型,那么 SOCKET 肯定就是一个类型了,说明创建一个 SOCKET 类型的变量,然后 socket() 是创建 socket 的函数,这个没毛病吧?
你说是里面的参数不懂?
小问题了,第一个 PF_INET 就表示指定 IPV4 ,也就是说先给个网络协议,那么多的网络协议你总要选一个吧。那为什么要用 IPv4 呢?我只能说用这个东西计算更快,毕竟咱们做个聊天软件是局域网通信,你就理解为,咱们做的东西是个“小东西”,没必要那么大“体量”,迷你更好用,那就用那个 IPV4 了,你想不开你也可以用 IPV6 试试。
那 SOCK_STREAM 是什么?SOCK_STREAM 表示咱们进行的通信是 TCP 通信,稳定可靠。在这里使用 SOCK_STREAM 也表示向我们的系统,或者你理解成“计算机”申请一个通信的端口,不然系统不给你“开个口子”,我的数据怎么传出去对吧,不然就是叫破喉咙都没人理我。
那最后一个参数 0 又是什么呢?
这里就是一个编号,说仔细点这个是 socket 所使用的传输协议编号,是不是不明白?其实这就是一个编号,不做设置,但是要给一个值,所以就给一个 0 咯。
2.3 第三步绑定信息
绑定信息这一步就有点玄了。在这里咱们要了解两个结构体,一个是 sockaddr_in,还有一个是 SOCKADDR。需要注意的是,这两个结构体包含的数据都是一样的,是一样的…
主要是使用上有区别。有啥区别?
sockaddr 是个系统用,而 sockaddr 是用来强制转换 sockaddr_in 结构体给系统调用的函数用。是不是迷茫?不要迷茫,一般都是这样做,那就这样做吧。你只需要记住,sockaddr 保存信息然后就别管了,而sockaddr 咱们就用来给参数给函数用。
在 socket 中,咱们使用 sockaddr_in 结构体绑定监听的 IP 信息,首先需要创建这个结构体:
struct sockaddr_in sockAddr;
接下来始绑定端口、IP类型,其中 127.0.0.1 表示本机、1234 表示监听端口:
sockAddr.sin_family = PF_INET; //IPv4
sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //服务器的IP
sockAddr.sin_port = htons(1234); //端口
这个懂没懂?
sockAddr.sin_family 是表示这个结构体中用于存储IP协议的结构体变量,PF_INET 之前说了是 ipV4,表示在这里设置 ipV4类型。
sockAddr.sin_addr.s_addr 这里是表示需要绑定的 ip 地址,在这里使用 inet_addr(“127.0.0.1”) 进行指定。那为什么指定个 ip 还需要 inet_addr?
inet_addr 的作用是将一个字符串格式的ip地址转换成一个uint32_t数字格式。为什么要转换?那肯定是因为 sockAddr.sin_addr.s_addr 是一个 uint32_t 这个类型了。
最后的 sockAddr.sin_port 是表示要指定某一个端口,在这里指定 1234 这个端口。
所以该部分的代码就写成这样了:
#include<stdio.h>
#include<WinSock2.h>
#include <stdlib.h>
int main(){
WSADATA wsaData;
WSAStartup(MAKEWORD(2, 2), &wsadata);
SOCKET serverSock = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in sockAddr;
sockAddr.sin_family = PF_INET; //IPv4
sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //服务器的IP
sockAddr.sin_port = htons(1234); //端口
}
最后就是绑定一下了:
bind(serverSock, (SOCKADDR*)&sockAddr, sizeof(SOCKADDR));
在这里 bind() 方法就是表示绑定信息了,第一个参数是 serverSock 就是表示要绑定的 socket,然后 (SOCKADDR*)&sockAddr 就是需要绑定的地址,最后一个就是一个地址长度。
(SOCKADDR*)&sockAddr 我们讲过,SOCKADDR 就是给函数使用的,sockAddr 就是给系统使用的,所以就这样写就没毛病了。
2.4 监听端口
先让你懵一下,下面是代码:
listen(serverSock, 20);
简单吧?listen 就是表示监听,第一个参数就是要监听的 socket 第二个就是表示 同时能处理的最大连接。终于简单了这一步,你爽我也爽,还不懂就看下面漫画。
2.5 有人请求聊天?设置个接待员
接下来就是有人请求给你聊天了,那怎么办呢?一个人忙不过来呢,那就设置个接待员。
SOCKADDR cIntAddr;
int nSize = sizeof(SOCKADDR);
SOCKET cIntSock = accept(serverSock, (SOCKADDR*)&cIntAddr, &nSize);
accept 函数就是一个接待员,有人连接来敲门了,就需要去接待,换句比较专业的话就是 accept 接收一个套接字中已建立的连接。
传入的参数第一个 serverSock 就是一个已连接的套接字,(SOCKADDR*)&cIntAddr 是一个按照规定的指向struct sockaddr的指针,所以我猜在前面创建,最后一个就是所指向这个指针的长度咯。
设置完后就等于创建了一个接待员 cIntSock 。
不过要注意,accept 没有连接的时候就会一直在等待,不然不会执行下面的代码的。
这一部分的代码如下:
#include<stdio.h>
#include<WinSock2.h>
#include <stdlib.h>
int main(){
WSADATA wsaData;
WSAStartup(MAKEWORD(2, 2), &wsadata);
SOCKET serverSock = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in sockAddr;
sockAddr.sin_family = PF_INET; //IPv4
sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //服务器的IP
sockAddr.sin_port = htons(1234); //端口
listen(serverSock, 20);
SOCKADDR cIntAddr;
int nSize = sizeof(SOCKADDR);
SOCKET cIntSock = accept(serverSock, (SOCKADDR*)&cIntAddr, &nSize);
}
2.6 开始循环聊天
在聊天的时候肯定是需要一个循环,不用循环只能发一次信息就完成了,所以肯定有一个 while:
while (1) {
}
那循环里面写啥?
当然是写你接收信息和发送信息的代码了,我一次性贴上,简简单单:
while (1) {
char sendBuf[50]={"Hello client"};
char recvBuf[50];
recv(cIntSock, recvBuf, 50, 0);
printf("来自客户端:");
printf("%s\n", recvBuf);
printf_s("请输入内容:");
scanf("%s",sendBuf);
//sendBuf="s";
//gets_s(sendBuf);
send(cIntSock, sendBuf, strlen(sendBuf) + 1, 0);
}
sendBuf就是一个字符数组,用来输入自己的要输入的内容。
主要看recv,recv 接收4个参数,第一个参数是建立的通信、第二个参数是是一个数组,接收数据存放的地方、之后会缓存大小,最后一个参数是指定调用方式,不用管一般设置为0。
cIntSock 就是刚刚从套接字里接受的那个接待员,现在就用接待员和他说话了。
接着就使用printf显示接待员听到的话,简简单单。
然后就到我们输入信息,使用scanf够简单了吧?
接着使用 send函数发送信息就可以了,第一个就是告诉接待员 cIntSock 要传达话了,sendBuf 就是咱们要说的话,第三个参数就是咱们说的话的长度,最后一个依旧是0,不用管。
这样就还差最后一步就完成服务端了,此时咱们只需要关闭套接字就可以了,最后还需要清理一下,完整代码如下了:
#include<stdio.h>
#include<WinSock2.h>
#include <stdlib.h>
int main()
{
WSADATA wsadata;
WSAStartup(MAKEWORD(2, 2), &wsadata);
SOCKET serverSock = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in sockAddr;
sockAddr.sin_family = PF_INET;
sockAddr.sin_addr.s_addr = htons(INADDR_ANY);
sockAddr.sin_port = htons(1234);
bind(serverSock, (SOCKADDR*)&sockAddr, sizeof(SOCKADDR));
listen(serverSock, 20);
SOCKADDR cIntAddr;
int nSize = sizeof(SOCKADDR);
SOCKET cIntSock = accept(serverSock, (SOCKADDR*)&cIntAddr, &nSize);
while (1) {
char sendBuf[50]={"Hello client"};
char recvBuf[50];
recv(cIntSock, recvBuf, 50, 0);
printf("来自客户端:");
printf("%s\n", recvBuf);
printf_s("请输入内容:");
scanf("%s",sendBuf);
send(cIntSock, sendBuf, strlen(sendBuf) + 1, 0);
}
//关闭
closesocket(cIntSock);
closesocket(serverSock);
WSACleanup();
return 0;
}
三、客户端编写
客户端和服务端是一样的你信吗?
下面是代码:
#include<stdio.h>
#include<winsock2.h>
int main()
{
WSADATA wsadata;
int nRes = WSAStartup(MAKEWORD(2, 2), &wsadata);
SOCKET sock = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in sockAddr;
sockAddr.sin_family = PF_INET;
sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //只需要在这里指向服务器 ip 就可以了
sockAddr.sin_port = htons(1234);
//连接服务器
connect(sock, (SOCKADDR*)&sockAddr, sizeof(SOCKADDR));
while (1) {
char recvBuf[50];
char sendBuf[50]={"Hello server"};
printf("跟服务端说: ");
scanf("%s",sendBuf);
send(sock, sendBuf, strlen(sendBuf) + 1, 0);
recv(sock, recvBuf, 50, 0);
printf("服务端跟你说: ");
printf("%s\n", recvBuf);
}
closesocket(sock);
WSACleanup();
system("pause");
}
不同的几个点只有使用了 connect 连接服务器就没了,难道你说不是吗?
简简单单对吧?那就行,解决。
下面是演示示例:
注意 若使用devc复制代码都报错,则点击编译->编译选项:
随后在出现的窗口中添加如下参数:
相关推荐
- Mysql和Oracle实现序列自增(oracle创建序列的sql)
-
Mysql和Oracle实现序列自增/*ORACLE设置自增序列oracle本身不支持如mysql的AUTO_INCREMENT自增方式,我们可以用序列加触发器的形式实现,假如有一个表T_WORKM...
- 关于Oracle数据库12c 新特性总结(oracle数据库19c与12c)
-
概述今天主要简单介绍一下Oracle12c的一些新特性,仅供参考。参考:http://docs.oracle.com/database/121/NEWFT/chapter12102.htm#NEWFT...
- MySQL CREATE TABLE 简单设计模板交流
-
推荐用MySQL8.0(2018/4/19发布,开发者说同比5.7快2倍)或同类型以上版本....
- mysql学习9:创建数据库(mysql5.5创建数据库)
-
前言:我也是在学习过程中,不对的地方请谅解showdatabases;#查看数据库表createdatabasename...
- MySQL面试题-CREATE TABLE AS 与CREATE TABLE LIKE的区别
-
执行"CREATETABLE新表ASSELECT*FROM原表;"后,新表与原表的字段一致,但主键、索引不会复制到新表,会把原表的表记录复制到新表。...
- Nike Dunk High Volt 和 Bright Spruce 预计将于 12 月推出
-
在街上看到的PandaDunk的超载可能让一些球鞋迷们望而却步,但Dunk的浪潮仍然强劲,看不到尽头。我们看到的很多版本都是为女性和儿童制作的,这种新配色为后者引入了一种令人耳目一新的新选择,而...
- 美国多功能舰载雷达及美国海军舰载多功能雷达系统技术介绍
-
多功能雷达AN/SPY-1的特性和技术能力,该雷达已经在美国海军服役了30多年,其修改-AN/SPY-1A、AN/SPY-1B(V)、AN/SPY-1D、AN/SPY-1D(V),以及雷神...
- 汽车音响怎么玩,安装技术知识(汽车音响怎么玩,安装技术知识视频)
-
全面分析汽车音响使用或安装技术常识一:主机是大多数人最熟习的音响器材,有关主机的各种性能及规格,也是耳熟能详的事,以下是一些在使用或安装时,比较需要注意的事项:LOUDNESS:几年前的主机,此按...
- 【推荐】ProAc Response系列扬声器逐个看
-
有考牌(公认好声音)扬声器之称ProAcTablette小音箱,相信不少音响发烧友都曾经,或者现在依然持有,正当大家逐渐掌握Tablette的摆位设定与器材配搭之后,下一步就会考虑升级至表现更全...
- #本站首晒# 漂洋过海来看你 — BLACK&DECKER 百得 BDH2000L无绳吸尘器 开箱
-
作者:初吻给了烟sco混迹张大妈时日不短了,手没少剁。家里有了汪星人,吸尘器使用频率相当高,偶尔零星打扫用卧式的实在麻烦(汪星人:你这分明是找借口,我掉毛是满屋子都有,铲屎君都是用卧式满屋子吸的,你...
- 专题|一个品牌一件产品(英国篇)之Quested(罗杰之声)
-
Quested(罗杰之声)代表产品:Q212FS品牌介绍Quested(罗杰之声)是录音监听领域的传奇品牌,由英国录音师RogerQuested于1985年创立。在成立Quested之前,Roger...
- 常用半导体中英对照表(建议收藏)(半导体英文术语)
-
作为一个源自国外的技术,半导体产业涉及许多英文术语。加之从业者很多都有海外经历或习惯于用英文表达相关技术和工艺节点,这就导致许多英文术语翻译成中文后,仍有不少人照应不上或不知如何翻译。为此,我们整理了...
- Fyne Audio F502SP 2.5音路低音反射式落地音箱评测
-
FyneAudio的F500系列,有新成员了!不过,新成员不是新的款式,却是根据原有款式提出特别版。特别版产品在原有型号后标注了SP字样,意思是SpecialProduction。Fyne一共推出...
- 有哪些免费的内存数据库(In-Memory Database)
-
以下是一些常见的免费的内存数据库:1.Redis:Redis是一个开源的内存数据库,它支持多种数据结构,如字符串、哈希表、列表、集合和有序集合。Redis提供了快速的读写操作,并且支持持久化数据到磁...
- RazorSQL Mac版(SQL数据库查询工具)
-
RazorSQLMac特别版是一款看似简单实则功能非常出色的SQL数据库查询、编辑、浏览和管理工具。RazorSQLformac特别版可以帮你管理多个数据库,支持主流的30多种数据库,包括Ca...
你 发表评论:
欢迎- 一周热门
-
-
前端面试:iframe 的优缺点? iframe有那些缺点
-
带斜线的表头制作好了,如何填充内容?这几种方法你更喜欢哪个?
-
漫学笔记之PHP.ini常用的配置信息
-
其实模版网站在开发工作中很重要,推荐几个参考站给大家
-
推荐7个模板代码和其他游戏源码下载的网址
-
[干货] JAVA - JVM - 2 内存两分 [干货]+java+-+jvm+-+2+内存两分吗
-
正在学习使用python搭建自动化测试框架?这个系统包你可能会用到
-
织梦(Dedecms)建站教程 织梦建站详细步骤
-
【开源分享】2024PHP在线客服系统源码(搭建教程+终身使用)
-
2024PHP在线客服系统源码+完全开源 带详细搭建教程
-
- 最近发表
-
- Mysql和Oracle实现序列自增(oracle创建序列的sql)
- 关于Oracle数据库12c 新特性总结(oracle数据库19c与12c)
- MySQL CREATE TABLE 简单设计模板交流
- mysql学习9:创建数据库(mysql5.5创建数据库)
- MySQL面试题-CREATE TABLE AS 与CREATE TABLE LIKE的区别
- Nike Dunk High Volt 和 Bright Spruce 预计将于 12 月推出
- 美国多功能舰载雷达及美国海军舰载多功能雷达系统技术介绍
- 汽车音响怎么玩,安装技术知识(汽车音响怎么玩,安装技术知识视频)
- 【推荐】ProAc Response系列扬声器逐个看
- #本站首晒# 漂洋过海来看你 — BLACK&DECKER 百得 BDH2000L无绳吸尘器 开箱
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)