【java】对数据库操作的那些事(包括数据库中的预处理)

一、连接问题

前面刚介绍了怎么连接数据库,也写了相应的模板,但是它的可维护性非常差,那么怎么解决这个问题呢?

首先写一个配置文件jdbc.properties

<span style="font-size:18px;">## MySQL
driver=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost:3306/hncu?useUnicode=true&characterEncoding=UTF-8
username=root
password=1234

## Oracle
#driver=oracle.jdbc.driver.OracleDriver
#url=jdbc:oracle:thin:@192.168.31.12:1521:orcl
#username=scott
#password=tiger
</span>

然后创建一个生成连接的工厂ConnFactory .java

<span style="font-size:18px;">package cn.hncu.hibernate0;

import java.io.IOException;
import java.sql.Connection;
import java.sql.DriverManager;
import java.util.Properties;

public class ConnFactory {
	private static Connection conn;
	static {
		try {
			//读取配置文件
			Properties p = new Properties();
			p.load(ConnFactory.class.getClassLoader().getResourceAsStream("jdbc.properties"));
			String driver = p.getProperty("driver");
			String url = p.getProperty("url");
			String username = p.getProperty("username");
			String pwd = p.getProperty("password");

			Class.forName(driver);
			conn = DriverManager.getConnection(url,username,pwd);
			System.out.println("已连接到数据库..."+conn);
		} catch (Exception e) {
			throw new RuntimeException("读取配置文件失败", e);
		}
	}
	public static Connection getConn(){
		return conn;
	}
	public static void main(String[] args) {
		getConn();
	}
}
</span>

最后直接通过ConnFactory.getConn()获得。这要做的好处,当改变所要连接的数据库类型时,只需要修改配置文件中的内容即可。

二、讲解getXXX()方法

<span style="font-size:18px;"><span style="white-space:pre">	</span>@Test
	public void getXXXDemo() throws Exception{
		Statement st = ConnFactory.getConn().createStatement();
		String sql = "select * from book";
		ResultSet rs = st.executeQuery(sql);
		while(rs.next()){
			Integer id = rs.getInt(1);//这里的1表示数据表中的第一列,下面同理
			String name = rs.getString(2);
			//Double price = (Double)rs.getObject(3);//出异常,因为内部是采用BigDecimal来处理
			Double price = rs.getDouble(3);
			Object dateTime = rs.getObject(4);//把日期和时间作为一个整体读取出来
			System.out.println(id+","+name+","+price+","+dateTime);

			String strDateTime = dateTime.toString();
			System.out.println(strDateTime);
			strDateTime = rs.getDate(4)+"--"+rs.getTime(4);//日期和时间可以单独获取
			System.out.println(":::"+strDateTime);
		}
		ConnFactory.getConn().close();
	}</span>

注:对于自动增长列,删除之后再插入新记录,序号不会回头,继续往前增长。即中间会出现空号

三、讲解Statement中的三个executeXXX()方法

1、executeQuery: 只能执行select语句

2、executeUpdate: 可以执行insert、delete和update语句,但不能执行select

3、execute:增删改查的4种(任意)语句都能执行。该方法若执行非select语句时返回false,执行select语句时返回true,且st对象会缓存该次查询的结果,我

们可通过ResultSet rs = st.getResultSet()来获得结果集

<span style="font-size:18px;"><span style="white-space:pre">	</span>@Test
	public void executeXXXDemo() throws Exception{
		Statement st = ConnFactory.getConn().createStatement();
	    String sql = "select * from book";
		//String sql = "insert into book(name,price,pub) values('软件工程',22.35,'2015-12-05 22:12:23')";
		//String sql = "update book set price=38.88 where name='软件工程'";
		//String sql = "delete from book where name='软件工程'";
	    //st.executeQuery(sql);
		//st.executeUpdate(sql);
		boolean boo = st.execute(sql);

		if(boo){
			ResultSet rs = st.getResultSet();
			while(rs.next()){
				System.out.println(rs.getObject(2));
			}
		}
	}</span>

四、数据库查询时防黑技术(预处理语句)

案例、用户登录(通过用户输入信息来拼接sql语句----很危险)

<span style="font-size:18px;"><span style="white-space:pre">	</span>@Test//用户登录
	public void login() throws Exception{
		Connection con = ConnFactory.getConn();
		Statement st = con.createStatement();
		Scanner sc = new Scanner(System.in);
		int id = sc.nextInt(); sc.nextLine();
		String name = sc.nextLine();
		String sql = "select count(*) from stud where id="+id+" and sname='"+name+"'";
		System.out.println("sql:"+sql);
		ResultSet rs = st.executeQuery(sql);
		rs.next();
		int a = rs.getInt(1);
		if(a<=0){
			System.out.println("登录不成功");
		}else{
			System.out.println("登录成功");
		}

		con.close();
	}</span>

