牛客网答题知识记录

1.servlet在多线程下其本身并不是线程安全的,如果在类中定义成员变量,在service中根据不同的线程对该成员变量进行更改,那么在并发的时候就会引起错误。最好是在方法中定义局部变量,而不是类变量或者对象的成员变量,由于方法中的局部变量是在栈中,彼此各自都拥有各自的运行空间而不会相互干扰,因此才做到线程安全。

servlet是线程不安全的,在servlet中可能会定义共享的类变量,这样在并发的多线程访问的情况下,不同的线程对成员变量的修改会引发错误。

init()是在servlet实例创建的时候调用的方法,用于创建或打开任意与servlet相关的资源和初始化servlet的状态,servlet规范保证调用init方法前不会处理任何请求。

service()方法是servlet真正处理客户端传过来的请求的方法,由web容器调用,根据http请求方法(get、post等),将请求分发到doget、dopost方法。

destory()是在servlet实例被销毁时由web容器调用,servlet规范确保在destory方法调用之前所有请求的处理均完成,需要覆盖destroy方法的情况,释放任何在init()方法中打开的与servlet相关的资源存储servlet状态。

2.动态与静态include

动态include用jsp:include动作实现,<jsp:include page = "include.jsp" flush="true"/>,它总是会检查所有文件中的变化,适合用于包含动态页面,并且可以带参数,各个文件可以分别先编译,然后组合成一个文件。

静态include用include伪码来实现,不会检查所含文件的变化,适用于包含静态页面<%@include file="included.htm"%>,先将文件代码原封不动的加入主页面从而合成一个文件,然后再进行翻译,此时不允许有相同的变量。

以下是对include两种用法的区别,主要有两个方面的不同:

<1>执行时间上

<%@include file="relativeURI"%>是在翻译阶段执行

<jsp:include page = "relativeURI" flush="true"/>在请求处理阶段执行

<2>引入内容的不同

<%@ include file="relativeURI"%>

引入静态文本(html,jsp),在jsp页面被转化成servlet之前和它融合到一起。

 <jsp:include page="relativeURI" flush="true" />引入执行页面或servlet所生成的应答文本

3.NullPointerException 不属于运行时异常

4.jvm中一个字节以下的整整型数据会在jvm启动的时候加载进内存,除非用new integer()显式的创建对象,否则都是同一个对象

5.方法区在jvm是个重要的区域,与堆一样是被线程共享的区域。在方法区中,存储了每个类的信息(包括类的名称,方法信息,字段信息)、静态变量、常量以及编译器编译后的代码。

6.forward,服务器获取跳转页面内容传给用户,用户地址栏不变,redirect,是服务器向用户发送转向的地址,redirect后地址栏变成新的地址

7.创建Servlet的实例是由Servlet容器来完成的,且创建Servlet实例是在初始化方法init()之前

8.

Java中的多线程是一种抢占式的机制,而不是分时机制。抢占式的机制是有多个线程处于可运行状态,但是只有一个线程在运行。 
共同点 : 
1>. 他们都是在多线程的环境下,都可以在程序的调用处阻塞指定的毫秒数,并返回。 
2>. wait()和sleep()都可以通过interrupt()方法 打断线程的暂停状态 ,从而使线程立刻抛出InterruptedException。 
如果线程A希望立即结束线程B,则可以对线程B对应的Thread实例调用interrupt方法。如果此刻线程B正在wait/sleep/join,则线程B会立刻抛出InterruptedException,在catch() {} 中直接return即可安全地结束线程。 
需要注意的是,InterruptedException是线程自己从内部抛出的,并不是interrupt()方法抛出的。对某一线程调用 interrupt()时,如果该线程正在执行普通的代码,那么该线程根本就不会抛出InterruptedException。但是,一旦该线程进入到 wait()/sleep()/join()后,就会立刻抛出InterruptedException 。 
不同点 :  
1>.每个对象都有一个锁来控制同步访问。Synchronized关键字可以和对象的锁交互,来实现线程的同步。 
sleep方法没有释放锁,而wait方法释放了锁,使得其他线程可以使用同步控制块或者方法。 
2>.wait,notify和notifyAll只能在同步控制方法或者同步控制块里面使用,而sleep可以在任何地方使用 
3>.sleep必须捕获异常,而wait,notify和notifyAll不需要捕获异常 
4>.sleep是线程类(Thread)的方法,导致此线程暂停执行指定时间,给执行机会给其他线程,但是监控状态依然保持,到时后会自动恢复。调用sleep不会释放对象锁。
5>.wait是Object类的方法,对此对象调用wait方法导致本线程放弃对象锁,进入等待此对象的等待锁定池,只有针对此对象发出notify方法(或notifyAll)后本线程才进入对象锁定池准备获得对象锁进入运行状态。
原文地址:https://www.cnblogs.com/zaks/p/11239857.html