具体案例:使用树莓派ds18b20温度传感器实现温度上报

  • 首先需要获得传感器文件中保存的温度信息:

  温度信息通常保存在路径为“/sys/bus/w1/devices/28-xxxxxxxxxxxx/w1_slave”的文件中:  (这里的28-xxxxxxxxxxxx为产品序列号因此我们需要在代码中解决序列号不同导致程序复用性低的问题)

  可以通过cat命令显示当前温度信息:

  cat w1_slave

     crc为循环冗余校验码;实际温度即为t/1000即27.625℃

  以下为服服务器端的代码    

1 #include <stdlib.h>
2 #include <sys/types.h>
3 #include <sys/socket.h>
4 #include <errno.h>
5 #include <stdio.h>
6 #include <string.h>
7 #include <netinet/in.h>
8 #include <arpa/inet.h>
9 #include <unistd.h>
10 #include <sys/stat.h>
11 #include <fcntl.h>
12 #include <dirent.h>
13
14
15 #define SERVER_IP "127.0.0.1" //ip地址先设置为本机环回地址,端口随便,先本地测试一下能否正确得到温度
16 #define SERVER_PORT 1234517
18
19 float getTemper(){
20
21 char file_path[128]="/sys/bus/w1/devices/";
22 char buf[128];
23 char *temper_str = NULL;  //用于接收字符串类型的温度数据
24 int file_fd = -1;   //文件描述符
25 float temper = 0;    //接收温度的浮点数
26
27
28 DIR *DIR_str = NULL;
29 struct dirent *dirent_str = NULL;
30
31 DIR_str = opendir(file_path);
32 if(!DIR_str)
33 {
34 printf("directory open failed :%s",strerror(errno));
35 return -1;
36 }
37
38 while(dirent_str = readdir(DIR_str))
39 {
40 if(strstr(dirent_str -> d_name,"28-"))
41 {
42 bzero(&buf,sizeof(buf));
43 strncpy(buf,dirent_str->d_name,sizeof(buf));
44 }
45
46 }
47
48 strncat(file_path,buf,sizeof(file_path));
49 strncat(file_path,"/w1_slave",sizeof(file_path)-1);
50
51 file_fd =open(file_path,O_RDONLY);
52 if(file_fd < 0)
53 {
54 printf("file open failed:%s\n",strerror(errno));
55
56 }
57
58 bzero(&buf,sizeof(buf));
59
60 if(!read(file_fd,buf,sizeof(buf)))
61 {
62 printf("read file failed: %s\n",strerror(errno));
63 }
64
65 temper_str = strstr(buf,"t=");
66 if(temper_str == NULL)
67 {
68 printf("can not find 't=' \n");
69 }
70
71 temper_str += 2;
72 temper = atof(temper_str);
73 printf("acquire right now temperature is:%f\n",temper/1000);
74 return temper/1000;
75 }
    //以上为de18b20获取温度的代码
76
77
78
79
80 int main(int argc,char **argv){
81 int client_fd = -1;
82 int return_value = -1;
83
84 struct sockaddr_in server_addr ;
85 char buf[512] ;
86 char string[20] ;
87 float return_temper = getTemper();
88 sprintf(string,"%f",return_temper);
89
90 bzero(&server_addr,sizeof(server_addr));
91 server_addr.sin_family = AF_INET;
92 server_addr.sin_port = htons(SERVER_PORT);
93 //inet_aton(SERVER_IP,&server_addr.sin_addr);
94 inet_pton(AF_INET,SERVER_IP,&server_addr.sin_addr);
95
96 struct sockaddr *server_address = (struct sockaddr*)&server_addr;
97
98 client_fd=socket(AF_INET,SOCK_STREAM,0);
99 if(client_fd < 0)
100 {
101 printf("socket created failed: %s",strerror(errno));
102 return -1;
103 }
104 printf("socket created successfully\n");
105
106 return_value = connect(client_fd,(struct sockaddr*)&server_addr,sizeof(server_addr));
107 // return_value = connect(client_fd,server_address,sizeof(server_addr));
108
109 if(return_value < 0)
110 {
111 printf("connect failed :%s\n",strerror(errno));
112 return -2;
113 }
114 printf("connect successful\n");
115
116 bzero(&buf,sizeof(buf));
117
118 return_value = write(client_fd,string,strlen(string));
119
120 read(client_fd,string,strlen(string));
121
122 printf("%s\n",string);
123
124 close (client_fd);
125
126 return 0;
127 }

*DIR 结构体

函数   DIR *opendir(const char *path)

打开文件目录,返回指向DIR结构体的指针

*dirent结构体

定义:

