成员变量:作为类的成员而存在,直接存在于类中。所有类的成员变量可以通过this来引用。
局部变量:作为方法或语句块的成员而存在,存在于方法的参数列表和方法定义中。
1.成员变量可以被 public,protect,private,static等修饰符修饰,而局部变量不能被控制修饰符及 static修饰;两者都可以定义成final型。
2.成员变量存储在堆,局部变量存储在栈。局 部变量的作用域仅限于定义它的方法,在该方法的外部无法访问它。成员变量的作用域在整个类内部都是可见的,所有成员方法都可以使用它。如果访问权限允许,还可以在类的外部使用成员变量。
3.局部变量的生存周期与方法的执行期相同。 当方法执行到定义局部变量的语句时,局部变量被创建;执行到它所在的作用域的最后一条语句时,局部变量被销毁。类的成员变量,如果是实例成员变量,它和对象的生存期相同。而静态成员变量的生存期是整个程序运行期。
4.成员变量有默认值,基本类型的默认值为 0,复合类型的默认值为null。(被final修饰且没有static的必须显式赋值),局部变量不会自动赋值,所以局 部变量在定义后先要赋初值,然后才能使用。
5.局部变量可以和成员变量 同名,且在使用时,局部变量具有更高的优先级。
如果一个变量是成员变量,那么多个线程对同一个对象的成员变量进行操作时,它们对该成员变量是彼此影响的,也就是说一个线程对成员变量的改变会影响到另一个线程。
如果一个变量是局部变量,那么每个线程都会有一个该局部变量的拷贝(即便是同一个对象中的方法的局部变量,也会对每一个线程有一个拷贝),一个线程对该局部变量的改变不会影响到其他线程。
成员变量多线程访问安全
ThreadLocal是JDK引入的一种机制,它用于解决线程间共享变量,使用ThreadLocal声明的变量,即使在线程中属于全局变量,针对每个线程来讲,这个变量也是独立的。 它通过为每个线程提供一个独立的变量副本解决了变量并发访问的冲突问题。在很多情况下,ThreadLocal比直接使用synchronized同步机制解决线程安全问题更简单,更方便,且结果程序拥有更高的并发性。
关于成员变量与局部变量:如果一个变量是成员变量,那么多个线程对同一个对象的成员变量进行操作时,他们对该成员变量是彼此影响的(也就是说一个线程对成
员变量的改变会影响到另一个线程)
实例:
Servlet容器为了响应多个用户同时访问一个Servlet的HTTP请求,通常会为每个请求分配一个工作线程,这些工作线程并发执行同一个Servlet的service()方法。此时可能发生多线程对同一数据进行访问的情况。
- 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设置为局部变量。
- 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同步机制对多线程同步
- 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()中的同步代码块。只有当这个工作线程退出同步代码块时,其他工作线程才允许执行同步代码块。这使得任意时刻不会有两个线程同时操纵同一个实例变量,从而避免并发问题。