zoukankan      html  css  js  c++  java
  • jdbc

    jdbc.driver=com.mysql.jdbc.Driver
    jdbc.url=jdbc:mysql://localhost:3306/dangdang
    jdbc.username=root
    jdbc.password=root

    ----------------------------------------------------------------

    class.forName(oracle.jdbc.OracleDriver)
    jdbc:oracle:thin:@localhost:1521:xe
    "com.mysql.jdbc.Driver"
    jdbc:mysql://localhost:3306/student

    JDBC
    一,JDBC内容
    接口:sun定义 java.sql 和javax.sql
    实现类:数据库厂商提供
    二,jdbc使用
    1,如何使用jar
    --dos使用
    classpath中添加jar路径
    2,jdbc流程
    a:注册驱动类
    b:创建链接
    c:创建statement(不常用)
    d:执行sql
    e:处理结果
    f:释放资源statement 也要close
    get列时序号从1开始
    3,prepareStatementjava.sql statement子接口
    PrepareStatement ps =conn.prepareStatement(sql);
    ps.setxxx(1,name)
    ps.setxxx(2,password)
    ps.exq exu (使用午餐的构造)
    里面的参数不用拼接
    在执行前时插入
    4.两者区别
    state = com.cre...
    stat.execute(sql)
    不安全,有注入
    prep = conn.create..(sql)
    prep.execute()
    无sql注入,切性能更高
    服务器操作
    1检查权限,
    2检查语法
    3,sql转化为内部指令 前三是编译过程
    4,执行指令
    statement创建成功时不发送sql,执行时发送
    例:for(int i=0;i<10;i++{
    stat.ex(insert...)
    }
    prep 可只把setxxx写在for循环中
    只用编译一次即可,所以说也叫作预编译语句
    stat:异构sql
    prep:同构sql(只有参数不同
    日期转换:日期转换 strDate --- util.Date---sql.Date
    //插入或修改字段的类型是时间类型Date时,需要使用时间转换
    //
    1.设置格式,格式应与字符串相匹配。和数据库中to_date方法类似
    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
    //
    2.strDate--util.Date
    Date util
    Date = format.parse(strDate);
    //
    3.util.Date---sql.Date
    java.sql.Date date = new java.sql.Date(utilDate.getTime());


    1.事务控制
    JDBC默认的是自动的事务提交

    相关API:控制事务提交的对象为连接 Connection

    //设置事务提交方式为手动

    conn.setAutoCommit(false);//true---自动提交 false---手动提交

    //事务提交
    conn.commit();

    //事务回滚
    conn.rollback();

    2.业务类方法的书写步骤

    //1.获取Connection
    conn = JdbcUtil2.getConnection();

    System.out.println("这是service获得连接:"+conn);

    //2.设置事务提交为手动提交
    conn.setAutoCommit(false);

    //3.调用DAO中的方法完成业务操作
    dao.insert(clazz);

    //4.提交事务 or 回滚事务
    conn.commit();

    conn.rollback

    3.当前的事务控制存在问题

    首先service层获得的连接与Dao不是一条 其次通过单例可以解决这样的问题,但是会存在线程不安全的情况。

    为了解决线程安全,且可以控制事务的问题
    我们需要使用线程局部表变量,ThreadLocal
    ~~~

    ####
    九、线程局部变量

    ~~~java
    线程局部变量,再多线程环境下,可以为每个使用该变量的线程 分配线程内部独有的 线程变量值。

    1.API:
    ThreadLocal th = new ThreadLocal();

    th.set(obj);//往线程局部变量中设置

    obj = th.get();//返回该线程局变量的值

    th.remove();//移除该线程局变量的值

    2.作用:
    1)保证 同一个线程内部 使用相同的值
    2)保证不用线程使用不用的值

    3.原理:
    当调用 set方法时,会将当前线程对象当做Key,将set中的参数当做value 放入到一个map集合中 ,这个集合由Thread进行管理的。
    当使用get方法时,以当前线程为key来向该map集合取值。

    4.使用ThreadLoacl来修改getConnection方法
    //创建连接的方法

    private static ThreadLocal<Connection> th = new ThreadLocal<Connection>();

    public static Connection getConnection(){

    Connection conn = th.get();

    if(conn==null){//如果为空就创建

    //1.注册驱动

    try {

    Class.forName(p.getProperty("driver"));

    //2.建立连接

    conn = DriverManager.getConnection(p.getProperty("url"),p.getProperty("username"),p.getProperty("password"));

    //3.并把连接放入ThreadLocal

    th.set(conn);

    } catch (Exception e) {
    // TODO Auto-generated catch block

    e.printStackTrace();
    }
    }

    return conn;
    }

    if(conn!=null){

    try {

    conn.close();
    th.remove();
    } catch (SQLException e) {

    // TODO Auto-generated catch block
    e.printStackTrace();
    }
    }


    注意:1)在控制事务时,dao内不要关闭连接

    2)在关闭连接后,要清空线程局部变量中的废连接
    ~~~

    ####
    十、JDBC的

    分层分包:

    都在域名倒写的包下 比如 com.zzu.xjh


    1)sql 存放了关于该项目相关的一些sql语句文档


    2)conf 放置了配置文件


    3)util 放置了该项目的工具类


    4)test 放置了该项目相关的测试类


    5)entity 放置了该项目的实体类


    6)dao 放置该项目的DAO接口 在该包下应有一个impl包 里面放置 对应dao的实现类


    7)service 放置该项目的service 接口 在该包下应有一个impl包 里面放置 对应service 的实现类


    8)conroller/action 放置该项目的控制器


    9)view (可能出现)

    import java.sql.Connection;
    import java.sql.PreparedStatement;
    import java.sql.ResultSet;
    import java.util.ArrayList;
    import java.util.List;
    
    
    
    public class JdbcTemplate<T> {
        /*
         * 专门完成增 删 改操作
         * 
         * insert into t_user (id,name,password) values (suns_seq.nextval,?,?);
         * 
         * jdbcTemplate.update(sql,new Object[]{"suns","123456"});
         * 
         * delete from t_user where id = ?
         * 
         * jdbcTemplate.delete(sql,new Object[]{1});
         * 
         * update t_product set name=?,price=? where id = ? 
         * 
         * jdbcTemplate.update(sql,new Object[]{"suns",10.2,1});
         * 
         * 可变长参数  1 等同于数组
         *          2 只能放在参数表最后 ,只能有一个
         *          
         *  "delete from t_user where id = 10";        
         * 
         */
    	public void update(String sql,Object... args){
    		Connection conn = null;
    		PreparedStatement pstmt = null;
    		try{
    			conn = JDBCUtil.getConnection();
    			pstmt = conn.prepareStatement(sql);
    			
    			if(hasUpdateParam(args)){
    			   for(int i=0;i<args.length;i++){
    				 pstmt.setObject(i+1,args[i]);
    			   }
    			}   
    			
    			pstmt.executeUpdate();
    		}catch(Exception e){
    			e.printStackTrace();
    		}finally{
    			JDBCUtil.close(null, pstmt);
    		}
    		
    	}
    
    	private boolean hasUpdateParam(Object... args) {
    		return args.length!=0;
    	}
    	
    	
    	public List<T> query(String sql,RowMapper<T> rowMapper,Object... args){
    		Connection conn = null;
    		PreparedStatement pstmt = null;
    		ResultSet rs = null;
    		List<T> rets = new ArrayList<T>();
    		try{
    			conn = JDBCUtil.getConnection();
    			pstmt = conn.prepareStatement(sql);
    			if(hasUpdateParam(args)){
       			  for(int i=0;i<args.length;i++){
    				pstmt.setObject(i+1, args[i]);
    			  }
    			}
    			rs = pstmt.executeQuery();
    			while(rs.next()){
    			     T ret = rowMapper.mapRow(rs);
    			     rets.add(ret);
    			}
    			return rets;
    		}catch(Exception e){
    			e.printStackTrace();
    			return null;
    		}finally{
    			JDBCUtil.close(null, pstmt, rs);
    		}
    	}
    	
    	
    	public T queryForObject(String sql,RowMapper<T> rowMapper,Object... args){
    		Connection conn = null;
    		PreparedStatement pstmt = null;
    		ResultSet rs = null;
    		T ret = null;
    		try{
    			conn = JDBCUtil.getConnection();
    			pstmt = conn.prepareStatement(sql);
    			if(hasUpdateParam(args)){
       			  for(int i=0;i<args.length;i++){
    				pstmt.setObject(i+1, args[i]);
    			  }
    			}
    			rs = pstmt.executeQuery();
    			if(rs.next()){
    			    ret = rowMapper.mapRow(rs);
    			}
    			return ret;
    		}catch(Exception e){
    			e.printStackTrace();
    			return null;
    		}finally{
    			JDBCUtil.close(null, pstmt, rs);
    		}
    	}
    	
    	
    	
    	
    }

      

    import java.io.IOException;
    import java.io.InputStream;
    import java.sql.Connection;
    
    import java.sql.DriverManager;
    import java.sql.PreparedStatement;
    import java.sql.ResultSet;
    import java.util.Properties;
    
    /*
     * 性能 效率 内存使用
     * 维护性 (便于修改)
     * 基本功能
     */
    
    public class JDBCUtil {
        private static Properties p = new Properties();
        private static ThreadLocal<Connection> tl = new ThreadLocal<Connection>();
        
        static{
            //1 读入Properties文件
              InputStream is = JDBCUtil.class.getResourceAsStream("/jdbc.properties");
             //2 传入Properties集合
              try {
                p.load(is);
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
        
       
        //创建连接
        public static Connection getConnection() throws Exception{
              Connection conn = tl.get();
              if(conn==null){
                  Class.forName(p.getProperty("driver"));
                  conn = DriverManager.getConnection(p.getProperty("url"),p.getProperty("username"),p.getProperty("password"));
                  tl.set(conn);  
                  return conn;
              }
              return conn;
        }
        
        
        //关闭资源
        public static void close(Connection conn,PreparedStatement pstmt,ResultSet rs){
            if(rs!=null){try{rs.close();}catch(Exception e){}}
            if(pstmt!=null){try{pstmt.close();}catch(Exception e){}}
            if(conn!=null){try{conn.close();tl.remove();}catch(Exception e){}}
        }
        
        public static void close(Connection conn,PreparedStatement pstmt){
            if(pstmt!=null){try{pstmt.close();}catch(Exception e){}}
            if(conn!=null){try{conn.close();tl.remove();}catch(Exception e){}}
        }
        
    }

    1:1,一般是双向关系
    建实体类时,两边都建
    但建表时只有一个外键,建在不主要一方
    定义表时,定义外键,做唯一
    private Integer id;
    private String name;
    private String password;
    private Date birthday;
    private PassPort passport;
    ---------------------------
    private Integer id;
    private String serial;
    private Integer expire;
    private Person person;
    ----------------------------
    create table t_person(
    id integer primary key,
    name varchar(12),
    password varchar(12),
    birthday date
    );
    create table t_passport(
    id integer primary key,
    serial varchar(12),
    expire varchar(12),
    person_id integer unique references t_person(id)
    );
    --------------------------------------------------------------------------------------------
    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "mybatis-3-mapper.dtd" >
    <mapper namespace="com.dzf.dao.o2o.PersonDao">

    <resultMap id="personResultMap" type="Person" >
    <id property="id" column="person_id"/>
    <result property="name" column="person_name"/>
    <result property="password" column="person_password"/>
    <result property="birthday" column="person_birthday"/>
    <!-- 当只有passport一个对象 -->
    <association property="passport" javaType="PassPort">
    <id property="id" column="passport_id"></id>
    <result property="serial" column="passport_serial"/>
    <result property="expire" column="passport_expire"/>
    </association>
    </resultMap>
    <select id="queryPersonById" resultMap="personResultMap">
    select p.id as person_id,p.name as person_name,p.password as person_password,p.birthday as person_birthday
    ,pp.id as passport_id,pp.serial as passport_serial,pp.expire as passport_expire
    from t_person p inner join t_passport pp
    on p.id = pp.person_id
    where p.id = #{id}
    </select>
    </mapper>

    ----------------------------------------------
    n:1
    多对一单向
    private Integer id;
    private String name;
    private Double salary;
    private Dept dept;
    private Integer id;
    private String code;
    private String name;
    create table t_employee(
    id integer primary key,
    name varchar(12),
    salary decimal(12,2),
    dept_id integer references t_dept(id)
    );
    create table t_dept(
    id integer primary key,
    code varchar(12),
    name varchar(12)
    );
    <resultMap id="employeeResultMap" type="Employee" >
    <id property="id" column="employee_id"/>
    <result property="name" column="employee_name"/>
    <result property="salary" column="employee_salary"/>
    <association property="dept" javaType="Dept">
    <id property="id" column="dept_id"></id>
    <result property="code" column="dept_code"/>
    <result property="name" column="dept_name"/>
    </association>
    </resultMap>
    <select id="queryEmployeeByName" resultMap="employeeResultMap">
    select e.id as employee_id,e.name as employee_name,e.salary as employee_salary,
    d.id as dept_id,d.code as dept_code,d.name as dept_name
    from t_employee e inner join t_dept d
    on e.dept_id = d.id
    where e.name = #{name}
    </select>
    <select id="queryDeptById" resultType="Dept">
    select id,name,code
    from t_dept
    where id = #{id}
    </select>
    --------------------------------------------------------------------------------------------------------------
    1:n单向(n:1注意更换时 表不用改,表天生是双向的,俩表就一外键,在多的一方)
    private Integer id;
    private String name;
    private Double salary;
    private Integer id;
    private String code;
    private String name;
    private List<Employee> list = new ArrayList<Employee>();记得初始化
    <resultMap id ="deptResultMap" type="Dept" >
    <id property="id" column = "dept_id"/>
    <result property = "code" column="dept_code"/>
    <result property = "name" column="dept_name"/>
    <collection property="list" ofType="Employee"><!-- 集合中元素类型 -->
    <id property="id" column="employee_id"/>
    <result property="name" column="employee_name"/>
    <result property="salary" column="employee_salary"/>
    </collection>
    </resultMap>
    <select id="queryDeptByName" resultMap="deptResultMap">
    select d.id as dept_id ,d.code as dept_code,d.name as dept_name,
    e.id as employee_id,e.name as employee_name,e.salary as employee_salary
    from t_dept d inner join t_employee e
    on d.id = e.dept_id
    where d.name = #{name}
    </select>
    <select id="queryEmployeeById" resultType="Employee">
    select id,name,salary
    from t_employee
    where id= #{id}
    </select>
    -----------------------------------------------------------------------------------------------------------------
    多对多双向,student,course
    外键如何建?,不能建,
    解决办法:引入第三张关系表

    ______________________________________________________________________

    import java.sql.Connection;
    import java.sql.SQLException;
    
    /*
     *  事务管理器 用于控制事务
     */
    public class TransactionManager {
    
        public static void begin(){
            try {
                Connection conn = JDBCUtil.getConnection();
                conn.setAutoCommit(false);
            } catch (SQLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
        
        public static void commit(){
            Connection conn = null;
            try {
                conn = JDBCUtil.getConnection();
                conn.commit();
            } catch (SQLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }finally{
                JDBCUtil.close(conn, null);
            }
            
            
        }
        
        public static void rollback(){
            Connection conn = null;
            try {
                conn = JDBCUtil.getConnection();
                conn.rollback();
            } catch (SQLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }finally{
                JDBCUtil.close(conn, null);
            }
            
        }
        
    }
  • 相关阅读:
    关于ACID,BASE和CAP定理的探究
    2020年10月3日——武汉,成都,南京房价揭秘
    程序员如何选择自己的保险
    Yarn系列(一)——Yarn整体介绍
    利用媒体查询实现响应式布局
    移动端web布局:适配
    scss在编辑器中保存自动编译css插件及安装
    移动端web布局:像素与成像的基本原理
    微信小程序:路由
    自定义vue指令
  • 原文地址:https://www.cnblogs.com/Dean0731/p/11565226.html
Copyright © 2011-2022 走看看