多线程常见手撕问题

Posted oahaijgnahz

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了多线程常见手撕问题相关的知识,希望对你有一定的参考价值。

1. 多线程按序循环打印ABC

package com.demo.test;

/**
 * 基于一个锁和一个状态变量实现连续打印ABCABC...
 * @author lixiaoxi
 *
 */
public class StateLockPrinter {
    //状态变量
    private volatile int state=0;
    
    // 打印线程
    private class Printer implements Runnable {
        //打印次数
        private static final int PRINT_COUNT=10;
        //打印锁
        private final Object printLock;
        //打印标志位 和state变量相关
        private final int printFlag;
        //后继线程的线程的打印标志位,state变量相关
        private final int nextPrintFlag;
        //该线程的打印字符
        private final char printChar;
        
        public Printer(Object printLock, int printFlag,int nextPrintFlag, char printChar) {
            super();
            this.printLock = printLock;
            this.printFlag=printFlag;
            this.nextPrintFlag=nextPrintFlag;
            this.printChar = printChar;
        }

        @Override
        public void run() {
            //获取打印锁 进入临界区
            synchronized (printLock) {
                //连续打印PRINT_COUNT次
                for(int i=0;i<PRINT_COUNT;i++){
                    //循环检验标志位 每次都阻塞然后等待唤醒
                    while (state!=printFlag) {
                        try {
                            printLock.wait();
                        } catch (InterruptedException e) {
                            return;
                        }
                    }
                    //打印字符
                    System.out.print(printChar);
                    //设置状态变量为下一个线程的标志位
                    state=nextPrintFlag;
                    //注意要notifyall,不然会死锁,因为notify只通知一个,
                    //但是同时等待的是两个,如果唤醒的不是正确那个就会没人唤醒,死锁了
                    printLock.notifyAll();
                }
            }
        }
        
    }

    public void test() throws InterruptedException{
        //锁
        Object lock=new Object();
        //打印A的线程
        Thread threadA=new Thread(new Printer(lock, 0,1, 'A'));
        //打印B的线程
        Thread threadB=new Thread(new Printer(lock, 1,2, 'B'));
        //打印C的线程
        Thread threadC=new Thread(new Printer(lock, 2,0, 'C'));
        //一次启动A B C线程
        threadA.start();
        Thread.sleep(1000);
        threadB.start();
        Thread.sleep(1000);
        threadC.start();
    }
    
    public static void main(String[] args) throws InterruptedException {
        
        StateLockPrinter print = new StateLockPrinter();
        print.test();
    }
}

2. 双线程交替打印奇偶数

public class EvenAndOdd {

    static int count = 0;
    static final int MAX = 100;
    static String lockObj = "lock";

    class Print implements Runnable {

        @Override
        public void run() {
            while (count <= MAX) {
                synchronized (lockObj) {

                    System.out.println(Thread.currentThread().getName()+":"+count++);

                    lockObj.notifyAll();

                    try {
                        if(count<=100){
                            lockObj.wait();
                        }
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }


    public static void main(String[] args) throws InterruptedException {
        EvenAndOdd evenAndOdd = new EvenAndOdd();
        Thread even = new Thread(evenAndOdd.new Print(),"偶数");
        Thread odd = new Thread(evenAndOdd.new Print(),"奇数");

        even.start();
        Thread.sleep(10);
        odd.start();
    }
}

以上是关于多线程常见手撕问题的主要内容,如果未能解决你的问题,请参考以下文章

去年去阿里面试,被问到java 多线程,我是这样手撕面试官的

手把手写C++服务器(37):手撕代码——高并发多线程技术基石之异步connect万字长文

Redis是多线程还是单线程?看完“手撕”面试官

手撕三种分布式锁

手撕几种常见单例模式

手撕几种常见单例模式