Maison >Java >javaDidacticiel >Comment résoudre le problème du multithread dans Java for loop

Comment résoudre le problème du multithread dans Java for loop

WBOY
WBOYavant
2023-04-29 16:55:071861parcourir

    Java utilise le multi-threading pour accélérer l'efficacité des boucles

    Pool de threads avec verrouillage

    Connaissances impliquées : exécuteurs (pool de threads), CountDownLatch (verrouillage)

    Avantages : code concis, facile à lire, performances stables ;

    Inconvénients : Le pool de threads créé par les exécuteurs est public. Si cette méthode multi-thread en boucle est utilisée à plusieurs endroits, les ressources du pool de threads seront volées et la vitesse d'exécution sera également réduite

    import java.util.*;
    import java.util.concurrent.CountDownLatch;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
     
    public class test{
     
        /** 固定的线程池(当前线程池大小为5) */
        private static final ExecutorService executor = Executors.newFixedThreadPool(5);
        
        public static void main(String[] args) throws Exception {
            /**
             *  两个要点:
             *  1.用Executors实现固定大小的线程池,从而达到控制硬件资源消耗的目的。
             *  2.用CountDownLatch(闭锁),来确保循环内的多线程都执行完成后,再执行后续代码
             */
            
            // 初始化数据
            List<Map<String,Object>> list = new ArrayList<>();
            for(int i=0;i<50;i++){
                Map<String,Object> object = new HashMap<>();
                object.put("index",i);
                list.add(object);
            }
     
            // 初始化计时器
            CountDownLatch cdl = new CountDownLatch(list.size());
            System.out.println("====== 线程开始 =====");
     
            // 遍历
            for(Map<String,Object> object:list){
                // 开启线程
                executor.submit(new Runnable() {
                    @Override
                    public void run() {
                        try {
                            Thread t = Thread.currentThread();
                            String name = t.getName();
                            // 模拟运行耗时
                            Thread.sleep(500);
                            System.out.println(name+":执行到"+object.get("index"));
                            object.put("status","已经执行过");
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        // 闭锁-1
                        cdl.countDown();
                    }
                });
            }
            
            // 调用闭锁的await()方法,该线程会被挂起,它会等待直到count值为0才继续执行
            // 这样我们就能确保上面多线程都执行完了才走后续代码
            cdl.await();
            
            //关闭线程池
            executor.shutdown();
            System.out.println("====== 线程结束 =====");
     
            // 校验多线程正确性
            for(Map<String,Object> object:list){
                System.out.println(object.get("index") + ":" + object.get("status"));
            }
     
        }
    }

    Utilisez le multi-thread dans ; boucle for

    Chaque service est responsable d'une entreprise. Pour effectuer des opérations répétées plusieurs fois, une boucle for doit être utilisée, par exemple parcourir une collection d'identifiants stockés et créer quelque chose pour chaque identifiant.

    Mais utiliser un seul thread pour exécuter des tâches fera perdre beaucoup de temps à attendre que la dernière tâche soit exécutée, et une fois qu'une erreur est signalée lors d'une exécution, la tâche cessera de s'exécuter, ce qui ne répond évidemment pas à nos exigences.

    Dans ce cas, utilisez le multi-threading dans la boucle for.

    import java.util.concurrent.LinkedBlockingQueue;
    import java.util.concurrent.ThreadPoolExecutor;
     
    public class Test {
     
        /**
         * ThreadPool 自定义一个线程池
         */
        private static final ThreadPoolExecutor EXECUTOR = new ThreadPoolExecutor(2, 4, 60L, TimeUnit.SECONDS, new LinkedBlockingQueue<>(60000));
     
        public static void main(String[] args) {
            //自己的数组或集合,这里不再进行填充
            String[] ids = new String[10]
            
            for (String id : ids) {
                EXECUTOR.execute(new Runnable() {
                    @Override
                    public void run() {
                        try {
                            //需要执行的业务逻辑
                            System.out.println("业务逻辑正在执行");
                        } catch (Exception e) {
                            //todo
                        }
                    }
                });
            }
        }
     
    }

    Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

    Déclaration:
    Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer