Java程序员面试分类真题7(总分:100.00,做题时间:120分钟)一、单项选择题(总题数:30,分数:60.00)1.
以下可以对对象加互斥锁的关键字是______。
(分数:2.00)
A.synchronized
√
B.serialize
C.volatile
D.static解析:对于选项A,synchronized(同步的)是Java语言的关键字,主要用来给对象和方法或者代码块加锁,当它锁定一个方法或者一个代码块的时候,同一时刻最多只有一个线程执行这段代码。当两个并发线程访问同一个对象中的这个加锁同步代码块时,同一时间只能有一个线程执行。所以,选项A正确。
对于选项B,serialize是序列化的意思,所谓对象的序列化指的是把对象转换为字节序列的过程,所谓对象的反序列化指的是把字节序列恢复为对象的过程。通常,对象的序列化主要有以下两种用途:①把对象的字节序列永久地保存到硬盘上,通常存放在一个文件中;②在网络上传送对象的字节序列。在Java语言中,序列化通过Serializable接口来实现。所以,选项B不正确。
对于选项C,在由Java语言编写的程序中,有时候为了提高程序的运行效率,编译器会做一些优化操作,把经常被访问的变量缓存起来,程序在读取这个变量的时候有可能会直接从寄存器中来读取这个值,而不会去内存中读取。这样做的一个好处是提高了程序的运行效率,但当遇到多线程编程时,变量的值可能被其他线程改变了,而该缓存的值不会做相应的改变,从而造成应用程序读取的值和实际的变量值不一致。关键字volatile正好能够解决这一问题,被关键字volatile修饰的变量编译器不会做优化,每次都会从内存中读取。所以,选项C不正确。
对于选项D,关键字static主要有以下两种作用:第一,为某特定数据类型或对象分配单一的存储空间,而与创建对象的个数无关;第二,希望某个方法或属性与类而不是对象关联在一起,也就是说,在不创建对象的情况下就可以通过类来直接调用方法或使用类的属性。总之,被static修饰的属性(方法)是类的属性(方法),不属于任何对象。所以,选项D不正确。2.
JavaTbread中的方法resume()负责恢复哪些线程的执行?______
(分数:2.00)
A.通过调用wait()方法而停止运行的线程
B.通过调用sleep()方法而停止运行的线程
C.通过调用stop()方法而停止运行的线程
D.通过调用suspend()方法而停止运行的线程
√解析:对于选项A,wait()方法是一种使线程暂停执行的方法,例如,当线程交互时,如果线程对一个同步对象发出了一个wait()调用请求,那么该线程会暂停执行,被调对象进入等待状态,直到被唤醒(通常使用notify方法唤醒)或等待时间超时。所以,选项A错误。
对于选项B,sleep()方法的作用是使当前运行的线程休眠指定的时间。所以,选项B错误。
对于选项C,可以使用stop()方法来终止线程的执行。当使用Tbread.stop()方法来终止线程时,它会释放已经锁定的所有的监视资源。如果当前任何一个受这些监视资源保护的对象处于一个不一致的状态,其他的线程将会看到这个不一致的状态,这可能会导致程序执行的不确定性,并且这种问题很难被定位。因此,不推荐使用。所以,选项C错误。
对于选项D,suspend()方法就是将一个线程挂起(暂停),并且不会自动恢复,必须通过调用对应的resume()方法,才能使得线程重新进入可执行状态。所以,选项D正确。3.
在一个线程中,sleep(100)方法将使得该线程在______后获得对CPU的控制(假设睡眠过程中不会有其他事件唤醒该线程)。
(分数:2.00)
A.正好100ms
B.100ms不到
C.≥100ms
√
D.不一定解析:Tbread.sleep(longmillis)和Tbread.sleep(longmillis,intnanos)静态方法强制当前正在执行的线程休眠(即暂停执行),当线程睡眠时,它睡在某个地方,在苏醒之前不会返回到可运行状态。当睡眠时间到期,则返回到可运行状态。所以,sleep()方法指定的时间为线程不会运行的最短时间。当线程休眠时间结束后,会返回到可运行状态,注意不是运行状态,如果要到运行状态还需要等待CPU调度执行。因此,sleep()方法不能保证该线程睡眠到期后就开始执行。所以,选项C正确。
引申:sleep与wait有什么区别?
sleep()是使线程暂停执行一段时间的方法。wait()也是一种使线程暂停执行的方法,例如,当线程交互时,如果线程对一个同步对象x发出一个wait()调用请求,那么该线程会暂停执行,被调对象进入等待状态,直到被唤醒或等待时间超时。
具体而言,sleep与wait的区别主要表现在以下几个方面:
1)原理不同。sleep是Thread类的静态方法,是线程用来控制自身流程的,它会使此线程暂停执行指定时间,而把执行机会让给其他线程,等到计时时间到时,此线程会自动苏醒。例如,当线程执行报时功能时,每一秒钟打印出一个时间,那么此时就需要在打印方法前面加上一个sleep方法,以便让自己每隔一秒执行一次,该过程如同闹钟一样。而wait是Object类的方法,用于线程间的通信,这个方法会使当前拥有该对象锁的进程等待,直到其他线程调用notify方法(或notifyAll方法)时才醒来,不过开发人员也可以给它指定一个时间,自动醒来。与wait配套的方法还有notify和notifyAll。
2)对锁的处理机制不同。由于sleep方法的主要作用是让线程休眠指定的一段时间,在时间到时自动恢复,不涉及线程间的通信,因此,调用sleep方法并不会释放锁。而wait方法则不同,当调用wait方法后,线程会释放掉它所占用的锁,从而使线程所在对象中的其他synchronized数据可被其他线程使用。举个简单例子,在小明拿遥控器期间,他可以用自己的sleep方法每隔十分钟调一次电视台,而在他调台休息的十分钟期问,遥控器还在他的手上。
3)使用区域不同。由于wait方法的特殊意义,所以,它必须放在同步控制方法或者同步语句块中使用,而sleep方法则可以放在任何地方使用。
4)sleep方法必须捕获异常,而wait、notify以及notifyall不需要捕获异常。在sleep的过程中,有可能被其他对象调用它的interrupt(),产生InterruptedException异常。
由于sleep不会释放“锁标志”,容易导致死锁问题的发生,所以,一般情况下,不推荐使用sleep方法,而推荐使用wait方法。4.
下面不是mead类的方法是______。
(分数:2.00)
A.run!()
B.start()
C.exit()
√
D.stop()解析:在Java语言中,Tharead类位于java.lang命名空间下。Thread类主要用于创建并控制线程、设置线程优先级并获取其状态。
本题中,对于选项A,要实现多线程,在继承了Thread类后必须实现run()方法,也就是说,线程的核心逻辑都存在于run()方法中。这个方法被start()方法调用来实现多线程的功能,如果直接调用run()方法,那么就与调用普通的方法类似。所以,选项A错误。
对于选项B,Tbread类提供了一个start()方法,该方法的功能是让这个线程开始执行,当开始执行后,JVM将会调用这个线程的run()方法来执行这个线程的任务。所以,选项B错误。
对于选项C,Tbread类没有exit()这个方法。所以,选项C正确。
对于选项D,Thread类的stop()方法是用来停止一个线程的,但是由于这个方法不是线程安全的,因此,通常不推荐使用。所以,选项D错误。5.
有如下代码:
publicstaticvoidmain(Stringargs[])
{
Threadt=DewThread()
{
publicvoidrun()
{
world();
}
};
t.run();
System.out.print("hello");
}
staticvoidworld()
{
System.out.print("world");
}
上面程序的运行结果是______。
(分数:2.00)
A.helloworld
B.worldhello
√
C.A和B都有可能
D.都不输出解析:Thread类提供了一个start()方法,这个方法的功能是让这个线程开始执行,当开始执行后,JVM将会调用这个线程的run()方法来执行这个线程的任务。如果直接调用run()方法就与调用普通的方法类似。
对于本题而言,首先调用t.run()方法,输出“world”,等调用结束后才会执行System.out.print("hello")语句,输出“hello”。所以,选项B正确。
如果把t.run()改成t.start(),在调用t.start()方法后不需要等这个线程结束,这个方法就会立即返回,然后执行语句System.out.print("Hllo"),在这种情况下,这两个输出语句的执行顺序是无法保证的,任何一个语句都有可能先执行,因此,答案就是选项C。6.
有如下代码:
publicclassTestextendsThread
{
publicstaticvoidmain(Stringargv[])
{
Testb=newTest();
b.run();
}
publicvoidstart()
{
for(inti=0;i<10;i++)
{
System.out.println("Valueofi="+i);
}
}
}
当编译并运行上面程序时,输出结果是______。
(分数:2.00)
A.编译错误,指明run方法没有定义
B.运行错误,指明run方法没有定义
C.编译通过并输出0到9
D.编译通过但无输出
√解析:在Java语言中,可以采用以下两种方法来创建线程:继承Thread类与实现Runnable接口。其中,在使用Runnable接口时,需要建立一个Thread实例。所以,无论是通过Thread类创建线程还是通过Runnable接口创建线程,都必须建立Thread类或它的子类的实例。
Thread类提供了一个start()方法,该方法的功能是让这个线程开始执行,当这个线程开始执行后,JVM将会调用这个线程的run()方法来执行这个线程的任务。在实现多线程时,在继承了Thread类后必须实现run()方法,也就是说,线程的核心逻辑都存在于run()方法中,这个方法被start()方法调用来实现多线程的功能,如果直接调用run()方法,就与调用普通的方法类似。
对于本题而言,Test类继承了Thread类,但是没有重写Thread类的run()方法,因此,b.run()实际上调用的是Tbread类的run()方法,而Thread类的run()方法的方法体为空,故这个程序能编译通过,但是没有输出结果。所以,选项D正确。7.
以下可以启动一个线程的方法是______。
(分数:2.00)
A.start()
√
B.ran()
C.begin()
D.notify()解析:Thread类提供了一个start方法,这个方法的功能是让这个线程开始执行,开始执行后,JVM将会调用这个线程的run方法来执行这个线程的任务。在实现多线程的时候,在继承了Thread方法后必须实现run方法,也就是说,线程的核心逻辑都存在于run方法中。这个方法被start方法调用来实现多线程的功能,如果直接调用run方法就与调用普通的方法类似。所以,选项A正确,选项B错误。
对于选项C,在Java语言中,线程是没有begin方法的。所以,选项C错误。
对于选项D,notify方法是用来唤醒一个线程的,而不是启动一个线程。所以,选项D错误。8.
对于Java语言中的Daemon线程,setDaemon设置必须要______。
(分数:2.00)
A.在start之后
B.在start之前
√
C.前后都可以
D.前后都不可以解析:Java语言提供了两种线程:守护线程(DaemonThread)与用户线程。守护线程又称为“服务进程”“精灵线程”或“后台线程”,是指当程序运行的时候,在后台提供一种通用服务的线程,这种线程并不属于程序中不可或缺的部分。通俗点讲,任何一个守护线程都是整个JVM中所有非守护线程的保姆。
用户线程和守护线程几乎一样,唯一的不同之处就在于:如果用户线程已经全部退出运行,只剩下守护线程存在,JVM也就退出了。因为当所有的非守护线程结束时,没有了被守护者,守护线程也就没有工作可做,也就没有继续运行程序的必要了,程序也就终止,同时会杀死所有守护线程。也就是说,只要有任何非守护线程还在运行,程序就不会终止。
在Java语言中,守护线程一般具有较低的优先级,它并非只由JVM内部提供,用户在编写程序时也可以自己设置守护线程。例如,将一个用户线程设置为守护线程的方法就是在调用start启动线程之前调用对象的setDaemon(true)方法,如果将以上参数设置为false,则表示的是用户进程模式。需要注意的是,当在一个守护线程中产生了其他线程,那么这些新产生的线程默认还是守护线程,用户线程也是如此。所以,选项B正确。9.
当线程1使用某个对象,而此对象又需要线程2修改后才能符合线程1的需要,这时线程1就要等待线程2完成修改工作,这种现象称为______。
(分数:2.00)
A.线程的同步
√
B.线程的调度
C.线程的就绪
D.线程的互斥解析:对于选项A,同步就是协同步调,按预定的先后次序进行运行。例如你说完,我再说。该定义正好与题目描述一致。所以,选项A正确。
对于选项B,假设计算机的CPU在任意时刻都只能执行一条机器指令,线程只有获得CPU的使用权才能执行对应的操作,多线程的并发运行原理为:各个线程轮流获得CPU的使用权,来执行对应的操作。线程调度是指按照特定机制为多个线程分配CPU的使用权。所以,选项B错误。
对于选项C,线程有多个状态,就绪是指这个线程已经有了除CPU外所有的资源,在等待获取CPU,一旦获取到CPU的控制权就可以立即执行。所以,选项C错误。
对于选项D,线程互斥是指某一资源同时只允许一个访问者对其进行访问,具有唯一性和排它性。但互斥无法限制访问者对资源的访问顺序,即访问是无序的。所以,选项D错误。10.
有如下代码:
publicclassXextendsThreadimplementsRunable
{
publicvoidrun()
{
System.out.println("thisisrun()");
}
publicstaticvoidmain(Stringargs[])
{
Threadt=newThread(newX());
t.start();
}
}
程序的运行结果为______。
(分数:2.00)
A.第一行会产生编译错误
B.第六行会产生编译错误
C.第六行会产生运行错误
D.程序会运行和启动
√解析:11.
可以对对象加互斥锁的关键字是______。
(分数:2.00)
A.transient
B.synchronized
√
C.serialize
D.static解析:12.
下列可以终止当前线程的运行的是______。
(分数:2.00)
A.当抛出一个异常时
√
B.当该线程调用sleep()方法时
C.当创建一个新线程时
D.当一个优先级高的线程进入就绪状态时解析:13.
GC线程______守护线程。
(分数:2.00)
A.是
√
B.不是
C.不确定解析:Java语言提供了两种线程:守护线程与用户线程。守护线程又被称为“服务进程”“精灵线程”或“后台线程”,是指当程序运行时,在后台提供一种通用服务的线程,这种线程并不属于程序中不可或缺的部分。通俗点讲,任何一个守护线程都是整个JVM中所有非守护线程的保姆。
用户线程和守护线程几乎一样,唯一的不同之处就在于如果用户线程已经全部退出运行,只剩下守护线程存在,Java虚拟机也就退出了。因为当所有的非守护线程结束时,没有了被守护者,守护线程也就没有工作可做,也就没有继续运行程序的必要了,程序也就终止,同时会杀死所有守护线程。也就是说,只要有任何非守护线程还在运行,程序就不会终止。
在Java语言中,守护线程一般具有较低的优先级,它并非只由JVM内部提供,用户在编写程序时,也可以自己设置守护线程。例如,将一个用户线程设置为守护线程的方法就是在调用start方法启动线程之前调用对象的setDaemon(true)方法,如果将以上参数设置为false,则表示的是用户进程模式。需要注意的是,当在一个守护线程中产生了其他线程,那么这些新产生的线程默认还是守护线程,用户线程也是如此。
守护线程的一个典型的例子就是垃圾回收器。只要JVM启动,它就始终在运行,实时监控和管理系统中可以被回收的资源。
所以,选项A正确。14.
事务隔离级别是由______实现的。
(分数:2.00)
A.Hibernate
B.Java应用程序
C.数据库系统
√
D.JDBC驱动程序解析:对于选项A,Hibemate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,使得Java程序员可以随心所欲地使用对象编程思维来操纵数据库。所以,选项A不正确。
对于选项B,Java应用程序可以通过JDBC或Hiberhate对数据库系统进行访问。虽然JDBC和Hibernate都提供了事务控制的接口,但这些接口只是把事务控制相关的命令发送给数据库系统,由数据库系统来控制事务的隔离级别。所以,选项B不正确。
对于选项C,数据库系统是为适应数据处理的需要而发展起来的一种较为理想的数据处理系统,也是一个为实际可运行的存储、维护和应用系统提供数据的软件系统,是存储介质、处理对象和管理系统的集合体。在数据库操作中,为了保证在并发情况下数据读写的正确性,提出了事务隔离级别。在标准SQL规范中,定义了4个事务隔离级别,分别为未授权读取,也称为读未提交(ReadUncommitted);授权读取,也称为读提交(ReadCommitted);可重复读取(RepeatableRead);序列化(Seiializable)。所以,事务隔离级别是由数据库系统实现的。所以,选项C正确。
对于选项D,JDBC驱动程序是一种用于执行SQL语句的JavaAPI,可以为多种关系数据库提供统一访问。所以,选项D不正确。15.
下列选项中,提供了Java存取数据库能力的包是______。
(分数:2.00)
A.java.sql
√
B.java.swing
C.java.io
D.java.awt解析:包(package)是Java语言提供的一种区别类的命名空间的机制,它是类的组织方式,是一组相关类和接口的集合,提供了访问权限和命名的管理机制。
在Java语言中,包的作用主要体现在以下三个方面:
1)方便查找与使用。将功能相近的类放在同一个包中,可以方便查找与使用。
2)避免命名冲突。由于在不同包中可以存在同名类,所以,使用包的好处是在一定程度上可以避免命名冲突。
3)访问权限设定。某次访问权限是以包为单位的。
本题中,对于选项A,java.sql包中主要包含一些访问数据库相关的接口。所以,选项A正确。
对于选项B,java.swing是一个用于开发Java应用程序用户界面的开发工具包。所以,选项B错误。
对于选项C,java.io提供了流处理的相关接口。所以,选项C错误。
对于选项D,java.awt包含用于创建用户界面和绘制图形图像的所有类。所以,选项D错误。16.
在JDBC中,用于表示数据库连接的对象是______。
(分数:2.00)
A.Statement
B.Connection
√
C.PreparedStatement
D.DriverManager解析:JDBC(JavaDataBaseConnectivity,Java数据库连接)用于在Java程序中实现数据库操作功能,它提供了执行SQL语句、访问各种数据库的方法,并为各种不同的数据库提供统一的操作接口,java.sql包中包含了JDBC操作数据库的所有类。通过JDBC访问数据库一般有如下几个步骤:
1)加载JDBC驱动器。将数据库的JDBC驱动加载到classpath中,在基于JavaEE的Web应用开发过程中,通常要把目标数据库产品的JDBC驱动复制到WEB-INF/lib下。
2)加载JDBC驱动,并将其注册到DriverManager中。一般使用反射Class.forName(StringdriveNamel。
3)建立数据库连接,取得Connection对象。一般通过DriverManager.getConnection(url,username,passwd)方式实现,其中,url表示连接数据库的字符串,username表示连接数据库的用户名,passwd表示连接数据库的密码。
4)建立Statement对象或者PreparedStatement对象。
5)通过Statement或PreparedStatement对象执行SQL语句。
6)访问结果集ResultSet对象。
7)依次将ResultSet、Statement、PreparedStatement、Connection等对象关闭,释放掉所占用的资源。
通过上述分析可知,选项B正确。17.
在Java语言中,用于调用存储过程的对象是______。
(分数:2.00)
A.DriverManager
B.ResultSet
C.CallableStatemet
√
D.PreparedStatement解析:存储过程(StoredProcedure)是在大型数据库系统中,一组为了完成特定功能的SQL语句集,存储在数据库中经过第一次编译后再次调用不需要再次编译,用户通过指定存储过程的名字并给出参数(如果该存储过程带有参数)来执行它。存储过程是数据库中的一个重要对象,任何一个设计良好的数据库应用程序都应该用到存储过程。
本题中,对于选项A,DriverManager是JDBC的管理层,它作用于用户和驱动程序之间,管理一组JDBC驱动程序的基本服务,提供用来建立数据库连接的一系列方法。除此以外,DriverManager还处理诸如驱动程序登录时间限制以及登录与跟踪消息的显示等事务。因此,选项A错误。
对于选项B,ResultSet表示查询的结果集,通常通过执行查询数据库的语句生成。ResultSet对象具有指向其当前数据行的光标。最初,光标被置于第一行之前,next方法将光标移动到下一行,因为该方法在ResultSet对象没有下一行时返回false,所以,可以在while循环中使用它来迭代结果集。因此,选项B错误。
对于选项C,CallableStatement提供了用来调用数据库中存储过程的接口,如果有输出参数要注册,说明是输出参数。因此,选项C正确。
对于选项D,PreparedStatement表示预编译的SQL语句的对象,用于执行带参数的预编译SQL语句。因此,选项D错误。18.
下面不是标准的Statement类的是______。
(分数:2.00)
A.Statement
B.CallableStatement
C.PreparedStatement
D.BatchedStatement
√解析:本题中,对于选项A,Statement是Java语言执行数据库操作的一个重要方法,用于在已经建立数据库连接的基础上,向数据库发送要执行的SQL语句,并返回它所生成结果的对象,每次执行SQL语句时,数据库都要编译该SQL语句。以下是一个最简单的SQL语句:
Statementstmt=conn.getStatement();
stmt.executeUpdate("insertintoclientvalues('aa','aaaa')");
所以,选项A错误。
对于选项B,CallableStatement提供了用来调用数据库中存储过程的接口。所以,选项B错误。
对于选项C,PreparedStatement表示预编译的SQL语句的对象,用于执行带参数的预编译SQL语句。所以,选项C错误。
对于选项D,不存在BatchedStatement方法。所以,选项D正确。19.
以下获取ResutlSet对象rst的第一行数据的方法中,正确的是______。
(分数:2.00)
A.rst.hashNext();
B.rst.next();
√
C.rst.first();
D.rst.nextRow();解析:ResultSet(结果集)是数据中查询结果返回的一种对象,通常通过执行查询数据库的语句生成。ResultSet对象不仅具有存储功能,还具有操纵数据的功能。
ResultSet对象具有指向其当前数据行的指针,它包含了符合SQL语句查询结果的所有行,提供了一套get方法(访问当前行的不同列)用于对这些行中数据进行访问。ResultSet的next方法用于移动到ResultSet中的下一行,使下一行成为当前行。因此,对于刚获取到的ResultSet对象,第一次调用next方法就会得到第一行的数据。所以,选项B正确。
所以,本题的答案为B。20.
为了让浏览器以UTF-8编码显示JSP页面,下列JSP代码正确的是______。
(分数:2.00)
A.<%pagecontentType=
B.<metahttp-equiv=
C.把所有的输出内容重新编码:newSTRing(content.getBytes())
D.response.setContentType()
√解析:选项A设置的是服务器端编码,选项B设置的是客户端编码,选项C设置的是某个字符串的编码。只有选项D满足题意。21.
Servlet处理请求的方式为______。
(分数:2.00)
A.以程序的方式
B.以进程的方式
C.以线程的方式
√
D.以响应的方式解析:SeiMet是采用Java语言编写的服务器端程序,运行于Web服务器的Servlet容器中,其主要功能是提供请求/响应的Web服务模式,可以生成动态的Web内容,工作原理如图所示。
Servlet工作原理
Servlet处理客户端请求通常有如下几个步骤:
1)用户通过单击一个链接来向Servlet发起请求。
2)Web服务器接收到该请求后,会把该请求提交给相应的容器来处理,当容器发现这是对Servlet发起的请求后,容器此时会创建两个对象:HttpServletResponse和HttpServletRequest。
3)容器可以根据请求消息中的URL消息找到对应的Servlet,然后针对该请求创建一个单独的线程,同时把第2)步中创建的两个对象以参数的形式传递到新创建的线程中。
4)容器调用Servlel的service()方法来完成对用户请求的响应,service()方法会调用doPost()方法或doGet()方法来完成具体的响应任务,同时把生成的动态页面返回给容器。
5)容器把响应消息组装成HTTP格式返回给客户端。此时,这个线程运行结束,同时删除第2)步创建的两个对象.HttpServletResponse和HttpServletRequest。
容器会针对每次请求创建一个新的线程进行处理,同时,会针对每次请求创建HttpServletResponse和HttpServletRequest两个对象,处理完成后,线程也就退出了。所以,选项C正确。22.
按照MVC设计模式,JSP用于实现______。
(分数:2.00)
A.Controller(控制器)
B.View(视图)
√
C.Model(模型)
D.Database(数据库)解析:使用JSP与Servlet实现的MVC模型如图所示。
MVC模型
在这个MVC模型中,视图模块采用JSP来实现,主要负责数据的展现,视图可以从控制器上获取模型的状态,当然不是直接从控制器上获取到的,而是控制器把模型的数据放到一个视图可以访问的地方,通过这种间接的方式来访问模型的数据。
控制器使用Servlet来实现,客户端的所有请求都发送给Servlet,它接受请求,并根据请求消息把它们分发给对应的JSP页面来响应,同时根据需求生成JavaBean实例供JSP来使用。
模型采用JavaBean来实现的,这个模块实现了实际的业务逻辑。
从以上分析可知,选项B正确。23.
在JSP叶旨令中,isELIgnored="boolean"的意思是______。
(分数:2.00)
A.决定该页面是否是一个错误处理页面
B.决定是否实现setvlet的单线程模式
C.决定是否支持EL表示
√
D.没有具体的含义解析:isELIgnored属性用来指定该JSP文件是否支持EL(ExpressionLanguage,表达式语言)表达式。如果值为true,那么对于类似于${..}这样的内容,直接会原样输出,而不会进行EL表达式运算;如果值为false,那么表示EL表达式不会被忽略,该EL表达式将会被执行。其属性配置语法格式如下:<%@pageisELIgnored="true|false"%>。所以,选项C正确。24.
在WEB-INF目录下,必须存放的文件为______。
(分数:2.00)
A.class文件
B.web.xml
√
C.html文件
D.jar文件解析:WEB-INF是Web应用的安全目录。所谓安全目录就是客户端无法访问,只有服务端可以访问的目录。如果想在页面中直接访问其中的文件,必须通过web.xml文件对要访问的文件进行相应的映射才行。WEB-INF文件夹下除了web.xml外,还存在一个classes文件夹,用以放置*.class文件,这些*.class文件是设计人员编写的类库,实现了JSP页面前台与后台服务的分离,使得网站的维护非常方便。web.xml文件为网站部署描述XML文件,它对网站的部署非常重要。因此,web.xml是必不可少的文件。所以,选项B正确。25.
在JavaScript中,以下验证一个数据是否是数字的描述中,正确的是______。
(分数:2.00)
A.imtI=value若报错就不是数字
B.如果用Integer.parseInt(yalue)有误就不是数字
C.没有方法验证
D.利用isNaN(value)返回的boolean进行判断
√解析:对于选项A,JavaScript是弱类型语言(也称为弱类型定义语言,与强类型定义相反。弱类型语言允许将一块内存看作多种类型,比如直接将整型变量与字符变量相加。C/C++是静态语言,是强类型语言;Perl与PHP是动态语言,但也是弱类型语言),只有一种类型var。所以,选项A错误。
对于选项B,Integer.parseInt(value)是Java语言中的方法,而不是JavaScript的方法。所以,选项B错误。
对于选项C,JavaScript中验证一个数据是否是数字是存在方法的,可以使用isNaN()函数判断,也可以使用正则表达式判断。所以,选项C错误。
对于选项D,JavaScript提供了一个isNaN()函数用于检查其参数是否是非数字值。所以,选项D正确。26.
以下不能作JSP的服务器的是______。
(分数:2.00)
A.JBoss
B.BEAWebLogic
C.Tomcat
D.PWS
√解析:Web服务器指的是提供Web功能的服务器,主要就是HTTP服务器,包括图片的下载等一系列和文本相关的资源。Web服务器支持以HTTP协议的方式来访问,当Web服务器接收到一个HTTP请求时,它同样会以HTTP协议格式返回一个响应,这个响应可以是一个静态的HTML页面,也可以是结果处理的一个动态的页面,还可以是音频、视频等信息。为了处理一个请求,Web服务器可以做出一个响应,并进行页面跳转,或者把动态响应的产生委托给一些其他的程序,例如CGl脚本、JSP、Servlet或者一些其他的服务器端程序。Web服务器一般都使用了一些特有的机制(例如容错机制)来保证Web服务器有较好的扩展性和不问断地提供服务。常见的Web服务器有IIS和Apache。
应用服务器提供访问业务逻辑的途径以供客户端应用程序使用。具体而言,它通过HTTP、TCP/IP、IIOP(InternetInter-ORBProtocol,互联网内部对象请求代理协议)或JRMP(JavaRemoteMethodProtocol,Java远程方法协议)等来提供业务逻辑接口。为了系统的可靠性,同样使用了一些可扩展性和容错机制。除此之外,它还为应用的开发提供了许多服务,例如事务管理、安全管理和对象生命周期管理等。常见的应用服务器有BEAWebLogicServer、IBMWebSphereApplicationServer、IPlanetApplicationServer、Oracle9iApplicationServer、JBoss和Tomcat等。
Web服务器一般是通用的,而应用服务器一般是专用的,例如Tomcat只处理Java应用程序而不能处理ASPX或PHP。需要注意的是,Web服务器与应用服务器是并列关系,二者不存在相互包容关系。在使用的时候,如果访问的页面只有HTML,用Web服务器就足够了,但是如果是JSP,此时就需要应用服务器,因为只有应用服务器才能解析JSP里的Java代码,并将解析结果以HTML的格式返回给用户。
从上面的分析可以看出,选项A、选项B和选项C都可以作为JSP的服务器。
对于选项D,PWS(PersonalWebSetwer,个人Web服务器)是微软开发的个人网站服务器,主要应用于解决个人信息共享和Web开发。它是一个桌面形的Web服务器,使用它可以自动创建个性化主页,以拖放的方式发布文档,在它的帮助下,用户可以快速简便地进行Web站点设置。由于它只是一个Web服务器,因此,它无法作为JSP的服务器。所以,选项D正确。27.
以下不是JSP操作指令的是______。
(分数:2.00)
A.setProperty
B.include
C.forward
D.import
√解析:JSP总共有6个操作指令:jsp:include、jsp:useBean、jsp:setProperty、jsp:getProperty、jsp:forward与jsp:plugin。以下将分别对这几种指令进行介绍。
jsp:include:用来在页面被请求的时候引入一个文件。使用示例如下:
<jsp:includepage="test.jsp"flush="true">
<jsp:paramname="name"value="value"/>
</jsp:include>
以上代码表示在当前文件中可以引入test.jsp文件。
jsp:useBean:用来寻找或者实例化一个JavaBean。它使得开发人员既可以发挥Java组件重用的优势,同时也避免了损失JSP区别于Servlet的方便性。使用示例如下:
<jsp:useBean
id="car"scope="session"class="com.Car">
以上代码表示实例化了一个com.Car类的实例。
jsp:setProperty:用来设置已经实例化的Bean对象的属性。使用示例如下:
<jsp:setPropertyname="car"property="colour"value="red"/>
以上代码用来设置名字为car的实例的colour属性为red。
jsp:getProperty:用来获取某个JavaBean的属性。使用示例如下:
Colom=<jsp:getPropertyname="car"property="colour"></jsp:getProperty>
以上代码用来获取名字为car的实例的colour属性。
jsp:foward:用来把请求转到一个新页面。使用示例如下:
<jsp:forwardpage="/Servlet/login"/>
以上代码把当前页面重定向到/Servlet/login来处理。
jsp:plugin:用于在浏览器中播放或显示一个对象。使用这个动作能插入所需的特定浏览器的OBJECT或EMBED元素来指定浏览器运行一个JAVAApplet所需的插件。使用示例如下:
<jsp:plugintype="applet"codebase="/ch5"code="Hello.class"height="40"width="320">
以上代码用来在浏览器中运行一个applet插件。
由此可见,选项D中的import不是JSP的操作指令。所以,选项D正确。28.
在配置tomcat虚拟目录时,需要打开的文件是______。
(分数:2.00)
A.A.web.xml
B.B.index.jsp
C.C.server.xml
√
D.D.以上都不是
E.解析:一般情况下,配置虚拟目录的方法为:
在tomcatconf下server.xml中找到<Hostname="localhost"appBase="webapps"unpackWARs="true"autoDeploy="true"xmlValidation="false"xmlNamespaceAware="false"></Host>,在其中添加:<Contextpath=""docBase="自定义目录"reloadable="true"></Context>。因此,选项C正确。
对于选项A,web.xml可以被看作是JSP的一个配置文件,其中一个重要的作用是用来配置Servlet的路径。所以,选项A错误。
对于选项B,index.jsp一般是一个网站的首页,不包含与tomcat相关的配置信息。所以,选项B错误。29.
下面不是表单标记的是______。
(分数:2.00)
A.RADIO
B.INPUT
C.CHECKBOX
D.TR
√解析:本题中,对于选项A,RADIO为单选按钮控件标签,是表单标记。所以,选项A错误。
对于选项B,INPUT表示Form表单中的一种输入对象,其又随Type类型的不同而分为文本输入框、密码输入框、单选/复选框及提交/重置按钮等,是表单标记。所以,选项B错误。
对于选项C,CHECKBOX为多选复选框标签,是表单标记。所以,选项C错误。
对于选项D,TR是表中的行标签,不属于表单标记。所以,选项D正确。30.
下面不是response对象的方法的是______。
(分数:2.00)
A.addCookie(Cookiecookie)
B.setHeader(Stringheademarne,Stringheadervalue)
C.getParameter(Stringstr)
√
D.sendError(interrorcode)解析:response对象所提供的方法有如下几类:
(1)设定响应头的方法
voidaddCookie(Cookiecookie)
新增cookie
voidaddDateHeader(Stringname,longdate)
新增long类型的值到name响应头
voidaddHeader(Stringname,Stringvalue)
新增String类型的值到name响应头
voidaddIntHeader(Stringname,intvahe)
新增int类型的值到name响应头
voidsetDateHeader(Stringname,longdate)
指定long类型的值到name响应头
voidsetHeader(Stringname,Stringvalue)
指定String类型的值到name响应头
voidsetIntHeader(Stringname,intvalue)
指定int类型的值到name响应头
booleancontainsHeader
判断指定名字的HTTP文件头是否已经存在,然
后返回真假部尔值
(2)设定响应状态码的方法
voidsendError(intsc)
传送状态码(StatusCode)
voidsendError(intsc,Stringmsg)
传送状态码和错误信息
voidsetStatus(intsc)
设定状态码
(3)用来URL重写(Rewriting)的方法
StringencodeRedirectURL(Stringurl)
对使用sendRedirect()方法的URL予以编码
(4)设置重定向
sendRedirect()
设置重定向页面
(5)设置不同浏览器对应的数据
setContentType(StringcontentTypestr):使客户端浏览器,区分不同种类的数据,并根据不同的MIME(MultipurposeInternetMailExtensions,多用途互联网邮件扩展类型)调用浏览器内不同的程序嵌入模块来处理相应的数据。
所以,选项C正确。二、多项选择题(总题数:6,分数:12.00)1.
以下布局管理器中,使用的是组件的最佳尺寸(PreferredSize)的是______。
(分数:2.00)
A.FlowLayout
√
B.BorderLayout
C.GridLayout
D.CardLayout
E.GridBagLayout
√解析:2.
以下关于线程的描述中,错误的是______。
(分数:2.00)
A.sleep方法必须写在同步方法或同步块中
√
B.wait方法执行时会释放对象锁
C.sleep方法执行时会释放对象锁
√
D.wait方法必须写在同步方法或同步块中解析:3.
Tbread类中本身的方法(不包括继承)有______。
(分数:2.00)
A.start()
√
B.sleep(longmi)
√
C.notify()
D.wait()解析:对于选项A,start方法是Thread类中比较重要的方法,JVM通过调用这个方法启动一个线程。所以,选项A正确。
对于选项B,sleep是使线程暂停执行一段时间的方法。所以,选项B正确。
对于选项C和选项D,wait和notify都是从Object类继承的方法。所以,选项C和选项D错误。4.
在Java语言中,如果需要编写一个多线程程序,可以使用的方法是______。
(分数:2.00)
A.实现Runnable接口
√
B.扩展类Thread
√
C.扩展类Runnable
D.实现接口Thread解析:Java多线程实现常用的有两种方法:继承Thread类与实现Runnable接口。所以,选项A与选项B正确。5.
下列可用于创建一个可运行的类的方法有______。
(分数:2.00)
A.publicclassXimplementsRunnable{publicvoidrun(){}}
√
B.publicclassXimplementsThread{publicvoidrun(){}}
C.publicclassXimplementsRunnable{publicintrun(){}}
D.publicclassXimplementsRunnable{protectedvoidrun(){}}
√
E.publicclassXimplementsThread{publicvoidrun(){}}解析:6.
下面可以在任何时候被任何线程调用的方法有______。
(分数:2.00)
A.notify()
B.wait()
C.notifyall()
D.sleep()
√
E.yield()
√
F.synchronized(this)
√解析:三、论述题(总题数:13,分数:28.00)1.
实现多线程的方法有哪几种?
(分数:4.00)__________________________________________________________________________________________
正确答案:(Java虚拟机(JavaVirtualMachine,JVM,是运行所有Java程序的抽象计算机,是Java语言的运行环境)允许应用程序并发地运行多个线程。在Java语言中,多线程的实现一般有以下三种方法:
1)实现Runnable接口,并实现该接口的run()方法。以下是主要步骤:
①自定义类并实现Runnable接口,实现run()方法。
②创建Thread对象,用实现Runnable接口的对象作为参数实例化该Thread对象。
③调用Thread的start()方法。
classMyThreadimplementsRunnable
{//创建线程类
publicvoidrun()
{
System.out.println("Threadbody");
}
}
publicclassTest
{
publicstaticvoidmain(String[]args)
{
MyThreadthread=newMyThread();
Threadt=newThread(thread);
t.start();
//开启线程
}
2)继承Thread类,重写run方法。Thread本质上也是实现了Runnable接口的一个实例,它代表一个线程的实例,并且,启动线程的唯一方法就是通过Thread类的start()方法。start()方法是一个native(本地)方法,它将启动一个新线程,并执行run()方法(Thread中提供的run()方法是一个空方法)。这种方式通过自定义类直接extendsThread,并重写run()方法,就可以启动新线程并执行自己定义的run()方法。需要注意的是,当start()方法调用后并不是立即执行多线程代码,而是使得该线程变为可运行态(Runnable),什么时候运行多线程代码是由操作系统决定的。
下例给出了Thread的使用方法。
classMyThreadextendsThread
{//创建线程类
publicvoidrun()
{
System.out.println("Threadbody");
//线程的方法体
}
}
publicclassTest
{
publicstaticvoidmain(String[]args)
{
MyThreadthread=newMyThread();
thread.start();
//开启线程
}
3)实现Callable接口,重写call()方法。Callable对象实际是属于Executor框架中的功能类,Callable接口与Runnable接口类似,但是提供了比Runnable更强大的功能,主要表现为以下三点:
①Callable可以在任务结束后提供一个返回值,Runnable无法提供这个功能。
②Callable中的call()方法可以抛出异常,而Runnable的run()方法不能抛出异常。
③运行Callable可以拿到一个Future对象,Future对象表示异步计算的结果。它提供了检查计算是否完成的方法。由于线程属于异步计算模型,所以无法从其他线程中得到方法的返回值,在这种情况下,就可以使用Future来监视目标线程调用call()方法的情况,当调用Future的get()方法以获取结果时,当前线程就会阻塞,直到call()方法结束返回结果。
示例代码如下:
importjava.util.concurrent.*;
publicclassCallableAndFuture
{
//创建线程类
publicstaticclassCallableTestimplementsCallable<String>
{
publicStringcall()throwsException
{
return"HelloWorld!";
}
}
publicstaticvoidmain(String[]args)
{
ExecutorServicethreadPool=Executors.newSingleThreadExecutor();
//启动线程
Future<String>future=threadPool.submit(newCallableTest());
try
{
System.out.println("waitingthreadtofinish");
System.out.println(future.get());//等待线程结束,并获取返回结果
}
catch(Exceptione)
{
e.printStackTrace();
}
}
}
上述程序的运行结果为:
waitingthreadtofinish
HelloWorid!
在以上三种方式中,前两种方式线程执行完后都没有返回值,只有最后一种是带返回值的。当需要实现多线程时,一般推荐实现Runnable接口的方式,原因如下:首先,Thread类定义了多种方法可以被派生类使用或重写,但是只有run方法是必须被重写的,在run方法中实现这个线程的主要功能。这当然是实现Runnable接口所需的同样的方法。而且,很多Java开发人员认为,一个类仅在它们需要被加强或修改时才会被继承。因此,如果没有必要重写Thread类中的其他方法,那么通过继承Thread的实现方式与实现Runnable接口的效果相同,在这种情况下最好通过实现Runnable接口的方式来创建线程。)解析:2.
多线程同步有几种实现方法?
(分数:2.00)__________________________________________________________________________________________
正确答案:(当使用多线程访问同一个资源时,非常容易出现线程安全的问题(例如,当多个线程同时对一个数据进行修改时,会导致某些线程对数据的修改丢失)。因此,需要采用同步机制来解决这种问题。Java主要提供了三种实现同步机制的方法:
(1)synchronized关键字
在Java语言中,每个对象都有一个对象锁与之相关联,该锁表明对象在任何时候只允许被一个线程所拥有,当一个线程调用对象的一段synchronized代码时,首先需要获取这个锁,然后去执行相应的代码,执行结束后,释放锁。
synchronized关键字主要有两种用法(synchronized方法和synchronized块),此外该关键字还可以作用于静态方法、类或某个实例,但这都对程序的效率有很大的影响。
1)synchronized方法。在方法的声明前加入synchronized关键字。例如:
publicsynchronizedvoidmutiThreadAccess();
只要把多个线程访问的资源的操作放到mutiThreadAccess方法中,就能够保证这个方法在同一时刻只能被一个线程访问,从而保证了多线程访问的安全性。然而,当一个方法的方法体规模非常大时,把该方法声明为synchronized会大大影响程序的执行效率。为了提高程序的执行效率,Java语言提供了synchronized块。
2)synchronized块。可以把任意的代码段声明为synchronized,也可以指定上锁的对象,有非常高的灵活性。用法如下:
synchronized(syncObject){
//访问syncObject的代码
}
(2)wait与notify
当使用synchronized来修饰某个共享资源的时候,如果线程A1在执行synchronized代码,另外一个线程A2也要同时执行同一对象的同一synchronized代码时,线程A2将要等到线程A1执行完成后,才能继续执行。在这种情况下,可以使用wait方法和notify方法。
在synchronized代码被执行期间,线程可以调用对象的wait方法,释放对象锁,进入等待状态,并且可以调用notify方法或notifyAll方法通知正在等待的其他线程,notify方法仅唤醒一个线程(等待队列中的第一个线程),并允许它去获得锁,而notifyAll方法唤醒所有等待这个对象的线程,并允许它们去获得锁(并不是让所有唤醒线程都获取到锁,而是让它们去竞争)。
(3)Lock
JDK5新增加了Lock接口以及它的一个实现类ReentrantLock(重入锁),Lock也可以用来实现多线程的同步,具体而言,它提供了如下的一些方法来实现多线程的同步:
1)lock()。以阻塞的方式来获取锁,也就是说,如果获取到了锁,则立即返回,如果其他线程持有锁,当前线程等待,直到获取锁后返回。
2)tryLock()。以非阻塞的方式获取锁。只是尝试性地去获取一下锁,如果获取到锁,则立即返回true,否则,立即返回false。
3)tryLock(longtimeout,TimeUnitunit)。如果获取了锁,立即返回true,否则,会等待参数给定的时间单元,在等待的过程中,如果获取了锁,就返回true,如果等待超时,则返回false。
4)lockInterruptibly()。如果获取了锁,则立即返回,如果没有获取锁,则当前线程处于休眠状态,直到获得锁,或者当前线程被其他线程中断(会收到InterruptedException异常)。它与lock()方法最大的区别在于:如果lock()方法获取不到锁,则会一直处于阻塞状态,且会忽略interrupt()方法。如下例所示:
importjava.util.concurrent.locks.Lock;
importjava.util.concurrent.locks.ReentrantLock;
publicclassTest{
publicstaticvoidmain(String[]args)throwsInterruptedException{
finalLocklock=newReentrantLock();
lock.lock();
Threadt1=newThread(newRunnable(){
publicvoidrun(){
try{
lock.lockInterruptibly();
//lock.lock();编译器报错
}catch(InterruptedExceptione){
System.out.println("interrupted.");
}
}
});
t1.start();
errupt();
Thread.sleep(1);
}
}
程序的运行结果为:
interrupted.
如果把lock.lockInterruptibly()替换为lock.lock(),编译器将会提示lock.lock()catch代码块无效,因为lock.lock()不会抛出异常,由此可见,lock()方法会忽略interrupt()引发的异常。)解析:3.
在多线程编程的时候有哪些注意事项?
(分数:2.00)__________________________________________________________________________________________
正确答案:(多线程编程是一项非常重要的技能。如何能避免死锁,如何提高多线程并发情况下的性能是非常重要的,下面列出一些在多线程编程情况下的指导原则:
1)如果能用volatile代替synchronized,尽可能用volatile。因为被synchronized修饰的方法或代码块在同一时间只允许一个线程访问,而volatile却没有这个限制,因此使用synchronized会降低并发量。由于volatile无法保证原子操作,因此在多线程的情况下,只有对变量的操作为原子操作的情况下才可以使用volatile。
2)尽可能减少synchronized块内的代码,只把临界区的代码放到synchronized块中,尽量避免用synchronized来修饰整个方法。
3)尽可能给每个线程都定义一个线程的名字,不要使用匿名线程,这样有利于调试。
4)尽可能用concurrent容器(ConcurrentHashMap)来代替synchronized容器(Hashtable)。因为synchronized容器使用synchronized关键字通过对整个容器加锁来实现多线程安全,性能比较低。而ConcurrentHashMap采用了更加细粒度的锁,因此可以支持更高的并发量。
5)使用线程池来控制多线程的执行。)解析:4.
一个文件中有10000个数,用Java语言实现一个多线程程序,将这10000个数输出到5个不同文件中(不要求输出到每个文件中的数量相同)。要求启动10个线程,两两一组,分为5组。每组两个线程分别将文件中的奇数和偶数输出到该组对应的一个文件中,需要偶数线程每打印10个偶数以后,就将奇数线程打印10个奇数,如此交替进行。同时需要记录输出进度,每完成1000个数就在控制台中打印当前完成数量,并在所有线程结束后,在控制台输出“Done”。
(分数:2.00)__________________________________________________________________________________________
正确答案:(本题考查的是对多线程编程的理解。为了便于理解,首先用随机函数随机生成10000个数放到文件中,以供测试使用。一次把这10000条记录读到内存中,平均分配给5组线程并行处理,因此,本题的难点是如何控制打印偶数的线程和打印奇数的线程轮流运行。
本题通过Java提供的Condition来实现线程的同步。Condition是在Java1.5中才出现的,它用来替代传统的Object类的wait()、notify()方法,以实现线程间的协作,相比使用Object类的wait()、nofify()方法,使用Condition的await()、signal()这种方式实现线程间协作,更加安全和高效。它主要有如下特点:
1)Condition最常用的方法为await()和signal(),其中,await()对应Object类的wait()方法,signal()对应Object类的notify()方法。
2)Condition依赖于Lock接口,生成一个Condition的代码为lock.newCondition()。
3)调用Condition的await()和signal()方法必须在lock保护之内。
对于本题而言,定义两个Condition(oddLock和evenLock),首先打印奇数的线程开始运行,通过调用evenLock.await()来等待打印偶数的线程执行。接着打印偶数的线程开始运行,当输出10个偶数或者没有偶数输出后,调用evenLock.signal()来通知打印奇数的线程开始运行,然后调用oddLock.wait方法来等待打印奇数的线程运行完成。通过这种方法来控制奇数线程与偶数线程的运行顺序,实现代码如下:
importjava.io.*;
importjava.util.Random;
importjava.util.concurrent.locks.Condition;
importjava.util.concurrent.locks.Lock;
importjava.util.concurrent.locks.ReentrantLock;
publicclassTest
{
privatestaticfinalintcount=10000;
privatestaticfinalintthreadGruopCount=5;
privatestaticfinalStringinputFile="testInput.txt";
publicstaticvoidgenerateTestFile()throwsIOException
{
//用随机数生成10000个测试数据放到文件中
PrintWriterpw=newPrintWriter(newFileWriter(newFile(inputFile)),true);
Randomrandom=newRandom();
for(inti=0;i<count;i++)
{
pw.write(Math.abs(random.nextInt())%count+",");
}
pw.flush();
pw.close();
}
publicstaticvoidmain(String[]args)
{
try
{
generateTestFile();
BufferedReaderreader=newBufferedReader(newFileReader(inputFile));
Stringstr=reader.readLine();
reader.close();
String[]strs=str.split(",");
intindex=0;
//为了简单,每个文件输出数字的个数相同
int
到此这篇java面试题大全及答案下载网盘(java面试题大全带答案pdf版)的文章就介绍到这了,更多相关内容请继续浏览下面的相关推荐文章,希望大家都能在编程的领域有一番成就!版权声明:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若内容造成侵权、违法违规、事实不符,请将相关资料发送至xkadmin@xkablog.com进行投诉反馈,一经查实,立即处理!
转载请注明出处,原文链接:https://www.xkablog.com/javal-zj/18321.html