基于Webserver的工业数据采集控制小项目

主要用到的知识点,http协议,modbus协议,以及进程间通信,消息队列,共享内存等

框架

数据采集

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <modbus.h>
#include <pthread.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <sys/msg.h>

#define N 1024      //共享内存空间
int ret;            //函数返回值
uint16_t buf[32];   //读取保持寄存器值
uint8_t set[32];    //设置线圈寄存器
modbus_t *sensor;   //传感器实例
modbus_t *hardware; //硬件实例
// char c[2];          //00:led关 01:led 开 10: 蜂鸣器关 11:蜂鸣器开
//消息队列结构体
typedef struct msgbuf
{
    long mytype;  //消息类型
    char buf[32]; //消息数据内容
} msg_t;
int msgid;
void *mythread(void *arg)
{
    while (1)
    {
        //读取消息队列
        msg_t msg_read;                                                                //读取到的消息
        msgrcv(msgid, &msg_read, sizeof(msg_read) - sizeof(long), 0, 0); //接收队列中的第一个消息
        printf("%s\n", msg_read.buf);
        if (msg_read.buf[0] == '0' && msg_read.buf[1] == '0')
        {
            ret = modbus_write_bit(hardware, 0, 0); //关闭LED
            break;
        }
        else if (msg_read.buf[0] == '0' && msg_read.buf[1] == '1')
        {
            ret = modbus_write_bit(hardware, 0, 1); //打开LED
        }
        else if (msg_read.buf[0] == '1' && msg_read.buf[1] == '0')
        {
            ret = modbus_write_bit(hardware, 1, 0); //关闭蜂鸣器
        }
        else if (msg_read.buf[0] == '1' && msg_read.buf[1] == '1')
        {
            ret = modbus_write_bit(hardware, 1, 1); //打开蜂鸣器
        }
    }
    // pthread_exit(NULL);
}
int main(int argc, char const *argv[])
{
    key_t key = ftok("./a.txt", 'a');                        //产生一个key值
    int shmid = shmget(key, N, IPC_CREAT | IPC_EXCL | 0777); //创建或打开共享内存
    if (shmid < 0)
    {
        if (errno == EEXIST)
        {
            printf("shmget eexist\n"); //已创建
            shmid = shmget(key, N, 0777);
        }
        else
        {
            perror("shmget err.");
            return -1;
        }
    }
    //映射共享内存
    char *p = (char *)shmat(shmid, NULL, 0666);
    if (p == (void *)-1)
    {
        perror("shmat err.");
        return -1;
    }
    //创建key值
    key_t key2 = ftok("./a.txt", 'b');
    //创建或打开消息队列
    msgid = msgget(key2, IPC_CREAT | IPC_EXCL | 0666);
    if (msgid < 0)
    {
        if (errno == EEXIST)
        {
            printf("msgget eexist\n"); //已创建
            msgid = msgget(key2, 0666);
        }
        else
        {
            perror("msgget err.");
            return -1;
        }
    }
    // 1.创建实例 modbus_new_tcp,端口号字符型转整型
    //设置传感器读取
    sensor = modbus_new_tcp(argv[1], atoi(argv[2]));
    // 2.设置从机id modbus_set_slave,连接
    ret = modbus_set_slave(sensor, 1);
    if (ret < 0)
    {
        printf("set err\n");
    }
    // 3.建立连接 modbus_connect
    ret = modbus_connect(sensor);
    if (ret < 0)
    {
        printf("connect err.\n");
    }
    //硬件设置
    hardware = modbus_new_tcp(argv[1], atoi(argv[2]));
    ret = modbus_set_slave(hardware, 2);
    if (ret < 0)
    {
        printf("set err\n");
    }
    ret = modbus_connect(hardware);
    if (ret < 0)
    {
        printf("connect err.\n");
    }
    char data[128];
    // 4.寄存器操作
    pthread_t tid; //创建一个线程
    pthread_create(&tid, NULL, mythread, NULL);
    pthread_detach(tid);
    while (1)
    {
        sleep(1);
        //4.从0开始读四个寄存器值. 0:光线传感器 1:加速度传感器 X 2:加速度传感器 Y 3:加速度传感器 Z
        ret = modbus_read_registers(sensor, 0, 4, buf);
        //将从设备读取的内容定向输出到共享内存中
        sprintf(data, "%d\n%d\n%d\n%d\n", buf[0], buf[1], buf[2], buf[3]);
        strcpy(p, data);
        printf("%s", p);
        putchar(10);
    }
    //取消映射
    shmdt(p);
    //删除映射
    shmctl(shmid, IPC_RMID, NULL);
    // 5.关闭套接字 modbus_close,先关闭套接字,再释放实例
    modbus_close(sensor);
    modbus_close(hardware);
    // 6.释放实例 modbus_free
    modbus_free(sensor);
    modbus_free(hardware);
    return 0;
}

