@[toc]
1.背景开发一个app与后台数据库交互,基于MysqL+jdbc+tomcat,没有使用dbutils或jdbc框架,纯粹底层jdbc实现.
以后逐步改用Spring框架,优化MysqL,进一步部署tomcat等等,现在项目刚刚起步,还有很多不懂的东西,得慢慢来......
这几天踩了很多坑,说得夸张点真是踩到我没有知觉,希望能帮助别人少踩坑...
mysqlV8.0.17驱动(注意这个要与自己的MysqL版本对应)
java-servlet-api-V4.0.1
其他版本可以来这里搜索下载
Maven仓库
2.github
这是源码地址,包括前后端与建表等所有代码.
(欢迎star)
这也是源码地址4.配置开发环境
IDE就不说了,重点说一下MysqL与tomcat9的安装
一. 安装MysqL8.0.17这个是目前比较新的MysqL版本.
服务器系统是centos
其他系统安装看这里
centos使用yum命令安装(参考链接)
(1) 下载MysqLsudo yum localinstall https://repo.MysqL.com//MysqL80-community-release-el7-1.noarch.rpm
(2) 安装MysqLsudo yum install MysqL-community-server
(3) 启动服务sudo service MysqLd start
(4) 查看初始化密码,用于下一步设置自己的root密码sudo grep 'temporary password' /var/log/MysqLd.log
(5) 本地使用root登录MysqL -u root -p
输入上一步看到的密码
(6) 更改密码alter MysqL.user 'root'@'localhost' IDentifIEd by 'password';
注意新版本的MysqL不能使用太弱的密码
如果出现如下提示
use MysqL;update user set host='%' where user='root';
这个可以根据自己的需要去修改,host='%'表明允许所有的ip登录,也可以设置特定的ip,若使用host='%'的话建议新建一个用户配置相应的权限.
(8) 配置防火墙(可选)由于作者使用的是阿里云的服务器,没配置防火墙的话远程连接不上,因此需要手动配置,如图
其中授权对象可以根据自己的需要更改,0.0.0.0/0表示允许所有的ip.
<br><br>
二.安装tomcat9(1) 先去官网下载,下载后上传文件到服务器作者使用的是scp命令,不会的可以看这里
scp apache-tomcat-xxxx.tar.gz username@xx.xx.xx.xx:/
改成自己的用户名和ip
(2) 连接到服务器,解压压缩包mkdir /usr/local/tomcatmv apache-tomcat-xxxx.tar.gz /usr/local/tomcattar -xzvf apache-tomcat-xxx.tar.gz
(3) 修改tomcat默认端口(可选)修改conf/server.xml文件,一般只需修改
<Connector port="8080" protocol="http/1.1" connectionTimeout="20000" redirectPort="8443" />
中的8080端口,修改这个端口即可
这个懒的话(比如作者)可以不改
运行bin目录下的startup.sh
cd bin./startup.sh
(5) 测试浏览器输入
服务器IP:端口
若出现
则表示成功.(6)开机启动
建议配置开机启动,修改/etc/rc.local文件
vim /etc/rc.local添加sh /usr/local/tomcat/bin/startup.sh
这个根据自己的tomcat安装路径修改,指定bin下的startup.sh即可
5.建库建表创建用户表,这里简化 *** 作(好吧我喜欢偷懒)就不创建新用户不授权了
这是一个在本地用root登录的示例,请根据实际情况创建并授权用户.
CREATE DATABASE userinfo;USE userinfo;CREATE table user( ID INT NOT NulL PRIMARY KEY auto_INCREMENT, name CHAR(30) NulL, password CHAR(30) NulL);
(2) 导入到数据库MysqL -u root -p < user.sql
6.后端部分(1) 创建项目选择web application
选好路径,改好名字后finish
(2) 添加jar包
创建一个叫lib的目录
添加两个jar包:
mysql-connector-java-8.0.17.jar
javax.servlet-API-4.0.1.jar
打开Project Structure
Modules--> + --> JARs or directorIEs
选择刚才新建的lib下的两个jar包
打勾,apply
(3) 创建包与类
总共4个包
com.servlet用于处理来自前端的请求,包含SignUp.java,SignIn.javacom.util
主要功能是数据库连接,包含dbutils.javacom.entity
用户类,包含User.javacom.dao
*** 作用户类的类,包含UserDao.java
(4) 先来处理dbutils类
这个是连接数据库的类,纯粹的底层jdbc实现,注意驱动版本.
package com.util;import java.sql.*;public class dbutils { private static Connection connection = null; public static Connection getConnection() { try { Class.forname("com.MysqL.cj.jdbc.Driver"); String url = "jdbc:MysqL://127.0.0.1:3306/数据库名字"; String usename = "账号"; String password = "密码"; connection = DriverManager.getConnection(url,usename,password); } catch (Exception e) { e.printstacktrace(); return null; } return connection; } public static voID closeConnection() { if(connection != null) { try { connection.close(); } catch (sqlException e) { e.printstacktrace(); } } }}
主要就是获取连接与关闭连接两个函数.
String url = "jdbc:MysqL://127.0.0.1:3306/数据库名字";String usename = "账号";String password = "密码";
这几行根据自己的用户名,密码,服务器ip和库名修改
注意,MysqL8.0以上使用的注册驱动的语句是
Class.forname("com.MysqL.cj.jdbc.Driver");
旧版的是
Class.forname("com.MysqL.jdbc.Driver");
注意对应.
(5) 接下来处理User类User类比较简单,就是就三个字段与getter,setter
package com.entity;public class User { private int ID; private String name; private String password; public int getID() { return ID; } public voID setID(int ID) { this.ID = ID; } public String getname() { return name; } public voID setname(String name) { this.name = name; } public String getpassword() { return password; } public voID setPassword(String password) { this.password = password; }}
(6) 接下来是UserDaopackage com.dao;import com.entity.User;import com.util.dbutils;import java.sql.Connection;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.sql.sqlException;public class UserDao { public boolean query(User user) { Connection connection = dbutils.getConnection(); String sql = "select * from user where name = ? and password = ?"; try { PreparedStatement preparedStatement = connection.prepareStatement(sql); preparedStatement.setString(1,user.getname()); preparedStatement.setString(2,user.getpassword()); ResultSet resultSet = preparedStatement.executequery(); return resultSet.next(); } catch (sqlException e) { e.printstacktrace(); return false; } finally { dbutils.closeConnection(); } } public boolean add(User user) { Connection connection = dbutils.getConnection(); String sql = "insert into user(name,password) values(?,?)"; try { PreparedStatement preparedStatement = connection.prepareStatement(sql); preparedStatement.setString(1,user.getname()); preparedStatement.setString(2,user.getpassword()); preparedStatement.executeUpdate(); return preparedStatement.getUpdateCount() != 0; } catch (sqlException e) { e.printstacktrace(); return false; } finally { dbutils.closeConnection(); } }}
主要就是查询与添加 *** 作,查询 *** 作中存在该用户就返回true,否则返回false
添加 *** 作中使用executeUpdate()与getUpdateCount() != 0.注意不能直接使用
return preparedStatement.execute();
去代替
preparedStatement.executeUpdate();return preparedStatement.getUpdateCount() != 0;
咋一看好像没有什么问题,那天晚上我测试的时候问题可大了,androID那边显示注册失败,但是数据库这边的却insert进去了.........我......
好吧说多了都是泪,还是函数用得不够熟练.<br><br>
executequery()返回ResultSet,表示结果集,保存了select语句的执行结果,配合next()使用delete,insert,update使用executeUpdate()
executeUpdate()返回的是一个整数,表示受影响的行数,即delete,insert,update修改的行数,对于drop,create *** 作返回0create,drop使用execute()
execute()的返回值是这样的:如果第一个结果是ResultSet对象,则返回true如果第一个结果是更新计数或者没有结果则返回false
所以在这个例子中
return preparedStatement.execute();
肯定返回false,所以才会数据库这边insert进去,但前端显示注册失败(这个BUG作者找了很久......)
(7) servlet包的SignIn与SignUp类SingIn类用于处理登录,调用jdbc查看数据库是否有对应的用户
SignUp类用于处理注册,把user添加到数据库中
这两个使用的是http连接,后期作者会采用https加密连接.
SignIn.java
package com.servlet;import com.dao.UserDao;import com.entity.User;import javax.servlet.servletexception;import javax.servlet.annotation.WebServlet;import javax.servlet.http.httpServlet;import javax.servlet.http.httpServletRequest;import javax.servlet.http.httpServletResponse;import java.io.IOException;@WebServlet("/SignIn")public class SingIn extends httpServlet { @OverrIDe protected voID doGet(httpServletRequest httpServletRequest, httpServletResponse httpServletResponse) throws IOException,servletexception { this.doPost(httpServletRequest,httpServletResponse); } @OverrIDe protected voID doPost(httpServletRequest httpServletRequest,httpServletResponse httpServletResponse) throws IOException, servletexception { httpServletRequest.setCharacterEnCoding("utf-8"); httpServletResponse.setCharacterEnCoding("utf-8"); httpServletResponse.setContentType("text/plain;charset=utf-8");//设置相应类型为HTML,编码为utf-8 String name = httpServletRequest.getParameter("name"); String password = httpServletRequest.getParameter("password"); UserDao userDao = new UserDao(); User user = new User(); user.setname(name); user.setPassword(password); if(!userDao.query(user))//若查询失败 { httpServletResponse.sendError(204,"query Failed.");//设置204错误码与出错信息 } }}
@WebServlet("/SignIn")
这行代码表示这是一个名字叫SignIn的servlet,可用于实现servlet与url的映射,如果不在这里添加这个注解,则需要在WEB-INF目录下的web.xml添加一个
<servlet-mapping>
叫servlet的映射
httpServletResponse.setContentType("text/plain;charset=utf-8");//设置相应类型为HTML,编码为utf-8
这行代码设置响应类型与编码
String name = httpServletRequest.getParameter("name");String password = httpServletRequest.getParameter("password");
httpServletRequest.getParameter(String name)方法表示根据name获取相应的参数
下面是SignUp.java
package com.servlet;import com.dao.UserDao;import com.entity.User;import javax.servlet.annotation.*;import javax.servlet.http.*;import javax.servlet.*;import java.io.IOException;@WebServlet("/SignUp")public class SignUp extends httpServlet { @OverrIDe protected voID doGet(httpServletRequest httpServletRequest,httpServletResponse httpServletResponse) throws IOException,servletexception { this.doPost(httpServletRequest,httpServletResponse); } @OverrIDe protected voID doPost(httpServletRequest httpServletRequest,httpServletResponse httpServletResponse) throws IOException,servletexception { httpServletRequest.setCharacterEnCoding("utf-8"); httpServletResponse.setCharacterEnCoding("utf-8");//设定编码防止中文乱码 httpServletResponse.setContentType("text/plain;charset=utf-8");//设置相应类型为HTML,编码为utf-8 String name = httpServletRequest.getParameter("name");//根据name获取参数 String password = httpServletRequest.getParameter("password");//根据password获取参数 UserDao userDao = new UserDao(); User user = new User(); user.setname(name); user.setPassword(password); if(!userDao.add(user)) //若添加失败 { httpServletResponse.sendError(204,"add Failed.");//设置204错误码与出错信息 } }}
(8) 添加servlet到web.xml<?xml version="1.0" enCoding="UTF-8"?><web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd" version="4.0"> <servlet> <servlet-name>SignIn</servlet-name> <servlet-class>com.servlet.SingIn</servlet-class> </servlet> <servlet> <servlet-name>SignUp</servlet-name> <servlet-class>com.servlet.SignUp</servlet-class> </servlet></web-app>
要把刚才创建的Servlet添加进web.xml,在<servlet>中添加子元素<servlet-name>与<servlet-class>
<servlet-name>是Servlet的名字,最好与类名一致.
<servlet-class>是Servlet类的位置.
如果在Servlet类中没有添加
@WebServlet("/xxxx")
这个注解,则需要在web.xml中添加
<servlet-mapping> <servlet-name>SignIn</servlet-name> <url-pattern>/SignIn</url-pattern></servlet-mapping>
其中<servlet-name>与<servlet>中的子元素<servlet-name>中的值一致
<url-pattern>是访问的路径
<!DOCTYPE HTML> <head> <Meta charset="utf-8"> <Title>Welcome</Title> </head> <body> Hello web. </body></HTML>
<br><br><br>
7.打包发布作者用的是IDEA,Eclipse的打包请看这里
(1) 打开project structure(2) 选择Artifacts,Web Application:Archive(3) 改名字,创建WEB-INF目录与子目录classes(4) 选中classes,添加Module Output,选择自己的web项目(5) 添加jar包,选中lib目录后添加jar包文件(那个lib文件夹被挡住了.....)
web.xml这个需要在WEB-INF目录里,Hello.HTML在WEB-INF外面
@H_419_523@
(8) 上传到服务器
把打包好的.war文件上传到服务器的tomcat的/webapps目录下的
scp ***.war username@xxx.xxx.xxx.xxx:/usr/local/tomcat/webapps
注意改成自己的webapps目录.
(9) 测试在浏览器输入
服务器IP:端口/项目/Hello.HTML
作者是在本地上开了tomcat后测试的
(2) MainActivity.java
package com.cx;import androID.os.Bundle;import androID.vIEw.VIEw;import androID.Widget.button;import androID.Widget.EditText;import androID.Widget.Toast;import androIDx.appcompat.app.AppCompatActivity;public class MainActivity extends AppCompatActivity { @OverrIDe protected voID onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentVIEw(R.layout.activity_main); button signin = (button) findVIEwByID(R.ID.signin); signin.setonClickListener(new VIEw.OnClickListener() { @OverrIDe public voID onClick(VIEw vIEw) { String name = ((EditText) findVIEwByID(R.ID.etname)).getText().toString(); String password = ((EditText) findVIEwByID(R.ID.etpassword)).getText().toString(); if (UserService.signIn(name, password)) runOnUiThread(new Runnable() { @OverrIDe public voID run() { Toast.makeText(MainActivity.this, "登录成功", Toast.LENGTH_SHORT).show(); } }); else { runOnUiThread(new Runnable() { @OverrIDe public voID run() { Toast.makeText(MainActivity.this, "登录失败", Toast.LENGTH_SHORT).show(); } }); } } }); button signup = (button) findVIEwByID(R.ID.signup); signup.setonClickListener(new VIEw.OnClickListener() { @OverrIDe public voID onClick(VIEw vIEw) { String name = ((EditText) findVIEwByID(R.ID.etname)).getText().toString(); String password = ((EditText) findVIEwByID(R.ID.etpassword)).getText().toString(); if (UserService.signUp(name, password)) runOnUiThread(new Runnable() { @OverrIDe public voID run() { Toast.makeText(MainActivity.this, "注册成功", Toast.LENGTH_SHORT).show(); } }); else { runOnUiThread(new Runnable() { @OverrIDe public voID run() { Toast.makeText(MainActivity.this, "注册失败", Toast.LENGTH_SHORT).show(); } }); } } }); }}
没什么好说的,就为两个button绑定事件,然后设置两个Toast提示信息.
(3) UserService.javapackage com.cx;import java.io.OutputStream;import java.net.httpURLConnection;import java.net.URL;import java.net.URLEncoder;public class UserService { public static boolean signIn(String name, String password) { MyThread myThread = new MyThread("http://本机IP:8080/cx/SignIn",name,password); try { myThread.start(); myThread.join(); } catch (InterruptedException e) { e.printstacktrace(); } return myThread.getResult(); } public static boolean signUp(String name, String password) { MyThread myThread = new MyThread("http://本机IP:8080/cx/SignUp",name,password); try { myThread.start(); myThread.join(); } catch (InterruptedException e) { e.printstacktrace(); } return myThread.getResult(); }}class MyThread extends Thread{ private String path; private String name; private String password; private boolean result = false; public MyThread(String path,String name,String password) { this.path = path; this.name = name; this.password = password; } @OverrIDe public voID run() { try { URL url = new URL(path); httpURLConnection httpURLConnection = (httpURLConnection) url.openConnection(); httpURLConnection.setConnectTimeout(8000);//设置连接超时时间 httpURLConnection.setReadTimeout(8000);//设置读取超时时间 httpURLConnection.setRequestMethod("POST");//设置请求方法,post String data = "name=" + URLEncoder.encode(name, "utf-8") + "&password=" + URLEncoder.encode(password, "utf-8");//设置数据 httpURLConnection.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");//设置响应类型 httpURLConnection.setRequestProperty("Content-Length", data.length() + "");//设置内容长度 httpURLConnection.setDoOutput(true);//允许输出 OutputStream outputStream = httpURLConnection.getoutputStream(); outputStream.write(data.getBytes("utf-8"));//写入数据 result = (httpURLConnection.getResponseCode() == 200); } catch (Exception e) { e.printstacktrace(); } } public boolean getResult() { return result; }}
MyThread myThread = new MyThread("http://本机IP:8080/cx/SignUp",name,password);MyThread myThread = new MyThread("http://本机IP:8080/cx/SignIn",name,password);
这两行换成自己的ip,本地ip的话可以用ipconfig或ifconfig查看,修改了默认端口的话也把端口一起改了.
路径的话就是
端口/web项目名/Servlet名
web项目名是再打成war包时设置的,Servlet名在web.xml中的<servlet>的子元素<servlet-name>设置,与java源码中的@WebServlet()注解中的一致
另外一个要注意的就是线程问题,需要新开一个线程进行http的连接
(4) activity_main.xml前端页面部分很简单,就两个button,用于验证功能.
<?xml version="1.0" enCoding="utf-8"?><linearLayout xmlns:app="http://schemas.androID.com/apk/res-auto" xmlns:androID="http://schemas.androID.com/apk/res/androID" androID:layout_height="match_parent" androID:layout_wIDth="match_parent" androID:orIEntation="vertical" > <linearLayout androID:layout_wIDth="wrap_content" androID:layout_height="wrap_content" androID:orIEntation="horizontal"> <TextVIEw androID:layout_wIDth="wrap_content" androID:layout_height="wrap_content" androID:text="用户名" /> <EditText androID:layout_wIDth="300dp" androID:layout_height="60dp" androID:ID="@+ID/etname" /> </linearLayout> <linearLayout androID:layout_wIDth="wrap_content" androID:layout_height="wrap_content" androID:orIEntation="horizontal"> <TextVIEw androID:layout_wIDth="wrap_content" androID:layout_height="wrap_content" androID:text="密码" /> <EditText androID:layout_wIDth="300dp" androID:layout_height="60dp" androID:ID="@+ID/etpassword" /> </linearLayout> <linearLayout androID:layout_wIDth="wrap_content" androID:layout_height="wrap_content" androID:orIEntation="horizontal"> <button androID:layout_wIDth="120dp" androID:layout_height="60dp" androID:text="注册" androID:ID="@+ID/signup" /> <button androID:layout_wIDth="120dp" androID:layout_height="60dp" androID:text="登录" androID:ID="@+ID/signin" /> </linearLayout></linearLayout>
9.测试(1) 注册测试随便输入用户名与密码
查看数据库
这里没有加密保存,后期会添加加密保存(2) 登录测试
perfect!10.注意事项(1) 数据库的用户名和密码一定要设置正确,要不然会这样提示
这个错误在加载驱动错误时也可能会出现这个错误,因此要确保打成war包时lib目录正确且jar包版本正确.
还有就是由于这个是jdbc的底层实现,注意手写的SQL语句不能错
千万千万别像我这样:
(2) 网络权限问题
这个需要在AndroIDManifest.xml添加网络权限
<uses-permission androID:name="androID.permission.INTERNET" /><uses-permission androID:name="androID.permission.ACCESS_WIFI_STATE" /><uses-permission androID:name="androID.permission.ACCESS_NETWORK_STATE" />
(3) 防火墙问题服务器的话一般会有相应的相应的网页界面配置,比如作者的是阿里云服务器,当然也可以手动配置iptables
修改/etc/sysconfig/iptables
vim /etc/sysconfig/iptables
添加
-A input -m state --state NEW -m tcp -p tcp --dport 8080 -j ACCEPT-A RH-Firewall-1-input -m state --state NEW -m tcp -p tcp --dport 8080 -j ACCEPT
重启iptables
service iptables restart
(4) 使用http注意事项由于从AndroID P开始,Google默认要求使用加密连接,即要使用httpS,所以会禁止使用http连接
使用http连接时会出现以下异常
W/System.err: java.io.IOException: Cleartext http traffic to **** not permitted java.net.UnkNownServiceException: CLEARTEXT communication ** not permitted by network security policy
两种建议:
1 使用httpS2 修改默认的AndroIDManifest.xml使其允许http连接在res下新建一个文件夹xml,创建一个叫network_security_config.xml的文件,文件内容如下
<?xml version="1.0" enCoding="utf-8"?><network-security-config> <base-config cleartextTrafficPermitted="true" /></network-security-config>
然后在AndroIDMainfest.xml中加入
<application androID:networkSecurityConfig="@xml/network_security_config"/>
即可
另一种办法是直接加入一句
androID:usesCleartextTraffic="true"
<application androID:usesCleartextTraffic="true"/>
(5) 线程问题从androID4.0开始,联网不能再主线程 *** 作,万一网络不好就会卡死,所以有关联网的 *** 作都需要新开一个线程,不能在主线程 *** 作.
(6) AVD问题这个BUG作者找了很久,http连接没问题,服务器没问题,数据库没问题,前端代码没问题,然后去了stackoverflow,发现是AVD的问题,我.......
简单来说就是卸载了再重启AVD,居然成功了.....11 最后
作者小白一枚,有什么不对的地方请大家指正,评论作者会好好回复的.
以下是我的CSDN博客
CSDN
参考网站
1.Android 通过Web服务器与Mysql数据库交互
2.Android高版本联网失败
3.IDEA 部署Web项目
4.PreparedStatement的executeQuery、executeUpdate和execute
5.preparedstatement execute() *** 作成功!但是返回false
6.HttpServletResponse(一)
7.HttpServletResponse(二)
8.HttpServletRequest
9.HttpUrlConnection
10.java.net.socketexception
以上是内存溢出为你收集整理的android通过web与后台数据库交互全部内容,希望文章能够帮你解决android通过web与后台数据库交互所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)