EMQX webhook实现转发消息到java web服务器并保存到MySQL数据库,物联网嵌入式开发开发最佳实践手册全网独一份

123 阅读6分钟

收集整理了一份《2024年最新物联网嵌入式全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升的朋友。 img img

如果你需要这些资料,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.io.PrintWriter; import java.io.Reader; import java.io.StringWriter; import java.io.Writer;

import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;

import org.apache.tomcat.util.http.fileupload.IOUtils;

import com.mysql.jdbc.StringUtils; import com.oracle.webservices.internal.api.message.ContentType; import com.sun.xml.internal.bind.CycleRecoverable.Context; import com.sun.xml.internal.ws.wsdl.writer.document.Service;

import net.sf.json.JSONArray; import net.sf.json.JSONObject;

public class FirstServlet extends HttpServlet { //用来读取post过来的json的缓存区的数据长度 private static final int BUFFER_SIZE = 1024 * 8; //一些emqx post过来的数据 private String app_id,device_id,remark,time,state,type;

/\*\*

* 不知道是什么,反正是必须的 */ private static final long serialVersionUID = 1L;

//用来处理get消息
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
	//doPost(request,response);
	//get请求用来获取数据库的数据
	
}

	//用来处理post消息
@Override

protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //获取post过来的输入流 InputStream in=request.getInputStream(); //创建一个缓存读取器来暂时存贮输入流里的数据 BufferedReader reader = new BufferedReader(new InputStreamReader(in));

	//body是json字符串,要解析字符串,拿到对应的值插入数据库
	//读取缓存读取器里的body数据并转为字符串格式,这里的body数据为json字符串格式
	String body = read(reader);
	//从json字符串里获取json对象
	JSONObject J=JSONObject.fromObject(body);
	
	//通过键值对的形式,获取json里的值并赋值给变量
		app_id= J.getString("app\_id");
	    
		device_id=J.getString("device\_id");
		time=J.getString("mytime");
		state=J.getString("state");
		type=J.getString("device\_type");
		remark=J.getString("remark");
		
	//把变量的值保存到数据库
	DBUutil.update(app_id,device_id,remark,time,state,type);
	

}

public static String read(Reader reader) throws IOException
{
    StringWriter writer = new StringWriter();
    try
    {
        write(reader, writer);
        return writer.getBuffer().toString();
    }
    finally{ writer.close(); }
}

public static long write(Reader reader, Writer writer) throws IOException
{
    return write(reader, writer, BUFFER_SIZE);
}
//把缓存器的json数据写入缓存区
public static long write(Reader reader, Writer writer, int bufferSize) throws IOException
{
    int read;
    long total = 0;
    char[] buf = new char[BUFFER_SIZE];
    while( ( read = reader.read(buf) ) != -1 )
    {
        writer.write(buf, 0, read);
        total += read;
    }
    return total;
}

}




> 

> **再新建一个java 类,dbutil,这个类的作用是对数据库进行操作,这里只写了对数据库进行插入数据的操作,其他操作如更新,删除,查询都可实现**

> 

> 

> 




package myweb;

import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.util.ArrayList; import java.util.Iterator; import java.util.List;

import com.mysql.jdbc.Connection; import com.mysql.jdbc.log.Log;

public class DBUutil { static List ESPlist=new ArrayList<>();//����豸������ private static ESP8266 Device=new ESP8266();//��ʼ������

private static String driver = "com.mysql.jdbc.Driver";// MySql驱动
private static String user = "app";// MySQL的用户名和密码
private static String password = "123456";

//连接数据库的方法 private static Connection getConn(String dbName){ Connection connection; connection = null; try{ Class.forName(driver);//加载驱动,需要驱动才能对数据库进行操作 String ip = "118.31.20.121";//数据库的ip地址ַ

