Java线程:线程的同步-同步方法

Java线程:线程的同步-同步方法
 
线程的同步是保证多线程安全访问竞争资源的一种手段。
线程的同步是Java多线程编程的难点,每每开发者搞不清楚什么是竞争资源、何时须要考虑同步,怎么同步等等问题,固然,这些问题没有很明确的答案,但有些原则问题须要考虑,是否有竞争资源被同时改动的问题?
 
在本文以前,请参阅《 Java线程:线程的同步与锁》,本文是在此基础上所写的。
 
对于同步,在具体的Java代码中须要完成一下两个操做:
把竞争访问的资源标识为private;
同步哪些修改变量的代码,使用synchronized关键字同步方法或代码。
固然这不是惟一控制并发安全的途径。
 
synchronized关键字使用说明
synchronized只能标记非抽象的方法,不能标识成员变量。
 
为了演示同步方法的使用,构建了一个信用卡帐户,起初信用额为100w,而后模拟透支、存款等多个操做。显然银行帐户User对象是个竞争资源,而多个并发操做的是帐户方法oper(int x),固然应该在此方法上加上同步,并将帐户的余额设为私有变量,禁止直接访问。
 
 
/** 
* Java线程:线程的同步 
* 
* @author leizhimin 2009-11-4 11:23:32 
*/ 
public class Test { 
        public static void main(String[] args) { 
                User u = new User("张三", 100); 
                MyThread t1 = new MyThread("线程A", u, 20); 
                MyThread t2 = new MyThread("线程B", u, -60); 
                MyThread t3 = new MyThread("线程C", u, -80); 
                MyThread t4 = new MyThread("线程D", u, -30); 
                MyThread t5 = new MyThread("线程E", u, 32); 
                MyThread t6 = new MyThread("线程F", u, 21); 

                t1.start(); 
                t2.start(); 
                t3.start(); 
                t4.start(); 
                t5.start(); 
                t6.start(); 
        } 
} 

class MyThread extends Thread { 
        private User u; 
        private int y = 0; 

        MyThread(String name, User u, int y) { 
                super(name); 
                this.u = u; 
                this.y = y; 
        } 

        public void run() { 
                u.oper(y); 
        } 
} 

class User { 
        private String code; 
        private int cash; 

        User(String code, int cash) { 
                this.code = code; 
                this.cash = cash; 
        } 

        public String getCode() { 
                return code; 
        } 

        public void setCode(String code) { 
                this.code = code; 
        } 

        /** 
         * 业务方法 
         * @param x 添加x万元 
         */ 
        public synchronized void oper(int x) { 
                try { 
                        Thread.sleep(10L); 
                        this.cash += x; 
                        System.out.println(Thread.currentThread().getName() + "运行结束,增长“" + x + "”,当前用户帐户余额为:" + cash); 
                        Thread.sleep(10L); 
                } catch (InterruptedException e) { 
                        e.printStackTrace(); 
                } 
        } 

        @Override 
        public String toString() { 
                return "User{" + 
                                "code='" + code + '\'' + 
                                ", cash=" + cash + 
                                '}'; 
        } 
}

 

 
输出结果:
线程A运行结束,增长“20”,当前用户帐户余额为:120 
线程F运行结束,增长“21”,当前用户帐户余额为:141 
线程E运行结束,增长“32”,当前用户帐户余额为:173 
线程C运行结束,增长“-80”,当前用户帐户余额为:93 
线程B运行结束,增长“-60”,当前用户帐户余额为:33 
线程D运行结束,增长“-30”,当前用户帐户余额为:3 

Process finished with exit code 0
 
 
反面教材,不一样步的状况,也就是去掉oper(int x)方法的synchronized修饰符,而后运行程序,结果以下:
线程A运行结束,增长“20”,当前用户帐户余额为:61 
线程D运行结束,增长“-30”,当前用户帐户余额为:63 
线程B运行结束,增长“-60”,当前用户帐户余额为:3 
线程F运行结束,增长“21”,当前用户帐户余额为:61 
线程E运行结束,增长“32”,当前用户帐户余额为:93 
线程C运行结束,增长“-80”,当前用户帐户余额为:61 

Process finished with exit code 0
 
很显然,上面的结果是错误的,致使错误的缘由是多个线程并发访问了竞争资源u,并对u的属性作了改动。
 
可见同步的重要性。
 
 
注意:
经过前文可知,线程退出同步方法时将释放掉方法所属对象的锁,但还应该注意的是,同步方法中还可使用特定的方法对线程进行调度。这些方法来自于java.lang.Object类。
 
void notify()    
                    唤醒在此对象监视器上等待的单个线程。    
void notifyAll()    
                    唤醒在此对象监视器上等待的全部线程。    
void wait()    
                    致使当前的线程等待,直到其余线程调用此对象的 notify() 方法或 notifyAll() 方法。    
void wait( long timeout)    
                    致使当前的线程等待,直到其余线程调用此对象的 notify() 方法或 notifyAll() 方法,或者超过指定的时间量。    
void wait( long timeout,  int nanos)    
                    致使当前的线程等待,直到其余线程调用此对象的 notify() 方法或 notifyAll() 方法,或者其余某个线程中断当前线程,或者已超过某个实际时间量。
 
结合以上方法,处理多线程同步与互斥问题很是重要,著名的生产者-消费者例子就是一个经典的例子,任何语言多线程必学的例子。

本文出自 “熔 岩” 博客,请务必保留此出处http://lavasoft.blog.51cto.com/62575/221914java

相关文章
相关标签/搜索