服务器端对网页信号进行处理

//处理获取设备数据请求

static int handle_get(int sock, const char *input)

{

    key_t key = ftok("./a.txt", 'a');                        //产生一个key值

    int shmid = shmget(key, N, IPC_CREAT | IPC_EXCL | 0777); //创建或打开共享内存

    if (shmid < 0)

    {

        if (errno == EEXIST)

        {

            printf("shmget eexist\n"); //已创建

            shmid = shmget(key, N, 0777);

        }

        else

        {

            perror("shmget err.");

            return -1;

        }

    }

    //映射共享内存

    char *= (char *)shmat(shmid, NULL, 0666);

    if (== (void *)-1)

    {

        perror("shmat err.");

        return -1;

    }

    char reply_buf[HTML_SIZE] = {0};

    send(sock, p, strlen(p), 0);

    return 0;

}

//处理控制设备数据请求

static int handle_post(int sock, const char *input)

{

    msg_t ctl;

    //创建key值

    key_t key2 = ftok("./a.txt", 'b');

    //创建或打开消息队列

    int msgid = msgget(key2, IPC_CREAT | IPC_EXCL | 0666);

    if (msgid < 0)

    {

        if (errno == EEXIST)

        {

            printf("msgget eexist\n"); //已创建

            msgid = msgget(key2, 0666);

        }

        else

        {

            perror("msgget err.");

            return -1;

        }

    }

    char reply_buf[HTML_SIZE] = {0};

    //分离请求内容为post

    char *post = strstr(input, "post");

    char *= strstr(input, "change=");

    *(- 1) = '\0';

    printf("请求 = %s\n", post);

    //分离控制内容为 *change

    char *change = p + strlen("change=");

    *(change+2)='\0';

    printf("操作change = %s\n", change);

    //设置消息类型为1

    ctl.mytype = 1;

    //将控制信号放入到消息数据中

    strcpy(ctl.buf, change);

    printf("消息内容为:%s\n",ctl.buf);

    //将控制消息发送到消息队列中

    msgsnd(msgid, &ctl, sizeof(ctl) - sizeof(long), 0);

    sprintf(reply_buf, "<script>localStorage.setItem('usr_user_name', '%s');</script>", post);

    strcat(reply_buf, "<script>window.location.href = '/index.html';</script>");

    send(sock, reply_buf, strlen(reply_buf), 0);

    //删除消息队列

    // msgctl(msgid,IPC_RMID,NULL);

    return 0;

}

网页端对页面信号处理

