历史上的今天
返回首页

历史上的今天

今天是:2025年03月25日(星期二)

正在发生

2018年03月25日 | Socket采用C/S模型进行设计的服务器模型

2018-03-25 来源:嵌入式ARM

Socket的中文翻译为“插座”,在计算机世界里称为套接字。Socket最初是作为网络上不同主机之间进程的通信接口,后来应用越来越广,在同一主机上的不同进程之间通信也可以用Socket。简单来说,当网络上不同主机之间的两个进程(A、B)采用Socket进行通信时,那么它们之间需要建立一个通信端点,即创建Socket,创建Socket时就分配端口号和网络地址。当进程A向进程B发送数据时,那么进程A必须要知道进程B的网络地址及端口号。


Socket采用C/S模型进行设计的,即Client/Server,面向客户端—服务器模型。


每一个Socket都用一个半相关描述:


{协议,本地地址,本地端口}


一个完整的Socket则用一个相关描述:


{协议,本地地址,本地端口,远程地址,远程端口}


一、Socket的类型


Socket有三种类型:


1、字节流套接字(SOCK_STREAM)


字节流的套接字可以提供可靠的数据传输、面向连接的通讯流。数据按何种顺序发送,就按何种顺序接收。例如,当我们按顺序发送A-B-C,那么在数据到达接收端时,它的顺序也是A-B-C。字节流套接字采用的是TCP(Transmission Control Protocol)协议。保证了数据传输的可靠性。


2、数据报套接字(SOCK_DGRAM)


数据报套接字定义了一种无连接的服务。所谓无连接服务,简单来说,即在发送数据时,无需在收发两端建立类似TCP那样的握手连接,在发送时,将数据打包,然后加上远程IP地址,即可把该数据包发送出去。


数据通过相互独立的报文进行传输。并且是无序的、不可靠的传输。


3、原始套接字(SOCK_ROW)


原始套接字是我们需要关心的,因为我们的Socket CAN采用的即是原始套接字。该接口允许对较底层协议进行操作,如IP、ICMP等。原始套接字常用于检验新的协议实现或访问现有服务中配置的新设备。


套接字的工作流程如下:


先启动服务器,通过调用socket()函数建立一个套接字,然后调用bind()函数将该套接字和本地网络地址联系在一起,再调用listen()函数使套接字做好侦听的准备,并规定它的请求队列的长度,之后就调用accept()函数来接收连接。客户端在建立套接字之后就可调用 connect()和服务器建立连接。连接一旦建立,客户端和服务器之间就可以通过调用recv()/recvfrom()函数和send()/sendto函数来进行发收数据。最后,待数据传送结束后,双方调用close()函数关闭套接字。


下面我们来写两个简单的基于Socket的CAN应用程序,但是我们采用的是SOCK_ROW,因此在套接字工作流程上有区别于SOCK_STREAM和SOCK_DGRAM。由于Socket采用C/S模型进行设计的,所以我们的这两个程序也分别为Server和Client。


首先是server端的程序,我们需要写一个服务器的程序,该程序接收来自客户端发来的数据,代码如下:

int can_recv()

{

int sock_fd;

unsigned long nbytes, len;

struct sockaddr_can addr;

struct ifreq ifr;

/*为了能够接收CAN报文,我们需要定义一个CAN数据格式的结构体变量*/

struct can_frame frame;

struct can_frame *ptr_frame;

/*建立套接字,设置为原始套接字,原始CAN协议 */

sock_fd = socket(PF_CAN,SOCK_RAW,CAN_RAW);

/*以下是对CAN接口进行初始化,如设置CAN接口名,即当我们用ifconfig命令时显示的名字 */

strcpy(ifr.ifr_name,"can0");

ioctl(sock_fd, SIOCGIFINDEX, &ifr);

printf("can0 can_ifindex = %x\n",ifr.ifr_ifindex);

/*设置CAN协议 */

addr.can_family = AF_CAN;

addr.can_ifindex = 0;

/*将刚生成的套接字与网络地址进行绑定*/

bind(sock_fd, (struct sockaddr*)&addr, sizeof(addr));

/*开始接收数据*/

nbytes = recvfrom(sock_fd, &frame, sizeof(struct can_frame), 0, (struct sockaddr *)&addr, &len);

/*get interface name of the received CAN frame*/

ifr.ifr_ifindex = addr.can_ifindex;

ioctl(sock_fd, SIOCGIFNAME, &ifr);

printf("Received a CAN frame from interface %s\n",ifr.ifr_name);

/*将接收到的CAN数据打印出来,其中ID为标识符,DLC为CAN的字节数,DATA为1帧报文的字节数*/

printf("CAN frame:\n ID = %x\n DLC = %x\n" \

"DATA = %s\n",frame.can_id,frame.can_dlc,frame.data);

ptr_frame = &frame;

return 0;

}

