MySQL---数据库从入门走向大神系列(十一)-Java获取数据库/结果集的元信息、将数据表写入excel表格

数据库的元信息:

首先介绍一下数据库的元信息(元数据):

元数据(Metadata)是关于数据的数据。

元数据是描述数据仓库内数据的结构和建立方法的数据。

存储的数据是什么类型,什么驱动等等,这些描述数据的数据,就是元数据!

准备:

package cn.hncu.pool3;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.sql.Connection;
import java.sql.DriverManager;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;

public class ConnsUtil {
    private static List<Connection> pool = new ArrayList<Connection>();
    private static final int NUM=3;
    static{
        try {
            //读取配置文件
            Properties p = new Properties();
            p.load(ConnsUtil.class.getClassLoader().getResourceAsStream("jdbc.properties"));
            String driver = p.getProperty("driver");
            String url = p.getProperty("url");
            String user = p.getProperty("username");
            String password = p.getProperty("password");
            Class.forName(driver);

            for(int i=0;i<NUM;i++){
                final Connection conn = DriverManager.getConnection(url, user, password);

                //只需要改这里就行了!
                //使用动态代理代理conn对象,实现对close方法的拦截
                Object obj = Proxy.newProxyInstance(
                        ConnsUtil.class.getClassLoader(),
                        conn.getClass().getInterfaces(),
                        new InvocationHandler() {

                            @Override
                            public Object invoke(Object proxy, Method method, Object[] args)
                                    throws Throwable {
                                if(method.getName().equalsIgnoreCase("close") && (args==null || args.length==0)){
                                    pool.add((Connection)proxy);
                                    return null;
                                }else{
                                    return method.invoke(conn, args);
                                }
                            }
                        });
                pool.add((Connection)obj);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static synchronized Connection getConn() throws Exception{
        if(pool.size()<=0){
            Thread.sleep(100);
            return getConn();
        }
        return pool.remove(0);
    }

}

使用java.sql 中的接口 DatabaseMetaData就可以实现:

演示类:

package cn.hncu.meta;

import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.Statement;

import org.junit.Test;

import cn.hncu.pool3.ConnsUtil;

public class MetaDataDemo {
    // ※元信息1:通过con获得DatabaseMetaData(数据库元信息)---数据库连接信息、数据库名、表名
    @Test
    public void databaseMetadataDemo() throws Exception {
        // 获取数据库的元信息
        Connection con = ConnsUtil.getConn();
        // ********关键
        DatabaseMetaData dm = con.getMetaData();

        // 获取此 JDBC 驱动程序的名称。
        System.out.println(dm.getDriverName());
        // 获取此 JDBC 驱动程序的主版本号。
        System.out.println(dm.getDriverMajorVersion());
        // 获取在此数据库中在同一时间内可处于开放状态的最大活动语句数。--返回结果为零意味着没有限制或限制是未知的
        System.out.println(dm.getMaxStatements());
        // 获取此驱动程序的主 JDBC 版本号。
        System.out.println(dm.getJDBCMajorVersion());
        // 还有很多方法,可以去API自己查
        System.out.println("=========================");

        // 下面是动态获取数据库名

        ResultSet rs = dm.getCatalogs();
        // 相当于执行:show databases;

        while (rs.next()) {
            System.out.println(rs.getString(1));
            // 进行元数据操作获得数据库名
        }

        // 知道数据库的名字
        con.createStatement().execute("use hncu");
        // 动态获取表名

        // 可以把参数null写成“%o%”进行模糊查询
        ResultSet rs2 = dm.getTables("hncu", "hncu", null,
                new String[] { "TABLE" });

        while (rs2.next()) {
            System.out.println(rs2.getString("TABLE_NAME"));
            // 进行元数据操作,获得表名
        }
    }

    // ※元信息2:通过rs获得ResultSetMetaData(结果集元信息)---表头(每个字段名)、表格行数、列数
    // 在知道数据库名和表名的情况下,把表头和表内容都查询出来。
    @Test
    // 站在结果集的高度---也就是表格
    public void resultSetMetaDataDemo() throws Exception{
        Connection con1 = ConnsUtil.getConn();
        Statement st = con1.createStatement();
        //如果是跨库查询,sql:“数据库名.表名”----select * from 数据库.表名
        String sql = "select * from stud";//我们的连接是hncu数据库的,访问hncu数据库直接写表名就可以
        ResultSet rs = st.executeQuery(sql);

        //结果集的元信息
        ResultSetMetaData rsmd = rs.getMetaData();
        int columns = rsmd.getColumnCount();
        //获得表格的列数

        //输出整个数据表(包括表头)
        //表头
        for(int i=0;i<columns;i++){
            String columnName = rsmd.getColumnName(i+1);
            System.out.print(columnName+"\t");
        }
        System.out.println();
        System.out.println("------------------------");
        //表内容
        while(rs.next()){
            for(int i=0;i<columns;i++){
                String content = rs.getString(i+1);
                System.out.print(content+"\t");
            }
            System.out.println();
        }
    }

}
getTables(String catalog, String schemaPattern, String tableNamePattern, String[] types)方法的参数解析:

catalog -类别名称:
它必须与存储在数据库中的类别名称匹配;该参数为 "" 表示获取没有类别的那些描述;为 null则表示该类别名称不应该用于缩小搜索范围 

schemaPattern - 模式名称的模式:
它必须与存储在数据库中的模式名称匹配;该参数为"" 表示获取没有模式的那些描述;为 null 则表示该模式名称不应该用于缩小搜索范围

tableNamePattern -表名称模式:
它必须与存储在数据库中的表名称匹配 

types - 要包括的表类型所组成的列表,必须取自从 getTableTypes()返回的表类型列表;null 表示返回所有类型

这样就遍历出来了。

将数据表写入excel表格

首先需要准备一个apache的Jar:

链接:

https://github.com/chenhaoxiang/Java

先创建一个简单的数据:

package cn.hncu.meta;

import java.io.FileOutputStream;
import java.io.IOException;

import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFRow;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.junit.Test;

public class ExportXls {

    @Test
    public void mkXlsDemo() throws IOException{
        //需求: 创建一个工作薄:a.xls, 工作表: 表1,  第4行第5列的单元格中写入文字:湖南城院
        HSSFWorkbook book = new HSSFWorkbook();
        HSSFSheet sheet = book.createSheet("表一");

        HSSFRow row4 = sheet.createRow(3);//行数为下标加1
        //该方法的参数值是从0开始的---真正的表格中的序号是从1开始标示

        HSSFCell cell5 = row4.createCell(4);

        FileOutputStream fout = new FileOutputStream("a.xls");
        book.write(fout);
    }
}

将数据库的所有表格数据遍历写入至excel表格

@Test
    public void exportTest() throws Exception{
        //这里我们只遍历存储hncu数据库
        String dbName="hncu";
        String xlsFileName="b.xls";
        exportDb2Xls(dbName,xlsFileName);
    }

    public void exportDb2Xls(String dbName,String xlsFileName) throws Exception{
        HSSFWorkbook book = new HSSFWorkbook();
        Connection con = ConnsUtil.getConn();
        DatabaseMetaData dm = con.getMetaData();

        //写代码时,尽量避免结果集套接操作,在一个结果集操作的内部进行其它结果集操作
        //如果有事务,一个结果集的回退或提交可能会波及另一个
        ResultSet rs = dm.getTables(dbName, dbName, null, new String[]{"TABLE"});

        List<String> tables = new ArrayList<String>();
        while(rs.next()){
            String tableName = rs.getString("TABLE_NAME");
            tables.add(tableName);
        }

        Statement st = con.createStatement();
        for(String tableName: tables){
            //创建一个表名为tableName的表
            HSSFSheet sheet = book.createSheet(tableName);
            //这样写,可以跨数据库访问
            String sql ="select * from "+dbName+"."+tableName;
            rs = st.executeQuery(sql);

            //把表头遍历出来且写到xls文件中
            HSSFRow row = sheet.createRow(0);//表头行
            ResultSetMetaData rsmd = rs.getMetaData();

            int colNum = rsmd.getColumnCount();
            for(int i=0;i<colNum;i++){
                HSSFCell cell = row.createCell(i);
                String colName = rsmd.getColumnName(i+1);
                cell.setCellValue(colName);
            }

            //所有数据行
            int idx = 1;
            while(rs.next()){
                row = sheet.createRow(idx++);
                for(int i=0;i<colNum;i++){
                    HSSFCell cell = row.createCell(i);
                    cell.setCellValue( rs.getString(i+1) );
                }
            }
        }

        FileOutputStream fout = new FileOutputStream(xlsFileName);

        book.write(fout);
    }

时间: 2024-08-07 17:00:25

MySQL---数据库从入门走向大神系列(十一)-Java获取数据库/结果集的元信息、将数据表写入excel表格的相关文章

MySQL---数据库从入门走向大神系列(七)-Java访问数据库配置及简单使用方法execute

从操作配置文件properties中读取连接字符串,通过该字符串进行数据连接,需要写三个文件其中,两个是java类,一个是后缀名为.properties的文件,该文件放在src工作目录下. 后缀为.properties的文件此处为其取名为jdbc.properties,其中的代码如下: ##MySQL driver=com.mysql.jdbc.Driver url=jdbc:mysql:///hncu?useUnicode=true&characterEncoding=utf-8 userna

MySQL---数据库从入门走向大神系列(九)-用Java向数据库读写大文本/二进制文件数据

介绍MySQL的文本和图形数据类型: Text 类型: 数据类型:描述 ------------------------------------------------------ char(size):保存固定长度的字符串(可包含字母.数字以及特殊字符).在括号中指定字符串的长度.最多 255 个字符. varchar(size):保存可变长度的字符串(可包含字母.数字以及特殊字符).在括号中指定字符串的最大长度.最多 255 个字符. 注释:如果值的长度大于 255,则被转换为 text类型

MySQL---数据库从入门走向大神系列(六)-事务处理与事务隔离(锁机制)

MySQL 事务处理 简单介绍事务处理: MySQL 事务主要用于处理操作量大,复杂度高的数据. 比如说,在人员管理系统中,你删除一个人员,你即需要删除人员的基本资料,也要删除和该人员相关的信息,如信箱,文章等等,这样,这些数据库操作语句就构成一个事务! 一个事务是一个连续的一组数据库操作,就好像它是一个单一的工作单元进行.换言之,永远不会是完整的事务,除非该组内的每个单独的操作是成功的.如果在事务的任何操作失败,则整个事务将失败. 事务处理可以用来维护数据库的完整性,保证成批的SQL语句要么全

MySQL---数据库从入门走向大神系列(八)-在java中执行MySQL的存储过程

http://blog.csdn.net/qq_26525215/article/details/52143733 在上面链接的博客中,写了如何用MySQL语句定义和执行存储过程 Java执行存储过程: 准备表stud: 列类型分别为: varchar,varchar,int 定义的存储过程分别为: p1:-无参 delimiter && create procedure p1() begin insert into stud values('P100','小李',43); select

MySQL---数据库从入门走向大神系列(十二)-构建MVC项目

这个是对前面技术的一个小总结吧,用到的大概技术有: MVC框架,加注解,Struts框架的思想,动态代理,线程管理对象ThreadLocal,Connection对象池,Properties文件读取,EL表达式,JSTL,JavaBean,Java访问MySQL数据库,增删改查- 其实做出来界面挺简单: 完整的项目链接: https://github.com/chenhaoxiang/Java/tree/master/myMvcWeb2 这里只写出一些核心的类的代码: Book.java: pa

MySQL---数据库从入门走向大神系列(十三)-BasicDataSource创建DataSource(DBCP连接池配置)

DBCP(DataBase connection pool),数据库连接池.是 apache 上的一个 java 连接池项目,也是 tomcat 使用的连接池组件.单独使用dbcp需要2个包:commons-dbcp.jar,commons-pool.jar由于建立数据库连接是一个非常耗时耗资源的行为,所以通过连接池预先同数据库建立一些连接,放在内存中,应用程序需要建立数据库连接时直接到连接池中申请一个就行,用完后再放回去. 首先,下载必须的jar包 dbcp包,目前版本是2.1.1 : htt

MySQL---数据库从入门走向大神系列(十六)-JavaWeb分页技术实例演示1

分页,是一种将所有数据分段展示给用户的技术.用户每次看到的不 是全部数据,而是其中的一部分,如果在其中没有找到自己想要的内容,用户可以通过指定页码或是点上/下一页的方式进行翻页. 本例演示静态分页,也就是先设置好每页显示10行,再根据总行数,来算出总页数,将所有页数的页号都显示出来. 相关算法(技术): 总行数(num): select count(1) from stud; 每页显示的行数(n): 固定值---已知的一个常量 页数: pageSize= num/n +( (num%n==0)?

MySQL---数据库从入门走向大神系列(十)-Connection对象池、装饰模式与动态代理模式

问题概述: 之前本系列博客写的,全部都是一个connection对象,不知道大家发现没有,我们既然做了一个Connection工具类,那么大家肯定都是从那里面拿Connection对象的,之前的如果是多线程运行,很容易出问题的,你想想事务处理就知道了,同时用事务处理操作同一个Connection,肯定会出问题的. 例如: 一方的事务在提交的时候,你正好运行了一个事务中的一个操作,那么你这个操作也会被提交,而且你后面的提交或回滚失效的,如果对方把Connection关闭了,你的程序还会挂. 等等问

MySQL---数据库从入门走向大神系列(十五)-Apache的DBUtils框架使用

DBUtils简介: commons-dbutils 是 Apache 组织提供的一个开源 JDBC工具类库,它是对JDBC的简单封装,学习成本极低,并且使用dbutils能极大简化jdbc编码的工作量,同时也不会影响程序的性能.因此dbutils成为很多不喜欢hibernate的公司的首选(嗯~商业竞争-..). commons-dbutilsAPI介绍: org.apache.commons.dbutils.QueryRunner org.apache.commons.dbutils.Res