Android开发中synchronized的实现原理

synchronized的三种使用方式

**1.修饰实例方法,**作用于当前实例加锁,进入同步代码前要获得当前实例的锁。

没有问题的写法:

public class AccountingSync implements Runnable{
   
    //共享资源(临界资源)
    static int i=0;

    /**
     * synchronized 修饰实例方法
     */
    public synchronized void increase(){
   
        i++;
    }
    @Override
    public void run() {
   
        for(int j=0;j<1000000;j++){
   
            increase();
        }
    }
    public static void main(String[] args) throws InterruptedException {
   
        AccountingSync instance=new AccountingSync();
        Thread t1=new Thread(instance);
        Thread t2=new Thread(instance);
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(i);
    }
    /**
     * 输出结果:
     * 2000000
     */
}

因为这段代码中只输入了一个AccountingSync实例。

下面是有问题的写法:

public class AccountingSyncBad implements Runnable{
   
    static int i=0;
    public synchronized void increase(){
   
        i++;
    }
    @Override
    public void run() {
   
        for(int j=0;j<1000000;j++){
   
            increase();
        }
    }
    public static void main(String[] args) throws InterruptedException {
   
        //new新实例
        Thread t1=new Thread(new AccountingSyncBad());
        //new新实例
        Thread t2=new Thread(new AccountingSyncBad());
        t1.start();
        t2.start();
        //join含义:当前线程A等待thread线程终止之后才能从thread.join()返回
        t1.join();
        t2.join();
        System.out.println(i);
    }
}

在上述代码中出现了一个严重的错误,虽然我们使用了sychronized修饰了increase方法,但new了两个不同的实例对象,这就意味着会出现两个不同的实例对象锁,这样的话ti和t2都会进入各自的对象锁,因此线程安全就得不到保证。想要解决这个问题,就需要将sychronized作用于静态的increase方法。

2.修饰静态方法,作用于当前类对象加锁,进入同步代码前要获得当前类对象的锁。

当sychronized使用静态方法时,对象锁就会是当前类的class对象锁。因为静态成员不属于任何一个实例对象,所以通过class对象锁可以控制静态 成员的并发操作。

如果一个线程A调用一个实例对象的非static synchronized方法,而线程B需要调用这个实例对象所属类的静态 synchronized方法,是允许的,不会发生互斥现象,因为访问静态 synchronized 方法占用的锁是当前类的class对象,而访问非静态 synchronized 方法占用的锁是当前实例对象锁,看如下代码:

public class AccountingSyncClass implements Runnable{
   
    static int i=0;

    /**
     * 作用于静态方法,锁是当前class对象,也就是
     * AccountingSyncClass类对应的class对象
     */
    public static synchronized void increase(){
   
        i++;
    }

    /**
     * 非静态,访问时锁不一样不会发生互斥
     */
    public synchronized void increase4Obj(){
   
        i++;
    }

    @Override
    public void run() {
   
        for(int j=0;j<1000000;j++){
   
            increase();
        }
    }
    public static void main(String[] args) throws InterruptedException {
   
        //new新实例
        Thread t1=new Thread(new AccountingSyncClass());
        //new心事了
        Thread t2=new Thread(new AccountingSyncClass());
        //启动线程
        t1.start();t2.start();

        t1.join();t2.join();
        System.out.println(i
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值