规范数据库设计

规范数据库设计

1、三大范式

引用自 关系型数据库设计:三大范式的通俗理解

1.1 第一范式(1NF) 保证每一列不可再分

? 要求数据库表的每一列都是不可分割的原子数据项。

举例:

规范数据库设计

? 在上面的表中,“家庭信息”和“学校信息”列均不满足原子性的要求,故不满足第一范式,调整如下

? 规范数据库设计

? 可见,调整后的每一列都是不可再分的,因此满足第一范式(1NF);

1.2 第二范式(2NF)

? 满足第一范式的前提下,每张表只描述一件事

? 在1NF的基础上,非码属性必须完全依赖于候选码(在1NF基础上消除非主属性对主码的部分函数依赖)

第二范式需要确保数据库表中的每一列都和主键相关,而不能只与主键的某一部分相关(主要针对联合主键而言)。

举例:

规范数据库设计

? 在上图所示的情况中,同一个订单中可能包含不同的产品,因此主键必须是“订单号”和“产品号”联合组成,

? 但可以发现,产品数量、产品折扣、产品价格与“订单号”和“产品号”都相关,但是订单金额和订单时间仅与“订单号”相关,与“产品号”无关,

? 这样就不满足第二范式的要求,调整如下,需分成两个表:

规范数据库设计规范数据库设计

1.3 第三范式(3NF)

? 满足第一范式和第二范式的前提下,第三范式需要确保数据表中的每一列数据都和主键直接相关,而不能间接相关

在2NF基础上,任何非主属性不依赖于其它非主属性(在2NF基础上消除传递依赖)

第三范式需要确保数据表中的每一列数据都和主键直接相关,而不能间接相关。

举例:

规范数据库设计

上表中,所有属性都完全依赖于学号,所以满足第二范式,但是“班主任性别”和“班主任年龄”直接依赖的是“班主任姓名”,

而不是主键“学号”,所以需做如下调整:

规范数据库设计

这样以来,就满足了第三范式的要求。

ps:如果把上表中的班主任姓名改成班主任教工号可能更确切,更符合实际情况,不过只要能理解就行。

2、JDBC

2.1 jdbc

SUN公司为了简化开发人员的(对数据库的统一-) 操作,提供了一个(ava操作数据库的)规范,俗称JDBC

java.sql javax.sql (默认就有)

还需导入一个数据库驱动包 mysql-connectorjava-5.1.47.jar

2.2 第一个JDBC程序

创建测试数据库

CREATE DATABASE `jdbcStudy` CHARACTER SET utf8 COLLATE utf8_general_ci;

USE jdbcStudy;

CREATE TABLE users(
 id INT PRIMARY KEY,
 NAME VARCHAR(40),
 PASSWORD VARCHAR(40),
 email VARCHAR(60),
 birthday DATE
);

 INSERT INTO users(id,NAME,PASSWORD,email,birthday)
VALUES(1,‘zhangsan‘,‘123456‘,‘zs@sina.com‘,‘1980-12-04‘),
(2,‘lisi‘,‘123456‘,‘lisi@sina.com‘,‘1981-12-04‘),
(3,‘wangwu‘,‘123456‘,‘wangwu@sina.com‘,‘1979-12-04‘);
  1. 创建一个普通项目

  2. 导入数据库驱动

  3. 编写测试代码

    package com.lantian.lesson01;
    
    import java.sql.*;
    
    public class jdbcFirst {
        public static void main(String[] args) throws ClassNotFoundException, SQLException {
            //1.加载驱动
            Class.forName("com.mysql.jdbc.Driver"); //固定写法, 加载驱动
            //2.用户信息和url
            //useUnicode=true&characterEncoding=utf8&yseSSL=true
            String url = "jdbc:mysql://localhost:3306/jdbcstudy?useUnicode=true&characterEncoding=utf8&yseSSL=true";
            String username = "root";
            String password = "123456";
            //3.连接成功,数据库对象
            Connection connection = DriverManager.getConnection(url, username, password);
            //4.执行SQL的对象 Statement 执行SQL的对象
            Statement statement = connection.createStatement();
            //5.执行SQL的对象 去执行SQL 可能存在结果,查看返回结果
            String sql = "SELECT * FROM users";
    
            ResultSet resultSet = statement.executeQuery(sql);//返回的结果集  结果集中封装了我们查询出来的全部结果
    
            while(resultSet.next()){
                System.out.println("id="+ resultSet.getObject("id"));
                System.out.println("name="+ resultSet.getObject("NAME"));
                System.out.println("pwd="+ resultSet.getObject("PASSWORD"));
                System.out.println("email="+ resultSet.getObject("email"));
                System.out.println("birthday="+ resultSet.getObject("birthday"));
                System.out.println("=======================================================");
            }
            //5.释放连接
            resultSet.close();
            statement.close();
            connection.close();
        }
    }
    
    

    步骤总结

    1. 加载驱动
    2. 连接数据库 DriverManager
    3. 获得执行sql的对象 Statement
    4. 获得返回的结果集
    5. 释放连接

2.3 jdbc对象解释

DriverManager

//1.加载驱动
        //DriverManager.registerDriver(new com.mysql.jdbc.Driver());
        Class.forName("com.mysql.jdbc.Driver"); //固定写法, 加载驱动

//connection 代表数据库
//数据库设置自动提交
//事务提交
//事务回滚
connection.rollback();
connection.commit();
connection.setAutoCommit();

URL

String url = "jdbc:mysql://localhost:3306/jdbcstudy?useUnicode=true&characterEncoding=utf8&yseSSL=true";

//mysql-->3306
//jdbc:mysql://主机地址:默认端口/数据库名?参数1&参数2&参数3...
//oralce-->1521
//jdbc:oralce:thin:@localhost:1521:sid

Statement 执行SQL的对象 PrepareStatement执行SQL的对象

         String sql = "SELECT * FROM users";	//编写SQL
       
		statement.executeQuery();   //查询操作返回 ResultSet
        statement.execute();        //执行任何SQL
        statement.executeUpdate();  //更新、插入、删除。都是用这个  返回一个受影响的行数。

ResultSet 查询的结果集:封装看所有的查询结果

获得指定的数据类型

resultSet.getobject(); //在不知道列类型的情况下使用
//如果知道列的类型就使用指定的类型
resultSet.getString() ;
resultset.getInt() ;
resu1 tSet.getF1oat();
resu1 tSet.getDate() ;
resu1tset.getobject();
...

遍历,指针

resultset.beforeFirst(); // 移动到最前面
resultSet.afterlast() ;// 移动到最后面
resultSet.next(); //移动到下一个数据
resultSet.previous(); //移动到前一行
resultSet.abso1ute(row); //移动到指定行

释放资源

//6、释放连接
resultSet.close();
statement.close();
connection.close();//耗资源,用完关掉

规范数据库设计

上一篇:搭建MySQL主从实现Django读写分离


下一篇:Mysql去重三种语句