        //连接数据库,驱动+ip地址+端口号+用户名+密码,端口号默认是3306
        connection = (Connection) DriverManager.getConnection("jdbc:mysql://" + ip + ":3306/" + dbName,
                user, password);
        

    }catch (Exception e){
        e.printStackTrace();
    
    }
    	//返回一个connection对象
    return connection;
}
//这个是添加设备到数据库,不用看
public static  void bind\_id(String app_id,String device_id){
	
         Connection connection=getConn("MQTTDATA");
    String sql="INSERT INTO user\_bind\_devices (app\_id,device\_id) VALUES (?,?)";
    if (connection!=null){
        try {
            PreparedStatement ps=connection.prepareStatement(sql);
            if (ps!=null){
                ps.setString(1,app_id);
                ps.setString(2,device_id);
                //执行语句,注意!!!如果你的SQL 语句是诸如update,insert的更新语句,应该用statement的execute()方法
                // select用的是statement的executeQuery()
                ps.execute();
                
                connection.close();
                ps.close();
            }
        } catch (SQLException e) {
            e.printStackTrace();
            
        }
    }
}
//这个是把数据保存到MQTTDATA库的 current表格
//我需要插入app\_id,device\_id。。。。。。
public static void update(String app_id,String device_id,String remark,String time,String state,String device_type) {
	//先跟MySQL数据库里的MQTTDATA库建立连接
	Connection connection=getConn("MQTTDATA");
	//定义一个语句,这个语句的功能是对current表格的app\_id,device\_id,remark,mytime,state,device\_type列分别插入我们的参数的值
	//这里的?可以看成一个傀儡,用ps.setString()方法可以将?替换成我们的参数的值
	String sql="INSERT INTO current (app\_id,device\_id,remark,mytime,state,device\_type) VALUES (?,?,?,?,?,?)";
	if	(connection!=null) {
		try {
			//准备我们的mysql操作语句
			PreparedStatement ps=connection.prepareStatement(sql);
			//把第一个?替换成参数里的app\_id,第二个?替换成device\_id........
			if(ps!=null) {
				ps.setString(1, app_id);
				ps.setString(2, device_id);
				ps.setString(3, remark);
				ps.setString(4, time);
				ps.setString(5, state);
				ps.setString(6, device_type);
				ps.execute();
				connection.close();
				ps.close();
			}
		} catch (SQLException e) {
			
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

}


**前面说了web.xml是web服务器的配置文件,这里我们需要在web.xml里对myhttpservlet进行配置**


在web.xml添加内容如下: (这里需要注意一下,app,名称可以随便取,主要是为了程序员方便查找。)


< servlet-class >myweb.myhttpservlet</ servlet-class > 前面是你的项目名称 .后面是我们刚刚写的myhttpservlet类。


< servlet-mapping >是把名称为app的httpservlet类映射到url的一个路径,/first的意思就是当你在浏览器输入:




> 

> **http://你的web服务器ip地址:端口号/first 就会执行myhttpservlet类里的内容。**

> 

> 

> 




myweb1 index.html index.htm index.jsp default.html default.htm default.jsp app myweb.myhttpservlet app /first

### 三,部署web服务器到主机上


**至此,我们一个简陋的web服务器差不多搭建好了,这时候我们就需要把web程序部署到主机的tomcat上。**


本地部署:右击项目,选择export > war file browse保存到你想保存的位置,系统会生成一个.war文件,我们把这个文件放进 apache-tomcat-8.5.45(你的tomcat文件夹)下的webapps文件夹下。


**压缩文件已上传百度云,仅供参考**  
 链接:<https://pan.baidu.com/s/1Klr_gDvmyFwwtqq3zUowww>  
 提取码:spf3


**远程云主机部署:用xftp将war文件放入同样的路径下,然后重新启动tomcat。**


tomcat在运行时会自动解压war文件生成web程序并开始执行  
 ![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/3b2c362ef5a045a8afbfc14a39f3ba49~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=n%2FiIxyVOI2xrdvwjrXY%2FkXHsgW8%3D)  
 ![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/03732dddbd214f329e6c7c24f83fb561~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=WNMg%2FbiGkDBB3pv3yYNKgdKTdHk%3D)


### 四,配置EMQX服务器


**配置的目的:让 EMQX 服务器post数据 给web服务器;**


#### 4.1 设置规则引擎


**登录emqx远程控制台,在规则引擎新建规则,规则引擎会对mqtt消息进行筛选,具体的筛选规则由我们来定义,下图是一些规则示例**  
 ![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/a86ff93e821b456d9cf8c6eb597884b9~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=nlaQGAuvJJgQ5taAkNHJ2%2BxD%2FAU%3D)  
 以下图的例子分析:


`SELECT` 下方的语句含义是 :**在mqtt消息的数据段(payload)中提取出remark,app\_id,device\_id变量的值 复制 到本地的remark,app\_id,device\_id变量中。**


`WHERE` 语句表示 从发往主题为`csdn`的mqtt消息中筛选。


随后我们用post把数据发送给web服务器时,数据的格式设置为**json格式{“remark”:“杀手”,“app\_id”:“37264726374”}**  
 ![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/e4d8b705a21a4b2f97a27b381b7f84f6~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=UjLygfljXt3SwdRVFc91hCsZ0fM%3D)


#### 4.2 测试规则引擎


我们还可以进行在线测试,测试输出的json数据是不是我们需要的内容。


**下图配置mqtt消息:topic设置为csdn,payload字段设置变量及其取值。**  
 ![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/c3eaf4671fc2413cabd4a73caef096a5~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=eP7ILQXZL89UqRcRkkkm2y%2FyKGU%3D)


**点击测试,输出如图的json数据:**  
 ![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/ca601e4354ff4c6c9474467843fd6e57~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=eEYW%2B7WdijNDWKl8uHC5wf5XECU%3D)


#### 4.3 设置响应动作


**当规则引擎筛选出指定的mqtt消息并提取出json数据后,还需要将数据post到我们的web服务器,这需要设置响应动作来完成这最后一步。**



> 
> 点击添加,动作选择发送数据到web服务,点击新建资源
> 
> 
> 


![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/1d4d1ab84ec34183915f145e6c252509~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=NolmPinc1B6ffvy%2FrB9bFEvCTUs%3D)



> 
> 在资源管理中设置 请求url 为web服务器地址,端口号(tomcat默认端口为8888,这里我改成8090),加上前面我们web.xml里设置的/first ; http请求就会发送到我们的web程序。
> 
> 
> 


![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/4259996c1bbe41bd97462afce67c276b~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=4ZjUrqgtZJccZExvLDa6KQb7qXE%3D)  
 最后,我们需要进行测试,进入websocket,连接,发布消息到指定主题  
 ![在这里插入图片描述](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/67f19e26b8224c2f976775be3a8daab2~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=pTfH3jDy8qOj585YLAkR6YPdojc%3D)  
 然后到你的数据库查看是否存贮到数据。


有疑问的同学可以留言评论。  


**收集整理了一份《2024年最新物联网嵌入式全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升的朋友。**
![img](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/09b613a0ca204a708f0d8cd3a4cfca07~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=C%2ByJvqzJbnH8JBa5hoLY%2F2pOkXQ%3D)
![img](https://p3-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/1472f4601a304da7b93d569c668cd0cc~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg55So5oi3MzM5MTQ5MjgwNjA=:q75.awebp?rk3s=f64ab15b&x-expires=1771924655&x-signature=T6SkTieoMEZmOoPei%2BtbHn0VDoI%3D)

**[如果你需要这些资料,可以戳这里获取](https://gitee.com/vip204888)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人**

**都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**