接下来是CAN的发送程序,即客户端,代码如下:       

int can_send()

{

int sock_fd;

unsigned long nbytes;

struct sockaddr_can addr;

struct ifreq ifr;

struct can_frame frame;

/*建立套接字,设置为原始套接字,原始CAN协议 */

sock_fd = socket(PF_CAN,SOCK_RAW,CAN_RAW);

/*以下是对CAN接口进行初始化,如设置CAN接口名,即当我们用ifconfig命令时显示的名字 */

strcpy((char *)(ifr.ifr_name), "can0");

ioctl(sock_fd, SIOCGIFINDEX, &ifr);

printf("can0 can_ifindex = %x\n", ifr.ifr_ifindex);

addr.can_family = AF_CAN;

addr.can_ifindex = ifr.ifr_ifindex;

/*将刚生成的套接字与CAN套接字地址进行绑定*/

bind(sock_fd, (struct sockaddr*)&addr, sizeof(addr));

/*设置CAN帧的ID号,可区分为标准帧和扩展帧的ID号*/

frame.can_id = 0x1122;

strcpy((char *)frame.data,"hello");

frame.can_dlc = strlen(frame.data);

printf("Send a CAN frame from interface %s\n", ifr.ifr_name);

/*开始发送数据*/

nbytes = sendto(sock_fd, &frame, sizeof(struct can_frame), 0, (struct sockaddr*)&addr, sizeof(addr));

return 0;

}


上面两个程序看完后,大家可能会有疑问,为什么这两个程序没有listen()和accept()函数呢?其实这两个程序是独立的运行的,并不像字节流套接字(SOCK_STREAM)和数据报套接字(SOCK_DGRAM),需要先运行服务器进行侦听。SOCK_STREAM和SOCK_DGRAM的两个server和client程序是通过网络相互收发数据。而CAN的socket的server和client程序收发数据的对象是CAN总线。server从CAN总线上接收数据,client将数据发到CAN总线上,当CAN总线上有数据时,server才能接收数据,当CAN总线空闲时,client才能将数据发送出去。


以上是对套接字的简单理解,并附上socket CAN的简单上层应用代码。

推荐阅读

史海拾趣

乾坤(Cyntec)公司的发展小趣事

面对工业4.0的浪潮,乾坤公司积极响应国家智能制造的号召,开始进行智能制造的转型。公司引进了先进的自动化设备,对生产线进行了全面升级,实现了生产过程的自动化、信息化和智能化。通过智能制造的转型,乾坤不仅提高了生产效率,还降低了生产成本,同时提升了产品的一致性和可靠性。这一转型为乾坤公司的持续发展注入了新的动力,使其在激烈的市场竞争中保持了领先地位。

ATC [American Technical Ceramics]公司的发展小趣事

在电子行业的发展历程中,技术突破一直是推动公司成长和市场扩张的关键因素。假设ATC在某一时期成功研发出了一种新型的陶瓷材料,这种材料在耐高温、抗腐蚀等方面具有显著优势,因此被广泛应用于半导体制造和电子设备领域。这一技术突破使得ATC的产品在市场上获得了极高的认可,公司也因此实现了快速的收入增长和市场占有率的提升。

Everbuild公司的发展小趣事

随着全球对环保和可持续发展的重视,Everbuild也将绿色环保理念融入到产品设计和生产过程中。公司采用环保材料和节能技术,减少生产过程中的污染和能源消耗。同时,Everbuild还积极参与环保公益活动,倡导绿色生活方式。这些举措不仅提高了公司的社会责任感,也为公司的可持续发展奠定了基础。

