java初入多线程15

作者: 胖琪的升级之路 | 来源:发表于2017-10-24 11:25 被阅读16次

无锁的线程安全整数: AtomicInteger

  • 方法介绍
    1. public final int get(); 取得当前值
    2. public final void set(int newValue); 设置当前值
    3. public final int getAndSet(int newValue); 设置新值 返回旧值
    4. public final boolean compareAndSet(int expect, int update); 如果当前值为expect ,则设置为update.
    5. public final int getAndIncrement(); 当前值加1 ,返回旧值
    6. public final int getAndDecrement(); 当前值减1 ,返回旧值
    7. public final int getAndAdd(int delta);当前值增加data, 返回旧值
    8. public final int incrementAndGet(); 当前值增加1 ,返回新值。
  • 代码实现
public class AtomicIntegerDemo {

    static AtomicInteger i = new  AtomicInteger() ;
    
    public static class AddThread implements Runnable{
        @Override
        public void run() {
            for(int k = 0 ; k < 10000 ; k++){
                i.incrementAndGet() ;
            }
        }
    }
    public static void main(String[] args) throws Exception {
        Thread[] ts = new  Thread[10];
        for(int k = 0 ; k < 10 ; k++){
            ts[k] = new  Thread(new AddThread()); 
        }
        for(int k = 0 ; k<10 ;k++){
            ts[k].start();
        }
        for(int k = 0; k < 10 ; k++ ){
            ts[k].join();
        }
        System.out.println(i);
    }
}

Unsafe 类:

  • 是封装了一些类似指针的操作。还有内部实现也是依赖CAS原子指令来完成的。这是一个不让我们在应用上使用的类型

无锁对象的引用:AtomicReference

  • 与AtomicInteger 类似 不过 该类是用来操作对象的。 根据当前值和期望值进行比较设置。当一致时候采用该类中的方法进行修改。
  • 但是有种情况就是 如果我们修改 值得时候出现 1-2-1 这个中情况我们最后拿到的还是1 那么程序还是认为是1 。只认为结果是正确的再进行修改。改为2 或者其他的。代码实现如下:
public class AtomicStampedDemo {
    static AtomicReference< Integer> money = new AtomicReference<>() ;
    
     
    public static void main(String[] args) {
        money.set(20);
        for(int i = 0 ;i < 3 ; i++ ){
            new Thread(){

                public void run() {
                    while(true){
                        Integer m = money.get(); 
                        if (m < 20 ){
                            if(money.compareAndSet(m, m+20 )){
                                System.out.println("余额小于20元,充值成功,余额: " + money.get() 
                                + " 元 ");
                            }
                        }else{
                            System.out.println("余额大于20元, 无须充值 ");
                            break;
                        }
                    }
                };
            }.start();
        }
        //用户消费线程, 模拟消费行为
        new Thread(){
            public void run() {
                for( int i = 0 ; i < 100 ; i++){
                     while(true){
                         Integer m =money.get() ;
                         if(m > 10){
                             System.out.println("大于10元");
                         if(money.compareAndSet(m, m-10)){
                                 System.out.println("成功消费10元,余额:" + money.get());
                                 break ;
                             };
                         }else{
                             System.out.println("没有足够的金额");
                             break ;
                         }
                     }
                     try {
                        Thread.sleep(100);
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }           };
        }.start();   
    }   
}

AtomicStampedReference

  • 为了防止出现多次提交的问题。比如我们限制 一个用户只参与一次活动,那上面的实现方式就不行了,因为只记录了每次的金额,没有显示出来变动的情况。我们改用AtomicStampedReference来实现 ,该类在原先的基础上增加了时间戳的记录 ,记录了过程的实现。
public class AtomicStampedReferenceDemo {

     static AtomicStampedReference<Integer> money = new AtomicStampedReference<Integer>(19, 0);  
     public static void main(String[] args) {
        for(int i = 0 ;i < 3 ; i++ ){
            final int timestamp = money.getStamp() ;
            new Thread(){
                public void run() {
                    while(true){
                        Integer m = money.getReference() ; 
                        if (m < 20 ){
                            if(money.compareAndSet(m, m+20,timestamp, timestamp + 1 )){
                                System.out.println("余额小于20元,充值成功,余额: " + money.getReference() 
                                + " 元 ");
                            }
                        }else{
                            System.out.println("余额大于20元, 无须充值 ");
                            break;
                        }
                    }
                };
            }.start();
        }
        
        //用户消费线程, 模拟消费行为
        new Thread(){
            public void run() {
                for( int i = 0 ; i < 100 ; i++){
                     while(true){
                         int timestamp = money.getStamp() ;
                         Integer m =money.getReference() ;
                         if(m > 10){
                             System.out.println("大于10元");
                             if(money.compareAndSet(m, m-10, timestamp, timestamp+1)){
                                 System.out.println("成功消费10元,余额:" + money.getReference());
                                 break ;
                             };
                         }else{
                             System.out.println("没有足够的金额");
                             break ;
                         }       
                     }
                     try {
                        Thread.sleep(100);
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            };
        }.start();       
    }
}

数组也能无锁: AtomicIntegerArray

  • 使用方式跟上面的AtomicInteger 类似 ,都采用的CAS 操作。

相关文章

  • java初入多线程15

    无锁的线程安全整数: AtomicInteger 方法介绍public final int get(); 取得当前...

  • java初入多线程4

    线程中断 概念 :让目标线程停止执行,但是是高知目标线程希望线退出,具体退出由目标线程自己决定。 相关的方法,暂时...

  • java初入多线程17

    使用Disruptor 实现消费者和生产者 我们在主方法操作中将缓冲区设置成1024 , 在这里有四个消费者, 有...

  • java初入多线程11

    核心线程池的内部实现机制。 阿里巴巴 code检验推荐自己实现线程池的创建。不是使用Executors的创建方法。...

  • java初入多线程12

    自定义线程创建:ThreadFactory 我们原先用的线程池ThreadPoolExecutor 里面的线程都...

  • java初入多线程10

    线程阻塞工具类 :LockSupport LockSupport 是一个非常实用的线程阻塞工具, 可以在线程内任意...

  • java初入多线程7

    同步控制 synchronized 扩展:重入锁 重入锁来代替synchronized,在Jdk1.6以后 syn...

  • java初入多线程14

    接下来几章说的是锁的优化和注意事项问题。 减小锁持有的时间 对于在方法执行的过程中有的步骤不需要进行同步,那么就在...

  • java初入多线程13

    并发集合简介 ConcurrentHashMap : 线程安全的HashMap; CopyOnWriteArray...

  • java初入多线程16

    并行模式与算法 单例模式 : 保证在系统中只生产一个实例。下面是几种单例模式。 这个容易出现的问题就是单例什么时候...

网友评论

    本文标题:java初入多线程15

    本文链接:https://www.haomeiwen.com/subject/nhhtpxtx.html