搜公众号
推荐 原创 视频 Java开发 开发工具 Python开发 Kotlin开发 Ruby开发 .NET开发 服务器运维 开放平台 架构师 大数据 云计算 人工智能 开发语言 其它开发 iOS开发 前端开发 JavaScript开发 Android开发 PHP开发 数据库
Lambda在线 > 陕西兄弟连IT > Java 面试秘典 第16课

Java 面试秘典 第16课

陕西兄弟连IT 2019-02-11
举报

1、用JDBC如何调用存储过程

代码如下:

package com.huawei.interview.lym;

 

import java.sql.CallableStatement;

import java.sql.Connection;

import java.sql.DriverManager;

import java.sql.SQLException;

import java.sql.Types;

 

public class JdbcTest {

 

    /**

     * @param args

     */

    public static void main(String[] args) {

       // TODO Auto-generated method stub

       Connection cn = null;

       CallableStatement cstmt = null;   

       try {

           //这里最好不要这么干,因为驱动名写死在程序中了

           Class.forName("com.mysql.jdbc.Driver");

           //实际项目中,这里应用DataSource数据,如果用框架,

           //这个数据源不需要我们编码创建,我们只需Datasource ds = context.lookup()

           //cn = ds.getConnection();        

           cn = DriverManager.getConnection("jdbc:mysql:///test","root","root");

           cstmt = cn.prepareCall("{callinsert_Student(?,?,?)}");

           cstmt.registerOutParameter(3,Types.INTEGER);

           cstmt.setString(1, "wangwu");

           cstmt.setInt(2, 25);

           cstmt.execute();

           //get第几个,不同的数据库不一样,建议不写

           System.out.println(cstmt.getString(3));

       } catch (Exception e) {

           // TODO Auto-generated catch block

           e.printStackTrace();

       }

       finally

       {

 

           /*try{cstmt.close();}catch(Exception e){}

           try{cn.close();}catch(Exception e){}*/

           try {

              if(cstmt != null)

                  cstmt.close();

              if(cn != null)             

                  cn.close();

           } catch (SQLException e) {

              // TODO Auto-generated catch block

              e.printStackTrace();

           }

       }

    }

2、JDBC中的PreparedStatement相比Statement的好处

答:一个sql命令发给服务器去执行的步骤为:语法检查,语义分析,编译成内部指令,缓存指令,执行指令等过程。

select * from studentwhere id =3----缓存--àxxxxx二进制命令

select * from studentwhere id =3----直接取-àxxxxx二进制命令

select * from studentwhere id =4--- -à会怎么干?

如果当初是select * from student where id =?--- -à又会怎么干?

 上面说的是性能提高

可以防止sql注入。

3、Class.forName的作用?为什么要用?

答:按参数中指定的字符串形式的类名去搜索并加载相应的类,如果该类字节码已经被加载过,则返回代表该字节码的Class实例对象,否则,按类加载器的委托机制去搜索和加载该类,如果所有的类加载器都无法加载到该类,则抛出ClassNotFoundException。加载完这个Class字节码后,接着就可以使用Class字节码的newInstance方法去创建该类的实例对象了。

有时候,我们程序中所有使用的具体类名在设计时(即开发时)无法确定,只有程序运行时才能确定,这时候就需要使用Class.forName去动态加载该类,这个类名通常是在配置文件中配置的,例如,spring的ioc中每次依赖注入的具体类就是这样配置的,jdbc的驱动类名通常也是通过配置文件来配置的,以便在产品交付使用后不用修改源程序就可以更换驱动类名。

4、大数据量下的分页解决方法。

答:最好的办法是利用sql语句进行分页,这样每次查询出的结果集中就只包含某页的数据内容。再sql语句无法实现分页的情况下,可以考虑对大的结果集通过游标定位方式来获取某页的数据。

sql语句分页,不同的数据库下的分页方案各不一样,下面是主流的三种数据库的分页sql:

sql server:

       String sql =

       "select top " + pageSize +" * from students where id not in" +

 

 "(select top " + pageSize *(pageNumber-1) + " id from students order by id)" +

 

 "order by id";

 

mysql:

 

       String sql =

       "select * from students order by idlimit " + pageSize*(pageNumber-1) + "," + pageSize;

      

oracle:

 

       String sql =

        "select * from " + 

        (select *,rownum rid from (select * fromstudents order by postime desc) where rid<=" + pagesize*pagenumber +") as t" +

        "where t>" +pageSize*(pageNumber-1);

5、用 JDBC 查询学生成绩单, 把主要代码写出来(考试概率极大).

Connection cn = null;

PreparedStatementpstmt =null;

Resultset rs = null;

try

{

       Class.forname(driveClassName);

       cn = DriverManager.getConnection(url,username,password);

       pstmt = cn.prepareStatement(“select  score.* from score ,student “ +

              “where score.stuId = student.idand student.name = ?”);

       pstmt.setString(1,studentName);

       Resultset rs = pstmt.executeQuery();

       while(rs.next())

       {

              system.out.println(rs.getInt(“subject”)  +  “    ” + rs.getFloat(“score”) );

       }

}catch(Exceptione){e.printStackTrace();}

finally

{

       if(rs != null) try{ rs.close()}catch(exception e){}

       if(pstmt != null)try{pstmt.close()}catch(exception e){}

       if(cn != null) try{ cn.close()}catch(exception e){}

}

 

 

6、这段代码有什么不足之处?

try {
Connection conn = ...;
Statement stmt = ...;

ResultSet rs = stmt.executeQuery("select * fromtable1");

while(rs.next()) {

}
} catch(Exception ex) {
}

答:没有finally语句来关闭各个对象,另外,使用finally之后,要把变量的定义放在try语句块的外面,以便在try语句块之外的finally块中仍可以访问这些变量。

 

7、说出数据连接池的工作机制是什么?

J2EE服务器启动时会建立一定数量的池连接,并一直维持不少于此数目的池连接。客户端程序需要连接时,池驱动程序会返回一个未使用的池连接并将其表记为忙。如果当前没有空闲连接,池驱动程序就新建一定数量的连接,新建连接的数量有配置参数决定。当使用的池连接调用完成后,池驱动程序将此连接表记为空闲,其他调用就可以使用这个连接。

实现方式,返回的Connection是原始Connection的代理,代理Connection的close方法不是真正关连接,而是把它代理的Connection对象还回到连接池中。

 

8、为什么要用 ORM?  和 JDBC 有何不一样?

orm是一种思想,就是把object转变成数据库中的记录,或者把数据库中的记录转变成objecdt,我们可以用jdbc来实现这种思想,其实,如果我们的项目是严格按照oop方式编写的话,我们的jdbc程序不管是有意还是无意,就已经在实现orm的工作了。

现在有许多orm工具,它们底层调用jdbc来实现了orm工作,我们直接使用这些工具,就省去了直接使用jdbc的繁琐细节,提高了开发效率,现在用的较多的orm工具是hibernate。也听说一些其他orm工具,如toplink,ojb等。

💗

推荐阅读

1、

2、

3、


- End -


版权声明:本站内容全部来自于腾讯微信公众号,属第三方自助推荐收录。《Java 面试秘典 第16课》的版权归原作者「陕西兄弟连IT」所有,文章言论观点不代表Lambda在线的观点, Lambda在线不承担任何法律责任。如需删除可联系QQ:516101458

文章来源: 阅读原文

相关阅读

关注陕西兄弟连IT微信公众号

陕西兄弟连IT微信公众号:shanxixdlIT

陕西兄弟连IT

手机扫描上方二维码即可关注陕西兄弟连IT微信公众号

陕西兄弟连IT最新文章

精品公众号随机推荐

举报