function get() {
            //v不仅仅是一个变量,与数组类似
            var v = document.getElementsByName("light");
            //v[0]表示的是第一个名字为username的标签
            // v[0].value="hello";
            var xhr = new XMLHttpRequest();//新建一个对象
            var url = "";
            xhr.open("post", url, true);
            xhr.onreadystatechange = function ()//检测发生变化
            {
                //表示响应完成并且成功
                if (xhr.readyState === 4 && xhr.status === 200) {
                    var response = xhr.responseText;
                    var x = response.split("\n");//将字符串以'\n'为分割分为一个字符串数组
                    console.log(x);
                    v[0].value = x[0];//响应正文
                    v[1].value = x[1];//响应正文
                    v[2].value = x[2];//响应正文
                    v[3].value = x[3];//响应正文
                }
            }
            xhr.send("get");
        }
        function fun(obj) {
            var xhr = new XMLHttpRequest();
            var url = "";
            xhr.open("POST", url, true);
            console.log(obj);
            if (obj == 'on') {
                console.log("postchange=01");
                xhr.send("postchange=01");
            }
            else if (obj == 'off') {
                console.log("postchange=00");
                xhr.send("postchange=00");
            }
        }
        function funs(obj) {
            var xhr = new XMLHttpRequest();
            var url = ""; // 设置正确的URL
            xhr.open("POST", url, true);
            if (obj == 'on') {
                console.log("postchange=11");
                xhr.send("postchange=11");
            } else if (obj == 'off') {
                console.log("postchange=10");
                xhr.send("postchange=10");
            }
        }

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
随着 科 学 技术的不断发展,嵌入式系统在工业控制和智能家电领域 得到了广泛的应用,同时控制逻辑和功能也变得越来越复杂,相对于早 期嵌入式系统所采用的简单的控制逻辑程序,现代嵌入式系统越来越倾 向于使用完整的操作系统,这就使得嵌入式操作系统在嵌入式系统领域 变得越来越重要. 目前 国 内 外对嵌入式操作系统的研究和开发已广泛展开,已经有了 很多的成熟的产品,嵌入式软件工程师已经从过去复杂的底层程序的编 写中解放出来,更多的是在特定的操作系统上进行开发。 本文 就 是 在这种背景下,将使用Linux操作系统的嵌入式系统用于工 业环境下的远程监控。本文的目标是实现一个内嵌WEB SERVER并且 具有一定数据处理能力的监控设备,可以对被控设备的状态进行采集并 通过WEB SERVER使远程主机可以监控被控设备的状态,同时还可以 实现一些简单的控制。 为了 实 现 这个目标,本文在分析讨论目前国内外嵌入式数据库和 WEB SERVER的发展的基础上,根据系统的需要,选用uClinux操作系 统,并基于uClinux实现了WEB SERVER的移植和开发,数据处理模块 的开发和数据采集模块的开发,在开发过程中解决了WEB SERVER与 数据模块的连接,提高系统的实时性能以及动态WEB的实现等一系列 关键性问题。 本文 从 嵌 入式系统的结构,嵌入式数据库的实现技术和嵌入式WEB SERVER的实现技术等几个方面论述了目标系统相应部分的实现,详细 介绍了实现的原理和方法,并在最后对本文的工作做了总结,展望了后 续工作的发展方向.
回答: WebServer MVC项目是指使用Web服务器作为后端的MVC(Model-View-Controller)模式的项目。在这种项目中,Web服务器充当了后端控制器(Controller)的角色,负责接收来自前端的请求并进行处理。同时,Web服务器还负责将处理结果返回给前端。在这个项目中,通常会使用Spring MVC框架来简化开发过程。Spring MVC是Spring Framework生态中的一个模块,它基于Servlet构建并使用MVC模式设计的Web框架。它的主要目的是简化传统的Servlet+JSP模式下的Web开发方式,并对Java Web中的MVC架构模式进行增强和扩展。在Spring MVC中,前端控制器(DispatcherServlet)负责接收请求并将其分发给后端控制器(Controller),后端控制器负责处理业务逻辑并与模型(Model)和视图(View)进行交互。在视图层,Spring MVC支持多种视图技术,如Freemarker、Velocity、JSP等。因此,WebServer MVC项目使用Spring MVC框架可以更加方便和灵活地开发MVC应用。\[1\] 在WebServer MVC项目中,可以通过获取应用上下文对象来获取Bean。可以通过ServletContextListener监听Web应用的启动,在Web应用启动时加载Spring的配置文件,创建应用上下文对象ApplicationContext,并将其存储在ServletContext域中。这样就可以在任意位置从域中获取应用上下文ApplicationContext对象,实现数据共享。\[2\]\[3\] #### 引用[.reference_title] - *1* *2* *3* [【SpringMVC】集成Web、MVC执行流程、数据响应、数据交互](https://blog.csdn.net/weixin_57535055/article/details/125945430)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

满山的猴子我的腚最红

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值