C语言 TCP协议

1.总结:注意接收数据的处理才是最麻烦的,最麻烦的,留此记性,下次注意!!(16进制转16进制字符串需要特殊处理,跟我以前用c#处理方式完全不同,也可能是因为接触c语言不多,知识狭隘)!代码可以在精简(留在下次),此处只有server端信息接收处理

server

#include <sys/stat.h>
 
#include <fcntl.h>
 
#include <errno.h>
 
#include <netdb.h>
 
#include <sys/types.h>
 
#include <sys/socket.h>
 
#include <netinet/in.h>
 
#include <arpa/inet.h>
 
#include <stdio.h>
 
#include <string.h>
 
#include <stdlib.h>
 
#include <unistd.h>
 
#define SERVER_PORT 9999
unsigned char IntToHexChar(unsigned char c)

{

if (c > 9)

return (c + 55);

else

return (c + 0x30);

}

int hexToDec (char *s)
{ 
     
     int t,len;
     len = strlen(s);
     long long sum=0;
     int i;
     for(i=0;i<len;i++){
         if(s[i]<='9')
             t=s[i]-'0';
         else
             t=s[i]-'A'+10;
         sum=sum*16+t;
     }
     //printf("%lld",sum);
     return sum;
}


/*
监听后,一直处于accept阻塞状态,
直到有客户端连接,
当客户端如数quit后,断开与客户端的连接
*/
 
int main()
 
{
 
//调用socket函数返回的文件描述符
 
int serverSocket;
 
//声明两个套接字sockaddr_in结构体变量,分别表示客户端和服务器
 
struct sockaddr_in server_addr;
 
struct sockaddr_in clientAddr;
 
int addr_len = sizeof(clientAddr);
 
int client;
 
char buffer[200];
 
int iDataNum;
 
//socket函数,失败返回-1
 
//int socket(int domain, int type, int protocol);
 
//第一个参数表示使用的地址类型,一般都是ipv4,AF_INET
 
//第二个参数表示套接字类型:tcp:面向连接的稳定数据传输SOCK_STREAM
 
//第三个参数设置为0
 
if((serverSocket = socket(AF_INET, SOCK_STREAM, 0)) < 0)
 
{
 
perror("socket");
 
return 1;
 
}
 
bzero(&server_addr, sizeof(server_addr));
 
//初始化服务器端的套接字,并用htons和htonl将端口和地址转成网络字节序
 
server_addr.sin_family = AF_INET;
 
server_addr.sin_port = htons(SERVER_PORT);
 
//ip可是是本服务器的ip,也可以用宏INADDR_ANY代替,代表0.0.0.0,表明所有地址
 
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
 
//对于bind,accept之类的函数,里面套接字参数都是需要强制转换成(struct sockaddr *)
 
//bind三个参数:服务器端的套接字的文件描述符,
 
if(bind(serverSocket, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0)
 
{
 
perror("connect");
 
return 1;
 
}
 
//设置服务器上的socket为监听状态
 
if(listen(serverSocket, 5) < 0)
 
{
 
perror("listen");
 
return 1;
 
}
 
while(1)
 
{
 
printf("listen: %d\n", SERVER_PORT);
 
//调用accept函数后,会进入阻塞状态
 
//accept返回一个套接字的文件描述符,这样服务器端便有两个套接字的文件描述符,
 
//serverSocket和client。
 
//serverSocket仍然继续在监听状态,client则负责接收和发送数据
 
//clientAddr是一个传出参数,accept返回时,传出客户端的地址和端口号
 
//addr_len是一个传入-传出参数,传入的是调用者提供的缓冲区的clientAddr的长度,以避免缓冲区溢出。
 
//传出的是客户端地址结构体的实际长度。
 
//出错返回-1
 
client = accept(serverSocket, (struct sockaddr*)&clientAddr, (socklen_t*)&addr_len);
 
if(client < 0)
 
{
 
perror("accept");
 
continue;
 
}
 
printf("wait...\n");
 
//inet_ntoa ip地址转换函数,将网络字节序IP转换为点分十进制IP
 
//表达式:char *inet_ntoa (struct in_addr);
 
printf("IP is %s\n", inet_ntoa(clientAddr.sin_addr));
 
printf("Port is %d\n", htons(clientAddr.sin_port));
 
while(1)
{
unsigned char msg_recv[5000];   
int n;   
while(1)   
{       
     n = recv(client,msg_recv,sizeof(msg_recv),0);   
     msg_recv[n] = '\0';
     int i; 
     const char mac1[20] = "47626B5714FA";   
    unsigned char data[1024]; 
     //char data1[1024]; 
    unsigned char temp;
    unsigned char  s_des[100] = {0};

for (i=0; i < n; i++) 
{

temp = msg_recv[i]&0xf0;

s_des[2*i] = IntToHexChar(temp >> 4);

temp = msg_recv[i]&0x0f;

s_des[2*i+1] = IntToHexChar(temp);

}
if(strstr(s_des, mac1))
{
    char *p;
    char hertRateHex[2];
    int hertRate;
    char stepNumHex[6];
    int stepNum;
    char temp[6];
    hertRateHex[2] = '\0';
    stepNumHex[6] = '\0';
    temp[6] = '\0';
    p = strstr(s_des, mac1);
    strncpy(hertRateHex, p + 14, 2);
    hertRate = hexToDec(hertRateHex);
    strncpy(stepNumHex, p + 16, 6);
    temp[0] = stepNumHex[4];
    temp[1] = stepNumHex[5];
    temp[2] = stepNumHex[2];
    temp[3] = stepNumHex[3];
    temp[4] = stepNumHex[0];
    temp[5] = stepNumHex[1];
    stepNum = hexToDec(temp);
    
    //sscanf(hertRateHex, "%d", &hertRate);
    printf("Found that the device is nearby!!\n");
    printf("hertRateHex:%s\n",hertRateHex);
    printf("hertRate:%d\n",hertRate);
    printf("stepNumHex:%s\n",temp);
    printf("stepNum:%d\n",stepNum);

}
printf("*** %s ***\n",s_des);

    
       }

if(n < 0)
 
{
 
perror("recv null");
 
continue;
 
}
 
if(strcmp(msg_recv, "quit") == 0)
 
break;
 
printf("msg_recv:%X\n", msg_recv);
 
 
}
 
}
 
close(serverSocket);
 
return 0;
 
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值