Android通过Web与后台数据库交互
2022.01.08 更新
已更新新版本博客,更新内容与原文章相比有点多,因此新开了一篇博客,请戳这里。
1 背景
开发一个App
与后台数据库交互,基于MySQL
+原生JDBC
+Tomcat
,没有使用DBUtils
或JDBC
框架,纯粹底层JDBC
实现。
这几天踩了很多坑,希望能帮助读者少踩坑。
2 开发环境
Windows10
- 服务器
CentOS 7
Android Studio 3.5.1
IntelliJ IDEA 2019.02
MySQL 8.0.17
Tomcat 9.0.26
3 准备环境
主要是安装MySQL
与Tomcat
,已经安装的可以略过。
3.1 安装MySQL
这个是目前比较新的MySQL
版本。
服务器系统是CentOS
。
其他系统安装看这里:
- Win10
- Ubuntu
- Fedroa
- ReaHat
CentOS
使用yum
安装即可。
3.1.1 下载并安装MySQL
sudo yum localinstall https://repo.mysql.com//mysql80-community-release-el7-1.noarch.rpm
sudo yum install mysql-community-server
3.1.2 启动服务并查看初始化密码
sudo service mysqld start
sudo grep 'temporary password' /var/log/mysqld.log
3.1.3 修改密码
首先使用root登录:
mysql -u root -p
输入上一步看到的密码,接着使用alter
修改密码:
alter mysql.user 'root'@'localhost' identified by 'password';
注意新版本的MySQL
不能使用太弱的密码,比如:
出现这种提示的话则说明密码太弱了,请使用一个更高强度的密码。
3.1.4 允许外部访问
use mysql;
update user set host='%' where user='root';
这个可以根据自己的需要去修改,host='%'
表明允许所有的ip
登录,也可以设置特定的ip
,若使用host='%'
的话建议新建一个用户配置相应的权限。
3.1.5 配置防火墙(可选)
一般来说需要在对应的云厂商的防火墙配置中开启响应端口,如图:
其中授权对象可以根据自己的需要更改,0.0.0.0/0
表示允许所有的ip
。
3.2 安装Tomcat
3.2.1 下载并上传到服务器
先去官网下载,下载后上传文件到服务器:
笔者使用的是scp
命令,使用不熟练的可以戳这里看看:
scp apache-tomcat-xxxx.tar.gz username@xx.xx.xx.xx:/
改成自己的用户名和ip
。
3.2.2 解压
ssh
连接到服务器,接着移动到/usr/local
并解压:
mkdir /usr/local/tomcat
mv apache-tomcat-xxxx.tar.gz /usr/local/tomcat
tar -xzvf apache-tomcat-xxx.tar.gz
3.2.3 修改默认端口(可选)
修改conf/server.xml
文件,一般只需修改
<Connector port="8080" protocol="HTTP/1.1" connectionTimeout="20000" redirectPort="8443" />
中的8080
端口,修改这个端口即可。
需要的话自行更改。
3.2.4 启动
运行bin
目录下的startup.sh
:
cd bin
./startup.sh
3.2.5 测试
浏览器输入:
服务器IP:端口
若出现:
则表示成功。
3.2.6 开机启动(可选)
建议配置开机启动,修改/etc/rc.local
文件,添加:
sh /usr/local/tomcat/bin/startup.sh
这个根据自己的Tomcat
安装路径修改,指定bin
下的startup.sh
。
4 建库建表
创建用户表,这里简化操作就不创建新用户不授权了。
这是一个在本地用root
登录的示例,请根据实际情况创建并授权用户。
4.1 用户表
CREATE DATABASE userinfo;
USE userinfo;
CREATE TABLE user
(id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,name CHAR(30) NULL,password CHAR(30) NULL
);
4.2 导入
mysql -u root -p < user.sql
这样准备工作就做好了,下面正式开始进行敲代码阶段。
5 后端部分
5.1 创建项目
选择Web Application
:
5.2 添加依赖库
创建一个叫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
:
5.3 创建包与类
总共4个包:
com.servlet
:用于处理来自前端的请求,包含SignUp.java
、SignIn.java
com.util
:主要功能是数据库连接,包含DBUtils.java
com.entity
:实体类,包含User.java
com.dao
:操作用户类的类,包含UserDao.java
5.4 DBUtils
连接数据库的类,纯粹的底层JDBC
实现,注意驱动版本。
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
和库名修改。
注意,MySQL 8.0以上使用的注册驱动的语句是:
Class.forName("com.mysql.cj.jdbc.Driver");
旧版的是:
Class.forName("com.mysql.jdbc.Driver");
5.5 User
User
类比较简单,就是就三个字段与Getter+Setter
。
public class User {private int id;private String name;private String password;//三个Getter与三个Setter//...
}
5.6 UserDao
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
进去了。。。
啊这。。。
好吧,还是函数的问题。
- 一般来说
select
使用executeQuery()
,executeQuery()
返回ResultSet
,表示结果集,保存了select
语句的执行结果,配合next()
使用 delete
,insert
,update
使用executeUpdate()
,executeUpdate()
返回的是一个整数,表示受影响的行数,即delete
,insert
,update
修改的行数,对于drop
,create
操作返回0
create
,drop
使用execute()
,execute()
的返回值是这样的,如果第一个结果是ResultSet
对象,则返回true
,如果第一个结果是更新计数或者没有结果则返回false
所以在这个例子中
return preparedStatement.execute();
肯定返回false
,所以才会数据库这边insert
进去,但前端显示注册失败(这个问题笔者找了是真的久。。。)
5.7 SignIn
+SignUp
servlet
包中:
SingIn
类用于处理登录,调用JDBC
查看数据库是否有对应的用户SignUp
类用于处理注册,把User
添加到数据库中
SignIn.java
如下:
@WebServlet("/SignIn")
public class SingIn extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws IOException,ServletException{this.doPost(httpServletRequest,httpServletResponse);}@Overrideprotected 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-8String 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
的Servlet
,可用于实现Servlet
与URL
的映射,如果不在这里添加这个注解,则需要在WEB-INF
目录下的web.xml
添加一个<servlet-mapping>
。
@WebServlet("/SignIn")
接着设置响应类型与编码:
httpServletResponse.setContentType("text/plain;charset=utf-8");//设置相应类型为html,编码为utf-8
HttpServletRequest.getParameter(String name)
方法表示根据name
获取相应的参数:
String name = httpServletRequest.getParameter("name");
String password = httpServletRequest.getParameter("password");
下面是SignUp.java
:
@WebServlet("/SignUp")
public class SignUp extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse) throws IOException,ServletException{this.doPost(httpServletRequest,httpServletResponse);}@Overrideprotected 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-8String 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错误码与出错信息}}
}
5.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>
是访问的路径。
5.9 Hello.html
测试文件
最后添加一个叫Hello.html
的测试文件。
<!DOCTYPE html><head><meta charset="utf-8"><title>Welcome</title></head><body>Hello web.</body>
</html>
6 打包发布
笔者用的IDEA
,Eclipse的打包请看这里。
6.1 Project Structure->Artifacts->Web Application:Archive
6.2 创建目录并添加模块
修改名字,并创建WEB-INF
目录与子目录classes
:
选中classes
,添加Module Output
,选择自己的web项目:
6.3 添加依赖库与其他文件
添加JAR
包,选中lib
目录后添加JAR包文件:
接着添加Hello.html
与web.xml
,web.xml
需要在WEB-INF
目录里,Hello.html
在WEB-INF
外面:
6.4 打包
Build->Build Artifacts
:
6.5 上传测试
打包好的WAR
文件上传到服务器Tomcat
的webapps
目录下:
scp ***.war username@xxx.xxx.xxx.xxx:/usr/local/tomcat/webapps
注意改成自己的webapps
目录。
Tomcat
启动后,在浏览器输入
服务器IP:端口/项目/Hello.html
笔者为了方便就在本地测试了:
7 Android
端
7.1 新建工程
7.2 MainActivity.java
public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button signin = (Button) findViewById(R.id.signin);signin.setOnClickListener(new View.OnClickListener() {@Overridepublic 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() {@Overridepublic void run() {Toast.makeText(MainActivity.this, "登录成功", Toast.LENGTH_SHORT).show();}});else {runOnUiThread(new Runnable() {@Overridepublic void run() {Toast.makeText(MainActivity.this, "登录失败", Toast.LENGTH_SHORT).show();}});}}});Button signup = (Button) findViewById(R.id.signup);signup.setOnClickListener(new View.OnClickListener() {@Overridepublic 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() {@Overridepublic void run() {Toast.makeText(MainActivity.this, "注册成功", Toast.LENGTH_SHORT).show();}});else {runOnUiThread(new Runnable() {@Overridepublic void run() {Toast.makeText(MainActivity.this, "注册失败", Toast.LENGTH_SHORT).show();}});}}});}
}
两个简单的Button
绑定事件,然后设置两个Toast
提示信息。
7.3 UserService.java
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;}@Overridepublic void run(){try {URL url = new URL(path);HttpURLConnection httpURLConnection = (HttpURLConnection) url.openConnection();httpURLConnection.setConnectTimeout(8000);//设置连接超时时间httpURLConnection.setReadTimeout(8000);//设置读取超时时间httpURLConnection.setRequestMethod("POST");//设置请求方法,postString 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>
设置,与源码中的@WebServlet()
注解一致。
另外一个要注意的就是线程问题,需要新开一个线程进行HTTP
的连接。
7.4 activity_main.xml
前端页面部分很简单,就两个按钮,用于验证功能。
<?xml version="1.0" encoding="utf-8"?>
<LinearLayoutxmlns: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"><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="horizontal"><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="用户名"/><EditTextandroid:layout_width="300dp"android:layout_height="60dp"android:id="@+id/etname"/></LinearLayout><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="horizontal"><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="密码"/><EditTextandroid:layout_width="300dp"android:layout_height="60dp"android:id="@+id/etpassword"/></LinearLayout><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="horizontal"><Buttonandroid:layout_width="120dp"android:layout_height="60dp"android:text="注册"android:id="@+id/signup"/><Buttonandroid:layout_width="120dp"android:layout_height="60dp"android:text="登录"android:id="@+id/signin"/></LinearLayout>
</LinearLayout>
8 测试
8.1 注册测试
随便输入用户名与密码:
查看数据库:
8.2 登录测试
9 注意事项
9.1 数据库用户名与密码
数据库的用户名和密码一定要设置正确,要不然会像下图一样抛出异常:
在加载驱动错误时也可能会出现这个错误,因此要确保打成WAR
包时lib
目录正确且JAR
包版本正确。
还有就是由于这个是JDBC
的底层实现,注意手写的SQL
语句不能错。
千万千万别像笔者这样:
9.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" />
9.3 安全组/防火墙问题
因为如果是现在的云服务器一般都使用安全组而不是直接使用防火墙配置安全设置,所以如果没法访问请到云服务器上面对应的安全组配置中开放端口,默认是8080
端口,一般是允许所有ip
通过,源ip
配置为0.0.0.0/0
,比如:
如果设置了安全组通过了,还是不能访问,可能是服务器防火墙的问题,可以查看iptables
(CentOS8
使用firewalld
)是否开启:
systemctl status iptables
开启的话可以选择关闭:
systemctl stop 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
:
systemctl restart iptables
9.4 HTTP
注意事项
由于从Android P
开始,默认要求使用加密连接,也就是要使用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
两种解决办法:
- 使用
HTTPS
- 修改默认的
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
中加入:
<applicationandroid:networkSecurityConfig="@xml/network_security_config"
/>
即可。
如果上面的方法不行,可以试一下直接在AndroidManifest.xml
直接加入一句:
<application
android:usesCleartextTraffic="true"
/>
9.5 线程问题
从Android 4.0
开始,联网不能在主线程操作,万一网络不好就会卡死,所以有关联网的操作都需要新开一个线程,不能在主线程操作。
9.6 AVD
问题
这个问题笔者找了很久,HTTP
连接没问题,服务器没问题,数据库没问题,前端代码没问题,然后去了StackOverflow
,发现是AVD
的问题。。。
简单来说就是卸载了APP
再重启AVD
,居然成功了。。。
9.8 404
根据评论反映(这里同时感谢评论的各位)会出现404
问题,出现404
的主要原因就是访问路径的不正确,这里分三种情况进行说明。
9.8.1 本地IDEA
中访问
IDEA
运行项目后,可以使用postman
进行测试,访问路径:
http://localhost:8080/SignIn
即可,因为此时还没有打包,所以不需要加上打包的项目名。
9.8.2 本地Tomcat
访问
打包后会有一个叫xxx.war
的打包文件,比如叫demo.war
,那么使用postman
测试的路径就要变为
http://localhost:8080/demo/SignIn
因为Tomcat
会自动把demo.war
解包,运行Tomcat
后会在webapps
下会有一个叫demo
的文件夹,里面就是打包后的文件,此时的路径需要以demo
为基准,因此SignIn
前需要加上demo
。
9.8.3 部署到服务器的Tomcat
部署到服务器后与本地Tomcat
是基本一样的,只是把对应的locahost
改成对应的域名或者公网ip
即可,比如:
http://www.example.com:8080/demo/SignIn
http://111.111.111.111:8080/demo/SignIn
10 源码与JAR
包
10.1 JAR
包
到这里搜索下载。
10.2 源码
- Github
- 码云
11 最后
首先感谢读者能看到末尾,笔者也确实花了不少时间写这篇文章,但是,总会有不对或让读者疑惑的地方,希望读者有疑问或者其他什么问题可以私信笔者或在评论中指出,感激不尽。
12 参考网站
- CSDN-Android 通过Web服务器与Mysql数据库交互
- CSDN-Android高版本联网失败
- CSDN-IDEA 部署Web项目
- CSDN-PreparedStatement的executeQuery、executeUpdate和execute
- CSDN-preparedstatement execute()操作成功!但是返回false
- CSDN-HttpServletResponse(一)
- CSDN-HttpServletResponse(二)
- CSDN-HttpServletRequest
- StackOverflow-HttpUrlConnection
- StackOverflow-java.net.socketexception
Android通过Web与后台数据库交互相关推荐
- Android之使用HttpPost提交数据到服务器(Android手机客户端和后台服务器交互)
这是一个小型的数据交互案例,即Android手机客户端和后台服务器交互(数据库mysql) 服务器端 首先服务器端数据库(用户名root密码123456),db_student.sql数据库表user ...
- hbuilder 读取MySQL_Hbuilder和后台数据库交互
可以在myeclipse里面搭建一个自己的过滤器,代码如下: //过滤器 public class crossDomainFilter implements Filter{ public void d ...
- 微信小程序和后台数据库交互原理
原理图如下:
- servlet向ajax传递数据库,一、JSP、servlet、SQL三者之间的数据传递(前台与后台数据交互)...
背景: 目前业界很流行的MVC(model-view-control)开发模式,理解为 模型是Bean, 视图是 Html/Jsp, 控制是Servlet, 关联数据库的Dao web的运行机制: 数 ...
- flask后台开发之数据库交互
目录 一.前言 二.环境安装 1.安装Python与SQL进行交互的库 2.使用docker搭建mysql服务 三.用Python连接mysql 四.与数据库的交互 1.创建表 2.删除表 3.插入数 ...
- android前端怎样php后台交互(基础篇)
android前端怎样php后台交互(基础篇) android客户端和php+mysql+apache搭建的服务器之间的简单交互,实现登入功能. 实现原理就是android客户端发送请求,传给服务器登 ...
- ajax提交到mysql_利用ajax的方式来提交数据到后台数据库及交互功能
怎么样用ajax来提交数据到后台数据库,并完成交互呢????? 一.当我们在验证表单的时候,为了阻止把错误的也发送到服务器,我们通常这样设置: $(function(){ var isusername ...
- Axure 经典实例高保真原型下载(Axure高保真酒店管理交互组件连锁酒店erp酒店企业web端后台管理财务管理会员管理网销管理报表管理))
作品介绍:Axure高保真酒店管理交互组件&连锁酒店erp&酒店企业web端后台管理&财务管理&会员管理&网销管理&报表管理 Axure原型演示及下载地 ...
- 在线生鲜订购配送系统,生鲜订购系统 生鲜配送系统 前端+后台 Android源码+SSH后台管理系统+MySQL数据库
在线生鲜订购配送的系统,包括Android源码+SSH后台管理系统+MySQL数据库. 客服端:功能划分如下 1.我的 登录:账户+密码+验证码 注册:邮箱/手机号注册 订单管理:查看/删除(显示订单 ...
最新文章
- 654. Maximum Binary Tree最大二叉树
- 杭电OJ-1062_Text Reverse
- linux命令中提取某一列,怎么用Linux命令提取表格文本中的某列
- PHP中的语法特点小结
- mysql 查询 in memory_In-Memory整体汇总
- ctf wav文件头损坏_【CTF入门第二篇】南邮CTF web题目总结
- 谷歌android wear智能腕表 价格,谷歌Android Wear 2.0更新推送:仅三款智能手表可享受...
- 智能指针(二):shared_ptr实现原理
- Qt 学习之路 2(1):序 笔记
- 2021年“泰迪杯”数据分析技能B题-肥料登记数据分析赛题
- 中兴设备交换机路由器清除清理指定接口计数的统计信息命令
- SpreadJS企业表格技术实践一:自定制表格编辑器
- php随机生成昵称,PHP随机生成姓名的方法介绍
- 苹果CMSv10系统标签,仿站必备
- 电脑时间不同步怎么办?
- GOM传奇文件目录功能说明
- 一图读懂IP数据包头结构
- java.lang.IllegalStateException: Underflow in restore - more restores than saves
- java基础_题库详解
- 一文读懂机智云物联网APP开发