struct dirent
{
  long d_ino;             /* inode number 索引节点号 */
  off_t d_off;              /* offset to this dirent 在目录文件中的偏移 */
  unsigned short d_reclen;       /* length of this d_name 文件名长 */
  unsigned char d_type;        /* the type of d_name 文件类型 */
  char d_name [NAME_MAX+1];    /* file name (null-terminated) 文件名,最长255字符 */                                 
}

它指向目录以及目录中的所有文件,其中搭配使用的    readdir( )函数:       成功则返回下个目录进入点. 有错误发生或读取到目录文件尾则返回NULL.

简单来说就是DIR结构体通过opendir打开目录(文件夹),通过dirent索引可以获得需要的文件名

不知到服务器的ip地址,但是知道域名,可以通过域名解析获取服务器ip地址

方法一:

  • 添加头文件并注释掉测试地址
#include <netdb.h>
//#define SERVER_IP             "127.0.0.1"
  • 在main函数增加hosten结构体
 struct  hostent         *host = gethostbyname(serverName);
  • 将注释掉的ip地址重新声明
char                     *SERVER_IP       =NULL;
  • 一个域名可能对应多个ip
for(int i = 0;host->h_addr_list[i];i++){

                SERVER_IP = inet_ntoa(*(struct in_addr*)host->h_addr_list[i]);
}

inet_ntoa()函数             将网络地址转换成“.”点隔的字符串格式

struct hostent
{
  char *h_name;        //正式主机名
  char **h_aliases;       //主机别名
  int h_addrtype;        //主机IP地址类型:IPV4-AF_INET
  int h_length;            //主机IP地址字节长度,对于IPv4是四字节,即32位
  char **h_addr_list;       //主机的IP地址列表     二级指针,哈人
};

以下为服务器端代码:

#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <sys/types.h> #define LISTEN_PORT 1299
#define BACKLOG 13 int main(int argc,char **argv){ int rv = -1;
int listen_fd,client_fd = -1;
struct sockaddr_in server_addr;
struct sockaddr_in client_addr;
socklen_t client_addr_len;
char buf[1024]; listen_fd =socket(AF_INET,SOCK_STREAM,0);
if(listen_fd<0){ printf("create socket failed :%s\n",strerror(errno));
return -1; }
printf("socket create fd[%d]\n",listen_fd); memset(&server_addr,0,sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port=htons(LISTEN_PORT);
server_addr.sin_addr.s_addr=htonl(INADDR_ANY); if (bind(listen_fd,(struct sockaddr *)&server_addr,sizeof(server_addr))<0){ printf("create socket failed: %s\n",strerror(errno));
return -2; }
printf("socket[%d] bind on port [%d] for all IP address ok\n",listen_fd,LISTEN_PORT); listen(listen_fd,BACKLOG); while(1)
{ printf("\n Start waiting an accept new client connect...\n");
client_fd = accept(listen_fd,(struct sockaddr*)&client_addr,&client_addr_len);
if(client_fd <=0){ printf("accept new socket failure: %s\n", strerror(errno));
return -2;
}
printf("Accept new client[%s:%d] with fd [%d]\n", inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port), client_fd); memset(buf,0,sizeof(buf));
if((rv = read(client_fd,buf,sizeof(buf)))<0){ printf("read data from client socket[%d] failed:%s\n",client_fd,strerror(errno));
close(client_fd);
continue; }
else if(rv ==0){ printf("client socket [%d] disconnected",client_fd);
close(client_fd);
continue; } printf("read %d bytes data from client [%d] and echo it back :'%s'\n",rv,client_fd,buf); if(write(client_fd,buf,rv) < 0){ printf("Write %d bytes data back to client [%d] failed:%s\n",rv,client_fd,strerror(errno));
close(client_fd);
} sleep(1);
close(client_fd);
}
close(listen_fd); }

效果:客户端发过去后接收到服务器发回的数据

服务器端接收到客户端发送的数据并发回相同内容

看得头晕,明天补下注释吧。。。

