java 线程---成员变量与局部变量
2016-02-04 17:05
381 查看
关于成员变量与局部变量:
如果一个变量是成员变量,那么多个线程对同一个对象的成员变量进行操作时,他们对该成员变量是彼此影响的(也就是说一个线程对成员变量的改变会影响到另一个线程) 。
如果一个变量是局部变量,那么每个线程都会有一个该局部变量的拷贝,一个线程对该局部变量的改变不会影响到其他的线程。
实例:
Servlet容器为了响应多个用户同时访问一个Servlet的HTTP请求,通常会为每个请求分配一个工作线程,这些工作线程并发执行同一个Servlet(即Servlet是单实例的)的service()方法。此时可能发生多线程对同一数据进行访问的情况。
Java代码
public class Hello extends HttpServlet{
private String name;
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,java.io.IOException{
response.setContentType("text/html;charset=UTF-8");
name = (String)request.getParameter("name"); //接收参数
PrintWriter out = response.getWriter();
out.println("<html><head><title>test</title></head><body>");
out.println("你好"+name);
out.println("</body></html>");
out.close();
}
}
如果多线程并发访问,会访问同一个实例变量,则会共用name,而出现用户得到数据不一致的现象。
解决办法:
1.将name设置为局部变量。
Java代码
public class Hello extends HttpServlet{
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,java.io.IOException{
response.setContentType("text/html;charset=UTF-8");
String name = (String)request.getParameter("name"); //接收参数
...
}
}
每当一个线程执行doPost()时,在线程的堆栈中就会创建name这个局部变量,当线程执行完该方法,局部变量就结束生命周期。如果多个线程同时执行该方法,那么每个线程都拥有自己的局部变量。
2.使用Java同步机制对多线程同步
Java代码
public class Hello extends HttpServlet{
private String name;
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,java.io.IOException{
response.setContentType("text/html;charset=UTF-8");
synchronized(this){
name = (String)request.getParameter("name"); //接收参数
...
}
...
}
}
这样可确保在任意一时刻,只允许有一个工作线程执行doPost()中的同步代码块。只有当这个工作线程退出同步代码块时,其他工作线程才允许执行同步代码块。这使得任意时刻不会有两个线程同时操纵同一个实例变量,从而避免并发问题。
3.实现被废弃的SingleThreadModel接口
public class Hello extends HttpServlet implements
SingleThreadModel{
...
}
如果实现此接口,则Servlet容器实现可以采用以下两种方式之一来运行Servlet:
1)在任意一时刻,只允许有一个工作线程执行Servlet的service()方法。如果有多个用户同时请求访问该Servlet,那么这些客户请求被
放入等待队列,容器会依次响应等待队列中的每个客户请求。这种实现方式实际上禁止了多个客户端对同一个Servlet的并发访问。
2)Servlet容器为每个Servlet创建一个对象池,在这个池中存放了同一个Servlet类的多个实例。如果有多个用户同时请求访问该
Servlet,Servlet容器会为每个请求分配一个工作线程,并且从对象池中取出一个空闲的Servlet实例,把它分配给工作线程。每个工作线程
执行自己的Servlet实例的service()方法。这种实现方式表面上允许客户端对同一个Servlet并发访问,但实际上不同客户端访问的是同一
个Servlet类的不同实例。
如果实例变量需要共享,用以进行计算,这种方法并不能有效的避免并发问题。
如果一个变量是成员变量,那么多个线程对同一个对象的成员变量进行操作时,他们对该成员变量是彼此影响的(也就是说一个线程对成员变量的改变会影响到另一个线程) 。
如果一个变量是局部变量,那么每个线程都会有一个该局部变量的拷贝,一个线程对该局部变量的改变不会影响到其他的线程。
实例:
Servlet容器为了响应多个用户同时访问一个Servlet的HTTP请求,通常会为每个请求分配一个工作线程,这些工作线程并发执行同一个Servlet(即Servlet是单实例的)的service()方法。此时可能发生多线程对同一数据进行访问的情况。
Java代码
public class Hello extends HttpServlet{
private String name;
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,java.io.IOException{
response.setContentType("text/html;charset=UTF-8");
name = (String)request.getParameter("name"); //接收参数
PrintWriter out = response.getWriter();
out.println("<html><head><title>test</title></head><body>");
out.println("你好"+name);
out.println("</body></html>");
out.close();
}
}
如果多线程并发访问,会访问同一个实例变量,则会共用name,而出现用户得到数据不一致的现象。
解决办法:
1.将name设置为局部变量。
Java代码
public class Hello extends HttpServlet{
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,java.io.IOException{
response.setContentType("text/html;charset=UTF-8");
String name = (String)request.getParameter("name"); //接收参数
...
}
}
每当一个线程执行doPost()时,在线程的堆栈中就会创建name这个局部变量,当线程执行完该方法,局部变量就结束生命周期。如果多个线程同时执行该方法,那么每个线程都拥有自己的局部变量。
2.使用Java同步机制对多线程同步
Java代码
public class Hello extends HttpServlet{
private String name;
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,java.io.IOException{
response.setContentType("text/html;charset=UTF-8");
synchronized(this){
name = (String)request.getParameter("name"); //接收参数
...
}
...
}
}
这样可确保在任意一时刻,只允许有一个工作线程执行doPost()中的同步代码块。只有当这个工作线程退出同步代码块时,其他工作线程才允许执行同步代码块。这使得任意时刻不会有两个线程同时操纵同一个实例变量,从而避免并发问题。
3.实现被废弃的SingleThreadModel接口
public class Hello extends HttpServlet implements
SingleThreadModel{
...
}
如果实现此接口,则Servlet容器实现可以采用以下两种方式之一来运行Servlet:
1)在任意一时刻,只允许有一个工作线程执行Servlet的service()方法。如果有多个用户同时请求访问该Servlet,那么这些客户请求被
放入等待队列,容器会依次响应等待队列中的每个客户请求。这种实现方式实际上禁止了多个客户端对同一个Servlet的并发访问。
2)Servlet容器为每个Servlet创建一个对象池,在这个池中存放了同一个Servlet类的多个实例。如果有多个用户同时请求访问该
Servlet,Servlet容器会为每个请求分配一个工作线程,并且从对象池中取出一个空闲的Servlet实例,把它分配给工作线程。每个工作线程
执行自己的Servlet实例的service()方法。这种实现方式表面上允许客户端对同一个Servlet并发访问,但实际上不同客户端访问的是同一
个Servlet类的不同实例。
如果实例变量需要共享,用以进行计算,这种方法并不能有效的避免并发问题。
相关文章推荐
- Groovy在不同JDK版本下的性能差异
- Java中各种(类、方法、属性)访问修饰符与修饰符的说明
- JAVA的字节流和字符流
- Java基础:包装类
- mac下破解MyEclipse 2015
- java中静态属性和和静态方法的继承问题 以及多态的实质
- 执行篇:解析JDK 7的动态类型语言支持
- Java中数据类型转换问题
- 设置Eclipse代码自动提示
- 从头认识Spring-1.15 对SpEl的值的操作(3)-逻辑运算以及条件表达式
- Java设计模式(十七)----责任链模式
- javaBean 相关知识
- eclipse报错
- spring activemq quartz 整合
- Java泛型用法总结
- eclipse安装插件:
- java的static块执行时机
- mybaits错误解决:There is no getter for property named 'id' in class 'java.lang.Long'
- 使用Hadoop分析数据——简单案例java语言编程之MaxTemperature
- Java集合类总结 (三)