• 2022-8-18 第七小组 学习日记 (day42)JDBC+练习题


    目录

    聚合函数(早上)

    数值型函数

    字符串型函数

    日期和时间函数

    获取时间和日期

    时间戳和日期转换函数

    根据日期获取年月日的数值

    时间日期的计算

    加密函数

    流程控制函数

    数据库设计

    三范式

    第一范式

    第二范式

    第三范式

    常见的表关系

    一对一

    一对多

    多对多

    JDBC(下午)

    JDBC是什么?

     数据的持久化:

    JDBC编程需要:

    1.数据库驱动:

    2.url地址:

    3.数据库的用户名:

    4.数据库的密码:

    导入jar包

    使用属性配置文件的好处:

    正题:执行数据库的DML语言---增删改(没有查哟)在java中有三个接口分别定义了对数据库的调用的不同的方式:

    JAVA上机题3道(晚上)

    题目一:从控制台中读入一个文件名,判断该文件是否存在你的某个盘下。如果该文件存在,则在原文件相同路径下创建一个文件名为“copy_原文件名”的新文件,该文件内容为原文件的拷贝。

    ​编辑题目二:随机生成10个百位以内数字并存入集合中(任意集合均可,自行决定)。利用自己擅长的方式对其10个数字进行排序,排序后按小到大方式输出。看到这个题,首先一个知识点:随机数、第二个存入到集合里、第三个进行排序,(我这里用的是冒泡排序法)

     题目三:随机生成100个,10位字符串。字符由a-z,A-Z,0-9随机组成。并将生成的100个字符串存入容器中(提示:数组,集合)。假定从键盘输入一一个字符串,判断这个字符串是否在生成的100字符中。并给出友好提示。

     总结


    聚合函数(早上)

    ——count:计数。count(*)≈count(1)>count(主键)

                    count(*):MySQL对count(*)底层优化,count(0)。

                    count(1)

                    count(主键)

                    count(字段)

    ——min:最小值

    ——max:最大值

    ——sum:求和

    ——avg:平均值

    数值型函数

    主要是对数值型进行处理。

    ——ceiling(x):向上取整

    ——floor(x):向下取整

    ——round(x):四舍五入

    ——truncate(x,y):返回数字x截断为y位小数的结果

    ——PI:圆周率,π

    ——rand:返回0到1的随机数

    ——abs:绝对值

    1. -- 绝对值
    2. select ABS(-4) 4的绝对值,ABS(-1.1);
    3. -- 向下取整,向上取整,四舍五入
    4. select CEILING(4.1),FLOOR(1.1),ROUND(-4.4)
    5. -- 取余
    6. select MOD(60,11);
    7. -- 随机数
    8. select RAND(),RAND(),RAND()
    9. -- 截断
    10. select TRUNCATE(2.33999999,2);

    字符串型函数

    对字符串进行处理。

    ——length(s):字符串的长度

    ——concat(s1,s2,.....sn):合并字符串

    ——lower(str):将字母转成小写

    ——upper(str):将字母转成大写

    ——left(str,x):返回字符串str的左边的x个字符

    ——right(str,x):返回字符串str右边的x个字符

    ——trim:去掉左右两边的空格

    ——replace:替换

    ——substring:截取

    ——reverse:反转

    1. select LEFT('abcdefg',2);
    2. select RIGHT('abcdefg',2);
    3. select REVERSE('hijklmn');
    4. select REPLACE('abcdefg','abc','x');

    日期和时间函数

    date,time,datetime,timestamp,year。

    获取时间和日期

    ——【curdate】和【current_date】,返回当前的系统日期。

    ——【curtime】和【current_time】,返回当前的系统时间。

    ——【now】和【sysdate】,返回当前的系统时间和日期。

    1. select CURRENT_DATE();
    2. select CURTIME();
    3. select now();

    时间戳和日期转换函数

    【UNIX_TIMESTAMP】获取unix时间戳函数

    【FROM_UNIXTIME】将时间戳转换为时间格式

    1. select UNIX_TIMESTAMP();
    2. select FROM_UNIXTIME(1660785720);

    根据日期获取年月日的数值

    1. select MONTH(SYSDATE());
    2. select MONTHNAME(SYSDATE());
    3. select DAYNAME(SYSDATE());
    4. select DAYOFWEEK(SYSDATE());
    5. select WEEK(SYSDATE());
    6. select DAYOFMONTH(SYSDATE());
    7. select YEAR(SYSDATE());

    时间日期的计算

    1. -- 日期加法
    2. select DATE_ADD(SYSDATE(),INTERVAL 70 DAY);
    3. -- 日期减法
    4. select DATE_SUB(SYSDATE(),INTERVAL 10 DAY);
    5. -- 时间间隔
    6. select DATEDIFF('2023-01-01',SYSDATE());
    7. -- 日期格式化
    8. select DATE_FORMAT(SYSDATE(),'%W %M %D %Y');

    加密函数

    1. -- 把传入的参数的字符串按照md5算法进行加密,得到一个32位的16进制的字符串
    2. select MD5('123456');

    SQL存储过程实例

    md5算法是不可逆的。

    流程控制函数

    可以进行条件判断,用来实现SQL语句的逻辑。

    ——if(test,t,f):如果test是真,则返回t,否则返回f

    ——ifnull(arg1,arg2):如果arg1不是空,返回arg1,否则返回arg2

    ——nullif(arg1,arg2):如果arg1=arg2返回null,否则返回arg1

    1. select IF(2 > 1,'a','b');
    2. select IFNULL(sal,0);
    3. select NULLIF(age,0);

    对一系列的值进行判断:

    1. -- 输出学生的各科的成绩,以及评级,60以下D,60-70是C,71-80是B,80以上是A
    2. SELECT
    3.     *,
    4. CASE
    5.         
    6.         WHEN score < 60 THEN 'D' WHEN score >= 60 
    7.         AND score < 70 THEN 'C' WHEN score >= 70 
    8.             AND score < 80 THEN 'B' WHEN score >= 80 THEN
    9.                 'A' 
    10.             END AS '评级' 
    11.     FROM
    12.     mystudent;
    13. -- 行转列
    14. SELECT
    15.     user_name,
    16.     max( CASE course WHEN '数学' THEN score ELSE 0 END ) '数学',
    17.     max( CASE course WHEN '语文' THEN score ELSE 0 END ) '语文',
    18.     max( CASE course WHEN '英语' THEN score ELSE 0 END ) '英语' 
    19. FROM
    20.     mystudent 
    21. GROUP BY
    22.     user_name

    数据库设计

    三范式

    • 第一范式:要求有主键,并且要求每一个字段的原子性不能再分。

    • 第二范式:要求所有的非主键字段完全依赖主键,不能产生部分依赖

    • 第三范式:所有非主键字段和主键字段之间不能产生传递依赖。

    第一范式

    不符合第一范式表结构:

    符合第一范式的表结构: 

    idname邮箱手机号
    1001aaaaaa@163.com12321321321
    1002bbbbbb@163.com32132654654
    1003cccccc@163.com45654654654

    必须有主键,这是数据库设计的基本要求,一般情况下我们采用数值型或定长字符串,列不能再分,比如:联系方式。

    关于第一范式,保证每一行的数据是唯一,每个表必须有主键。

    第二范式

    建立在第一范式的基础上,要求所有非主键字段完全依赖于主键,不能产生部分依赖。

    学号性别姓名课程编号课程名称教室成绩
    1001a2001java30189
    1002b2002mysql30290
    1003c2003html30391
    1004d2004python30452
    1005e2005c++30567
    1006f2006c#30684

    解决方案:

    学生表:学号是主键

    学号性别姓名
    1001a
    1002b
    1003c
    1004d
    1005e
    1006f

     安工大—数据库概论实验报告书

     课程表:课程编号是主键

    课程编号课程名称教室
    2001java301
    2002mysql302
    2003html303
    2004python304
    2005c++305
    2006c#306

     成绩表:学号和课程编号为联合主键

    学号课程编号成绩
    1001200189
    1002200290
    1003200391
    1004200452
    1005200567
    1006200684

    第三范式

    建立在第二范式基础上,非主键字段不能传递依赖于主键字段。

    不满足第三范式:

    学号姓名课程编号课程名称
    1001a2001java
    1002b2002mysql
    1003c2003html
    1004d2004python
    1005e2005c++
    1006f2006c#

    解决方案:

    学生表:学号是主键

    学号姓名课程编号
    1001a2001
    1002b2002
    1003c2003
    1004d2004
    1005e2005
    1006f2006

     课程表:课程编号是主键

    课程编号课程名称
    2001java
    2002mysql
    2003html
    2004python
    2005c++
    2006c#

    常见的表关系

    一对一

    学生信息表分为基本信息表和信息信息表。

    • 分为两张表,共享主键。

    • 分两张表,用外键连接。

    一对多

    两张表,外键在多的一方。

    • 分两张表存储,在多的一方加外键

    • 这个外键字段引用是一的一方的主键

    多对多

    • 分三张表存储,在学生表存储学生信息,在课程表存储课程信息。

    • 在成绩表中存储学生和课程的对应关系。

    JDBC(下午)

    JDBC是什么?

    JDBC指的是Java数据库连接,是一种标准Java应用编程接口,用来连接java编程和广泛的数据库

     数据的持久化:

    把数据永久的保存起来。主要的方式还是得存到硬盘上。 持久化的实现过程大部分是通过数据库来完成。

    JDBC编程需要:

    1.数据库驱动:

            所有连接数据库都要实现的最大接口:java.sql.Driver

            如何加载驱动?

            通过反射:class.forName("com.mysql.jdbc.Driver");

                              mysql:com.mysql.jdbc.Driver

                              oracle:oracle.jdbc.driver.OracleDriver

    2.url地址:

            需要链接数据库,其实就是链接一个服务器。

            我们之前链接服务器的时候用的是IP地址+端口号,但是针对于连接数据库的话,有一个专门的协议:jdbc协议:

            jdbc协议:jdbc:mysql://主机名称(主机地址):端口号/数据库的名字

          (这是我的地址)eg:jdbc:mysql://127.0.0.1:3306/wy?useUnicode=true&characterEncoding=utf8

    3.数据库的用户名:

    4.数据库的密码:

    导入jar包

    链接数据库需要导入外部依赖,需要引入mysql的驱动(指的是引入的jar包)

    jar包:点击进入官网

    我这里用的是5.1.17的,这里不推荐用8.0的,因为8.0变样了,而且非常的私有化,然后还需要解放私有化,非常麻烦,最关键的是,我不会。

    我们首先需要导入jar包:在项目下建立一个文件夹:lib

    然后把jar包放到文件夹里,之后右键点击add a liabrary

     以上的都是准备工作,我们来真正链接数据库:

    获取以上的内容

    1. String url="jdbc:mysql://127.0.0.1:3306/wy?useUnicode=true&characterEncoding=utf8";
    2. String username="root";//你自己的用户名字
    3. String password="root";//你自己的用户密码
    4. //驱动的全类名
    5. String driverName="com.mysql.jdbc.Driver";

    加载驱动类:

    Class clazz = Class.forName(driverName);

    实例化Driver对象

    Driver driver=(Driver)clazz.getDeclaredConstructor().newInstance();

    注册驱动

    DriverManager.registerDriver(driver);

    获取链接

    Connection connection=DriverManager.getConnection(url,username,password);

    接下来可以用Objects的nonNull的方法来判断是否连接成功了:

    System.out.println(Objects.nonNull(connection)?"数据库连接成功":"数据库连接失败");


    这下步骤太麻烦,而且在代码中,不易改变,所以我们用封装+外部文件获取驱动的方式,封装一个方法:

    获取JDBC连接的方法 

    jdbc.properties文件:注意,是文件,读取外部的properties属性文件,用外部文件获取驱动

    1. mysql.url=jdbc:mysql://127.0.0.1:3306/wy?useUnicode=true&characterEncoding=utf8
    2. mysql.username=root
    3. mysql.password=020323
    4. mysql.driverName=com.mysql.jdbc.Driver

    方法:

    1. public static Connection connection(){
    2.         Properties properties = new Properties();
    3.         Connection connection=null;
    4.         try {
    5.             properties.load(Test01.class.getClassLoader().getResourceAsStream("jdbc.properties"));
    6.             String url = properties.getProperty("mysql.url");
    7.             String driverName = properties.getProperty("mysql.driverName");
    8.             String username = properties.getProperty("mysql.username");
    9.             String password = properties.getProperty("mysql.password");
    10.             Class.forName(driverName);//加载器驱动类(可以省略实例化对象和注册驱动的过程)
    11.             connection = DriverManager.getConnection(url, username, password);
    12.         } catch (SQLException e) {
    13.             throw new RuntimeException(e);
    14.         } catch (IOException e) {
    15.             throw new RuntimeException(e);
    16.         } catch (ClassNotFoundException e) {
    17.             throw new RuntimeException(e);
    18.         }
    19.         return connection;
    20.     }

    使用属性配置文件的好处:

    1.实现了代码和数据的分离,如果需要修改配置信息,直接在属性文件中修改即可。不需要深入代码

    2.如果修改了配置信息,省去了编译的过程

    3.符合OCP原则

    正题:执行数据库的DML语言---增删改(没有查哟)
    在java中有三个接口分别定义了对数据库的调用的不同的方式:

    1.Statement:用来执行静态sql语句并返回它所生成的结果对象

    2.PreparedStatement:预编译,可以使用此对象多次高效的执行该语句

    3.CallableStatement:用来执行sql的存储过程//不学

    我们每次开启数据库连接,相对应的也要关闭连接:我们也封装一个工具类:

    1. public static void close(Connection conn, Statement statement){
    2.         if(Objects.nonNull(statement)){
    3.             try {
    4.                 statement.close();
    5.             } catch (SQLException e) {
    6.                 throw new RuntimeException(e);
    7.             }
    8.         }
    9.         if(Objects.nonNull(conn)){
    10.             try {
    11.                 conn.close();
    12.             } catch (SQLException e) {
    13.                 throw new RuntimeException(e);
    14.             }
    15.         }
    16.     }

     首先建立连接、创建Statement对象

    1.     Connection conn =null;
    2.         Statement statement=null;
    3.         try {
    4.             conn = JDBCUtil.connection();
    5.             statement = conn.createStatement();
    6.             //增加
    7.             String sql="INSERT INTO teacher (name) VALUES ('法外狂徒')";
    8. //            //返回值是执行sql语句影响的行数
    9.             //修改
    10. //            String sql="UPDATE teacher set name='jay'";
    11.             //删除
    12. //            String sql="DELETE FROM teacher where name='jay'";
    13.             int i = statement.executeUpdate(sql);
    14.             System.out.println(i);
    15.             System.out.println("操作成功!!!");
    16.         } catch (SQLException e) {
    17.             throw new RuntimeException(e);
    18.         }finally {
    19.             JDBCUtil.close(conn,statement);
    20.         }

    那么我们要进行查询操作该怎么做呢:

    首先获取连接,获取可以执行sql语句的statement对象,定义sql,执行查询的sql,会得到一个ResultSet,ResultSet就是封装了查询结果的一个对象,遍历结果集

    1.     @Test
    2.     public void test01() {
    3.         // 1.获取连接
    4.         Connection conn = JDBCUtil.getConnection();
    5.         Statement stmt = null;
    6.         ResultSet rs = null;
    7.         try {
    8.             // 2.获取可以执行sql语句的stmt对象
    9.             stmt = conn.createStatement();
    10.             // 3.定义sql
    11.             String sql = "select id tid,name tname from teacher where id = 7";
    12.             // 4.执行查询的sql,会得到一个ResultSet
    13.             // ResultSet就是封装了查询结果的一个对象
    14.             rs = stmt.executeQuery(sql);
    15.             // 5.遍历结果集ResultSet
    16.             while(rs.next()) {
    17.                 int id = rs.getInt("tid");
    18.                 String name = rs.getString("tname");
    19.                 System.out.println("id:" + id + ",name:" + name);
    20.             }
    21.         } catch (SQLException e) {
    22.             throw new RuntimeException(e);
    23.         } finally {
    24.             // 6.关闭资源
    25.             JDBCUtil.close(conn,stmt,rs);
    26.         }
    27.     }

     关闭资源的方法:

    1. public static void close(Connection conn, Statement stmt, ResultSet rs) {
    2.  
    3.         if(Objects.nonNull(stmt)) {
    4.             try {
    5.                 stmt.close();
    6.             } catch (SQLException e) {
    7.                 throw new RuntimeException(e);
    8.             }
    9.         }
    10.         if(Objects.nonNull(conn)){
    11.             try {
    12.                 conn.close();
    13.             } catch (SQLException e) {
    14.                 throw new RuntimeException(e);
    15.             }
    16.         }
    17.         if(Objects.nonNull(rs)){
    18.             try {
    19.                 rs.close();
    20.             } catch (SQLException e) {
    21.                 throw new RuntimeException(e);
    22.             }
    23.         }
    24.     }

    JAVA上机题3道(晚上)

    题目一:
    从控制台中读入一个文件名,判断该文件是否存在你的某个盘下。如果该文件存在,则在原文件相同路径下创建一个文件名为“copy_原文件名”的新文件,该文件内容为原文件的拷贝。

    首先肯定是要判断该文件名到底存不存在:

    我们定义一个方法:用来判断是否存在该文件,如果存在,获取父路径并拼接用来创建文件,如果不存在则打印“你输入的文件不存在”

    1. private static File createFile(String fileName) throws IOException {
    2.         File file = new File(fileName);
    3.         if (file.exists()) {
    4.             //获取父路径并拼接
    5.             File copyFile = new File(file.getParent() + "/copy_" + file.getName());
    6.             //复制后的路径
    7.             System.out.println(copyFile);
    8.             //创建文件
    9.             copyFile.createNewFile();
    10.             //返回创建的文件
    11.             return copyFile;
    12.         } else {
    13.             System.out.println("你输入的文件不存在");
    14.         }
    15.         return null;
    16.     }
    17. }

    文件创建完成后,进行传输文件内容工作:

    判断是否创建完成,我这里用是否为空来判断:

    文件传输我这里没有解决异常(try...catch)而是选择抛异常的方式:

    1. private static boolean copyFile(String fileName) throws IOException {
    2.         File copyFile = createFile(fileName);
    3.         //判断拷贝是否为空
    4.         if (copyFile == null) {
    5.             return false;
    6.         }
    7.         InputStream inputStream = new FileInputStream(fileName);//读源文件
    8.         OutputStream outputStream = new FileOutputStream(copyFile);//写到复制出的文件中
    9.         int len = 0;
    10.         while ((len = inputStream.read()) != -1) {
    11.             outputStream.write(len);
    12.         }
    13.         inputStream.close();
    14.         outputStream.close();
    15.         return true;
    16.     }

    主方法:

    1.  public static void main(String[] args) {
    2.         Scanner sc = new Scanner(System.in);
    3.         System.out.println("请输入一个文件名:");
    4.         String fileName = "C:/summerjava/" + sc.next();
    5.         boolean flag = false;
    6.         try {
    7.             flag = copyFile(fileName);
    8.         } catch (IOException e) {
    9.             throw new RuntimeException(e);
    10.         }
    11.         System.out.println(flag);//创建成功
    12.     }


    题目二:
    随机生成10个百位以内数字并存入集合中(任意集合均可,自行决定)。利用自己擅长的方式对其10个数字进行排序,排序后按小到大方式输出。
    看到这个题,首先一个知识点:随机数、第二个存入到集合里、第三个进行排序,(我这里用的是冒泡排序法)

     随机生成百位以内数字:

    1. int ran=new Random().nextInt(100)
    2. List list = new ArrayList<>();
    3.         for (int i = 0; i <10 ; i++) {
    4.           list.add(new Random().nextInt(100));
    5.         }//生成的10个随机数字,并存入到集合中
    6.         System.out.println("随机生成的:"+list);
    7.         for (int i = 0; i
    8.             for (int j = i; j
    9.                 if(list.get(i)>=list.get(j)){
    10.                     int temp=list.get(i);
    11.                     list.set(i,list.get(j));
    12.                     list.set(j,temp);
    13.                 }
    14.             }
    15.         }
    16.         System.out.println("排序后"+list);

     题目三:
    随机生成100个,10位字符串。字符由a-z,A-Z,0-9随机组成。并将生成的100个字符串存入容器中(提示:数组,集合)。假定从键盘输入一一个字符串,判断这个字符串是否在生成的100字符中。并给出友好提示。

    我首先定义一个方法用来生成十位随机字符串:

    1. public static String random(){
    2.     String chars = "qwertyuioplkjhgfdsazxcvbnm0123456789QWERTYUIOPLKJHGFDSAZXCVBNM";
    3.     StringBuffer value = new StringBuffer();
    4.     for (int i = 0; i < 10; i++) {
    5.         value.append(chars.charAt((int)(Math.random() * 62)));
    6.     }
    7.     return value.toString();  
    8. }
    9.         Scanner sc = new Scanner(System.in);
    10.         System.out.println("请输入一个字符串");
    11.         String str = sc.next();
    12.         Boolean flag = false;
    13.         String [] arr =new String[100];
    14.         for (int i = 0; i
    15.             arr[i]=random();
    16.             if(str==arr[i]){
    17.                 flag=true;
    18.             }
    19.             //System.out.println(arr[i]);//可以打印看看字符串都有什么
    20.         }
    21.         System.out.println("您输入的字符串:"+str+(flag?"在":"不在")+"随机生成的字符中");

     

     总结

            今天学习了JDBC,这是一个全新的内容,对我来说相当难,它的作用就是将前面的Mysql和java相结合,Mysql进行数据存储,java则对数据库中的数据进行增删改操作 ,代码量太大看不懂

  • 相关阅读:
    HTML 超链接 a 标签
    数据挖掘神经网络算法,人工神经网络分析方法
    【EMQX 5.0】2.2.4 Authentication 认证
    分页条件搜索
    计算机外设:显示器是如何工作的?
    聪明的红绿灯,已经学会主动给你开路了
    FDbus NDK以及QNX交叉编译
    项目实战第三十三讲:标准中心-属性体系
    聊聊神经网络的优化算法
    HTML5期末大作业【红色的电影售票平台网站】web前端 html+css+javascript网页设计实例 企业网站制作
  • 原文地址:https://blog.csdn.net/weixin_56982770/article/details/126404278