通过Linux的socket套接字实现客户端与服务器端的通信的更多相关文章

  1. Python进阶----SOCKET套接字基础, 客户端与服务端通信, 执行远端命令.

    Python进阶----SOCKET套接字基础, 客户端与服务端通信, 执行远端命令. 一丶socket套接字 什么是socket套接字: ​ ​  ​ 专业理解: socket是应用层与TCP/IP ...

  2. Linux之socket套接字编程20160704

    介绍套接字之前,我们先看一下传输层的协议TCP与UDP: TCP协议与UDP协议的区别 首先咱们弄清楚,TCP协议和UCP协议与TCP/IP协议的联系,很多人犯糊涂了,一直都是说TCP/IP协议与UD ...

  3. 【网络编程】Socket套接字网络编程模型

    一.Linux网络模型 -- Socket套接字编程 图片:Socket 抽象层 Socket编程--不同协议,统一接口 Socket的实质就是一个接口, 利用该接口,用户在使用不同的网络协议时,操作 ...

  4. linux网络环境下socket套接字编程(UDP文件传输)

    今天我们来介绍一下在linux网络环境下使用socket套接字实现两个进程下文件的上传,下载,和退出操作! 在socket套接字编程中,我们当然可以基于TCP的传输协议来进行传输,但是在文件的传输中, ...

  5. linux网络编程-(socket套接字编程UDP传输)

    今天我们来介绍一下在linux网络环境下使用socket套接字实现两个进程下文件的上传,下载,和退出操作! 在socket套接字编程中,我们当然可以基于TCP的传输协议来进行传输,但是在文件的传输中, ...

  6. 进程、线程、socket套接字-资源大小 & 切换代价

    另,进程 & 线程的很多知识可以看这里:http://www.cnblogs.com/charlesblc/p/6135666.html 今天查了一下进程和线程的资源占用. 比如问:栈和堆的大 ...

  7. 进程间通信系列 之 socket套接字及其实例

    进程间通信系列 之 概述与对比   http://blog.csdn.net/younger_china/article/details/15808685  进程间通信系列 之 共享内存及其实例   ...

  8. Python之异常处理和socket套接字连接7

    一.异常处理 1)异常处理的使用意义 什么是异常处理 异常是程序发生错误的信号,即程序一旦出错就会立刻产生一个异常,如果该异常没有被处理 那么异常就抛出来,程序的运行也随之终止 异常分为三部分: 异常 ...

  9. 19 网络编程--Socket 套接字方法

    1.Socket(也称套接字)介绍 socket这个东东干的事情,就是帮你把tcp/ip协议层的各种数据封装啦.数据发送.接收等通过代码已经给你封装好了 ,你只需要调用几行代码,就可以给别的机器发消息 ...

  10. 网络编程基础之Socket套接字简单应用

    一.Socket套接字实现通信循环 所谓通信循环,简单理解就是客户端可以给服务端循环发送信息并获得反馈的过程. 1.基础版 通信循环的程序分为两部分,即两个python模块,分别为客户端.py和服务端 ...

随机推荐

  1. 2022 icpc 沈阳站 记录(非题解)

    赛前 大概是赛前三周才突然知道拥有了比赛机会. 赛前训练和vp频率很高,有一段时间cf上都是绿的.比赛的那一周只有一天没在vp,到了周六热身赛我人都有点麻木.(可能正赛也是类似的状态吧) 比赛的过程b ...

  2. SpringMVC文件上传详解

    声明 源码基于Spring Boot 2.3.12.RELEASE.Spring Framework 5.2.15.RELEASE Servlet3.0 文件上传 Servlet 3.0对于HttpS ...

  3. 百题计划-3 codeforces 657div2 C. Choosing flowers 贪心+枚举+二分

    https://codeforces.com/contest/1379/problem/C 题意: 给m种花(a,b),从中取出n朵,每种花可以取0和无限朵,如果取出第i朵花的个数为c>0,则贡 ...

  4. vue html转pdf并打印

    //文件名随便取一个如:htmlToPdf.js // 导出页面为PDF格式 import html2Canvas from 'html2canvas' import JsPDF from 'jspd ...

  5. AVL tree rotate

    AVL tree single rotate /** * Rotate binary tree node with left child. * For AVL trees, this is a sin ...

  6. 嵌入式开发er的C语言能力自测(面试)题---top 16

    准备面试刷到的,链接里是原文和答案: (a-c-test-the-0x10-best-questions-for-would-be-embedded-programmers) 这里我先只给出问题,可以 ...

  7. shopt 内置命令启用shell选项 (extglob)

    使用shopt 内置命令启用shell选项 (extglob) 则会识别几个扩展模式匹配运算符 模式列表是由 | 分割 查看shell选项 extglob shopt |grep extglob 启动 ...

  8. kubectl的vistor模式

    package main import ( "encoding/json" "encoding/xml" "log" ) type Visi ...

  9. ubuntu启动盘制作

    转自https://www.cnblogs.com/silentdoer/p/13044305.html 1. 从Ubuntu官网http://cn.ubuntu.com/download/下载系统的 ...

  10. Mongodb设置账号密码登录

    Mongodb设置.首先设置Data目类和Log目录,然后新建mongodb.conf,设置内容大体如下 port = 27017 #数据目录 dbpath = /usr/softs/data/db ...