Freqtech Ohg公司的发展小趣事

面对电子行业快速迭代的市场环境,Freqtech Ohg公司敏锐地察觉到传统业务面临的挑战。公司果断决定进行业务转型升级,从单一的高频电子元件生产向系统解决方案提供商转变。通过整合上下游产业链资源,Freqtech成功开发出了一系列针对特定应用场景的高频电子系统解决方案,如5G通信基站、卫星导航系统等。这一转型不仅为公司带来了新的增长点,还进一步巩固了其在行业内的领先地位。

Aptiv公司的发展小趣事

Aptiv公司在电子行业中以其领先的技术实力著称。早在1995年,公司便成立了自动驾驶研究实验室,开始探索自动驾驶技术的可能性。经过多年的研发与试验,Aptiv在自动驾驶领域取得了显著的突破。2004年,公司首次在内华达州成功测试了自动驾驶汽车,获得了州政府的批准,这标志着Aptiv在自动驾驶技术上迈出了坚实的步伐。此后,Aptiv不断推动自动驾驶技术的商业化进程,为全球汽车行业带来了新的发展机遇。

BusBoard Prototype Systems公司的发展小趣事

BusBoard Prototype Systems公司起源于一位电子工程师的梦想。这位工程师发现,在电子产品的开发过程中,原型制作是一个既耗时又昂贵的环节。于是,他萌生了创建一个专门提供快速、高效原型制作服务的公司的想法。经过数月的筹备,BusBoard Prototype Systems正式成立,以其独特的BusBoard技术和定制化的服务,迅速在行业内崭露头角。

问答坊 | AI 解惑

TD340芯片在直流调速系统中的应用

文中介绍了TD340芯片特点,给出了TD340芯片在直流调速系统中的电路…

查看全部问答>

PFC直接电流控制策略综述

    0 引言    PFC的控制策略按照输入电感电流是否连续,分为电流断续模式(DCM)和电流连续模式(CCM),以及介于两者之间的临界DCM(BCM)。有的电路还根据负载功率的大小,使得变换器在DCM和CCM之间转换,称为混连模式(Mixed Con ...…

查看全部问答>

焊接材料的选配

大家可以下来看看…

查看全部问答>

做移动机器人仿真程序心得

一些实验的数据如下: [700pixel*500pixel/14000mmx10000mm]实验室标准长度; 中等线速度为0.5m/s,最大线速度为1M/S,最小可以 为0,当在最小测距信息为49pixel~24.5pixel时可以0.5m/s~1m/s 之间变化,机器人身宽半径为0.39m;K =700pixel/1400 ...…

查看全部问答>

C51内存单元释放问题???????

不是说局部变量可以自动释放的吗???我的为什么不行??例如:大概写一 点,说明问题。。 key()   //定义一个函数 { int a; } main()  //主函数 { key(); while(1); } 为什么我在key函数内每定义一个变量,KELL ...…

查看全部问答>

求学ARM

我是还没有出道的学生!什么都不懂,我不知道从何开始学ARM,请大家多多帮忙,谢谢!…

查看全部问答>

【藏书阁】电磁场理论(经典电磁场书籍)

目录: 第一章 静电场 第二章 静电场的边值问题 第三章 稳定电场 第四章 静磁场 第五章 似稳磁场 第六章 时变电磁场 第七章 平面电磁波 第八章 电磁导波和电磁振荡 第九章 电磁辐射和电磁绕射 第十章 解析函数及其应用 第十一章 特殊 ...…

查看全部问答>

请问一个基本共射放大电路问题

1请问在基本共射放大电路中Ubeq是一个常量吗?它只跟管子构造有关吗?如果是这样的话为什么? 2请问si管是0.6-0.8,Ge管是0.1-0.3伏吗?…

查看全部问答>

为什么会有这些波形呢?

如上图,我将前几天的IGBT改为MOS管,但是还是发热,测D极的波形,发现是这种尖峰波形了,大家帮忙看看是怎么回事?…

查看全部问答>