黑的方法,输入:1002(回车)   1‘ or ‘1‘=‘1

因此:如果需要用 用户输入的信息 来拼接 sql语句,就不能用statement,否则用户可以通过构建where子句中的一个true条件来突破防护对于上面的情

况,应该用PreparedStatement来解决!

<span style="font-size:18px;"><span style="white-space:pre">	</span>@Test//用户登录     黑:1002(回车)   1' or '1'='1
	public void login2() throws Exception{
		Scanner sc = new Scanner(System.in);

		Connection con = ConnFactory.getConn();
		String sql = "select count(*) from stud where id=? and sname=?";//需要用户输入的地方,用占位符('?')来代替,然后在后续通过设参来给占位符赋值
		PreparedStatement pst = con.prepareStatement(sql);
		//设置参数
		int id = sc.nextInt(); sc.nextLine();
		pst.setInt(1, id); //参数1----代表第1个占位符
		String name = sc.nextLine();
		pst.setString(2, name);//参数2

		ResultSet rs = pst.executeQuery();

		rs.next();
		int a = rs.getInt(1);
		if(a<=0){
			System.out.println("登录不成功");
		}else{
			System.out.println("登录成功");
		}

		con.close();
	}</span>

五、获取自动增长列(如id)

<span style="font-size:18px;"><span style="white-space:pre">	</span>@Test //演示获取自动增长列如id
	public void saveAuto() throws Exception{
		Connection con = ConnFactory.getConn();
		String sql = "insert into book(name,price,pub) values('JavaEE',100.8,'2013-06-12 08:30:30')";
		Statement st = con.createStatement();

		//st.executeUpdate(sql);
		st.executeUpdate(sql,Statement.RETURN_GENERATED_KEYS);
		ResultSet rs = st.getGeneratedKeys();//里面封装了自动生成的所有值
		if(rs.next()){
			int id = rs.getInt(1);//获取第1个自动增长列
			System.out.println("自动增长的id:"+id);
		}
		System.out.println("-----------------");

		//预处理语句
		sql =  "insert into book(name,price) values(?,?)";
		PreparedStatement pst = con.prepareStatement(sql,Statement.RETURN_GENERATED_KEYS);
		pst.setString(1, "计算机基础");
		pst.setDouble(2, 28);
		pst.executeUpdate();
		ResultSet rs2 = pst.getGeneratedKeys();//里面封装了自动生成的所有值
		if(rs2.next()){
			int id = rs2.getInt(1);//获取第1个自动增长列
			System.out.println("自动增长的id:"+id);
		}
	}</span>

六、演示批处理语句

public void batch() throws Exception{
		Connection con = ConnFactory.getConn();
		String sql = "insert into book(name,price,pub) values('JavaEE',100.8,'2015-06-12 08:30:30')";
		Statement st = con.createStatement();
		for(int i=0;i<10;i++){
			st.addBatch(sql);
		}
		sql = "update book set price=price*1.1 where price>100";
		st.addBatch(sql);
		int[] a = st.executeBatch();
		for(int r:a){
			System.out.println(r);//r为每条sql语句所影响的记录数
		}
	}

预处理

public void preparedBatch() throws Exception{
		Connection con = ConnFactory.getConn();
		String sql = "insert into book(name,price,pub) values(?,?,?)";
		PreparedStatement pst = con.prepareStatement(sql);
		for(int i=0;i<5;i++){
			pst.setString(1, "Java"+i);
			pst.setDouble(2, 55.85+i);
			pst.setString(3, "2016-12-10 07:07:08");
			pst.addBatch(); //pst的方式不能带参数sql
		}
		//pst.executeBatch();
		int[] a = pst.executeBatch();
		for(int r:a){
			System.out.println(r);//r为每条sql语句所影响的记录数
		}
	}

注意:预处理的方式不能带参数sql,普通的需要

时间: 2024-10-20 15:30:13

【java】对数据库操作的那些事(包括数据库中的预处理)的相关文章

MySQL数据库操作2-新建,删除数据库

新建数据库: 代码: create database 数据库名称; 注意: 1.数据库名称最好使用_db后缀,方便识别用户数据库, 2.用户建立的数据库仅用户本人账户和root有权使用. 示例: create database test_db; 删除数据库: 代码: drop database 数据库名; 示例: drop database test_db; 显示所有数据库: 代码: show databases; 注意: 1.databases是复数形式. 数据库存储引擎: 代码: show

【java】对数据库操作的那些事(包含数据库中的预处理)

一.连接问题 前面刚介绍了怎么连接数据库,也写了对应的模板.可是它的可维护性很差.那么怎么解决问题呢? 首先写一个配置文件jdbc.properties <span style="font-size:18px;">## MySQL driver=com.mysql.jdbc.Driver url=jdbc:mysql://localhost:3306/hncu? useUnicode=true&characterEncoding=UTF-8 username=roo

Oracle数据库操作大全(十)——数据库(表)的逻辑备份与恢复

数据库(表)的逻辑备份与恢复--导出 system可导出任何方案,例如: exp system/[email protected] owner(system,scott) file=d:system.dmp;//注:可导出两种方案system方案和Scott方案 exp system/[email protected] owner(scott) file=d:system.dmp;//只可导出Scott方案(这是利用system导出Scott方案,缘由:system权限高于Scott),导出方案

ThinkPhp框架的数据库操作(查询)

TP框架有一套自己的数据库操作的代码,包括数据库的增.删.改.查.本文主要讲解TP框架的数据库查询操作. 找到入口文件的控制器: 我这里的入口文件是Show文件夹下的控制器. 打开Login控制器. 代码写在index方法中. 数据库的定义方式有三种. 前两种要写模型函数,最后一种不需要. 模型的例子(放在Model文件夹下,名字与类名相同,本例的名字:UserModel.class.php) <?php namespace Admin\Model; use Think\Model; class

java——数据库操作

1.JDBC的驱动程序有几种 1>JDBC-ODBC 依靠ODBC驱动器和数据库通信,将ODBC二进制代码加载到使用该驱动程序的客户机上 2>本地API 把客户机的API上的JDBC调用转换为Oracle,DB2或其他DBMS的调用 3>JDBC网络驱动程序 将JDBC转换为与DBMS无关的网络协议,又被某个服务器转换为一种DBMS协议. 4>本地协议驱动 将JDBC直接转换为DBMS所使用的网络协议,允许客户机上直接调用DBMS服务器 2.存储过程 调用无参数的存储过程 crea

十三、EnterpriseFrameWork框架核心类库之数据库操作(多数据库事务处理)

本章介绍框架中封装的数据库操作的一些功能,在实现的过程中费了不少心思,针对不同数据库的操作(SQLServer.Oracle.DB2)这方面还是比较简单的,用工厂模式就能很好解决,反而是在多数据库同时操作方面走了不少弯路:现在从以下几个方面进行说明: 一.不同数据库操作 此处用到了工厂模式来实现不同数据库操作,看下图 AbstractDatabase是一个抽象类,定义了所有对数据库的操作抽象方法,包括执行一个SQL语句.执行存储过程.事务操作等 [Serializable] public abs

Django模型-数据库操作

前言 前边记录的URLconf和Django模板全都是介绍页面展示的东西,也就是表现层的内容.由于Python先天具备简单而强大的数据库查询执行方法,Django 非常适合开发数据库驱动网站. 这篇开始,进入到了Django模型,也就是数据库操作. 自带 Sqlite3 数据库查询方式 为了简单,使用Python自带的Sqlite3数据库进行实例说明. 先看一个传统的数据库操作示例: 1 from django.shortcuts import render 2 import sqlite3 3

Entity Framework 实体框架的形成之旅--几种数据库操作的代码介绍(9)

本篇主要对常规数据操作的处理和实体框架的处理代码进行对比,以便更容易学习理解实体框架里面,对各种数据库处理技巧,本篇介绍几种数据库操作的代码,包括写入中间表操作.联合中间表获取对象集合.递归操作.设置单一字段的修改等几种方式. 1.写入中间表操作 一般情况下,我们可以通过执行数据库脚本方式写入. /// <summary> /// 增加用户IP信息 /// </summary> /// <param name="userID"></param&

MySQL数据库操作类(PHP实现,支持连贯操作)

使用过ThinkPHP框架的同学可能会对于其中数据库模型操作特别有好感,ThinkPHP提供了数据库操作的简单的操作,对于连接数据库,数据库的增删改查等数据操作都非常的nice,同时支持连贯操作,对于那些不习惯写sql语句的同学真是大大的便利.(注:sql还是很重要的,不要因为用了框架就把原先的忘了). 而在笔者使用php操作redis实现后台任务的过程中,也想要借助这种便利,但无奈redis操作单独的类,直接访问其中的controller文件的话,总是会提示M方法失败,导致此模型方法不能使用.