Programação simultânea Java-Fork / Join

Fork / Join

Fork / Join é uma nova implementação de pool de threads adicionada ao JDK 1.7. Ela incorpora uma ideia de dividir para conquistar e é adequada para operações com uso intenso de CPU
que podem dividir tarefas .

  • A chamada divisão de tarefas consiste em dividir uma tarefa grande em tarefas menores com o mesmo algoritmo até que ela não possa ser dividida e possa ser resolvida diretamente . Alguns cálculos relacionados à recursão, como merge sort, sequência de Fibonacci, podem ser resolvidos pela ideia de dividir para conquistar.
  • Fork / Join adiciona multi-threading com base em dividir e conquistar, e a decomposição e fusão de cada tarefa pode ser completada por diferentes threads , o que melhora ainda mais a eficiência computacional.
  • Fork / Join criará um pool de threads com o mesmo tamanho que o número de núcleos de cpu por padrão.

exemplo

Encontre a soma dos inteiros entre 1 e n.

import lombok.extern.slf4j.Slf4j;
import java.util.concurrent.ForkJoinPool;
import java.util.concurrent.RecursiveTask;

@Slf4j(topic = "c.TestForkJoin2")
public class TestForkJoin2 {
    
    

    public static void main(String[] args) {
    
    
        ForkJoinPool pool = new ForkJoinPool(4);
        System.out.println(pool.invoke(new MyTask(5)));

        // new MyTask(5)  5+ new MyTask(4)  4 + new MyTask(3)  3 + new MyTask(2)  2 + new MyTask(1)
    }
}

// 1~n 之间整数的和
@Slf4j(topic = "c.MyTask")
class MyTask extends RecursiveTask<Integer> {
    
    

    private int n;

    public MyTask(int n) {
    
    
        this.n = n;
    }

    @Override
    public String toString() {
    
    
        return "{" + n + '}';
    }

    @Override
    protected Integer compute() {
    
    
        // 如果 n 已经为 1,可以求得结果了
        if (n == 1) {
    
    
            log.debug("join() {}", n);
            return n;
        }

        // 将任务进行拆分(fork)
        AddTask1 t1 = new AddTask1(n - 1);
        t1.fork();
        log.debug("fork() {} + {}", n, t1);

        // 合并(join)结果
        int result = n + t1.join();
        log.debug("join() {} + {} = {}", n, t1, result);
        return result;
    }
}



Resultado

12:50:07.716 c.MyTask [ForkJoinPool-1-worker-1] - fork() 5 + {4}
12:50:07.716 c.AddTask [ForkJoinPool-1-worker-0] - fork() 2 + {1}
12:50:07.716 c.AddTask [ForkJoinPool-1-worker-3] - fork() 3 + {2}
12:50:07.716 c.AddTask [ForkJoinPool-1-worker-2] - fork() 4 + {3}
12:50:07.721 c.AddTask [ForkJoinPool-1-worker-0] - join() 1
12:50:07.722 c.AddTask [ForkJoinPool-1-worker-0] - join() 2 + {1} = 3
12:50:07.722 c.AddTask [ForkJoinPool-1-worker-3] - join() 3 + {2} = 6
12:50:07.722 c.AddTask [ForkJoinPool-1-worker-2] - join() 4 + {3} = 10
12:50:07.722 c.MyTask [ForkJoinPool-1-worker-1] - join() 5 + {4} = 15
15


Insira a descrição da imagem aqui
Melhoria do processo de operação gráfica

import lombok.extern.slf4j.Slf4j;
import java.util.concurrent.ForkJoinPool;
import java.util.concurrent.RecursiveTask;

public class TestForkJoin {
    
    

    public static void main(String[] args) {
    
    
        ForkJoinPool pool = new ForkJoinPool(4);
//        System.out.println(pool.invoke(new AddTask1(5)));
        System.out.println(pool.invoke(new AddTask3(1, 5)));
    }
}

@Slf4j(topic = "c.AddTask")
class AddTask1 extends RecursiveTask<Integer> {
    
    

    int n;

    public AddTask1(int n) {
    
    
        this.n = n;
    }

    @Override
    public String toString() {
    
    
        return "{" + n + '}';
    }

    @Override
    protected Integer compute() {
    
    
        if (n == 1) {
    
    
            log.debug("join() {}", n);
            return n;
        }
        AddTask1 t1 = new AddTask1(n - 1);

        t1.fork();
        log.debug("fork() {} + {}", n, t1);
        int result = n + t1.join();
        log.debug("join() {} + {} = {}", n, t1, result);
        return result;
    }
}

@Slf4j(topic = "c.AddTask")
class AddTask2 extends RecursiveTask<Integer> {
    
    

    int begin;
    int end;

    public AddTask2(int begin, int end) {
    
    
        this.begin = begin;
        this.end = end;
    }

    @Override
    public String toString() {
    
    
        return "{" + begin + "," + end + '}';
    }

    @Override
    protected Integer compute() {
    
    
        if (begin == end) {
    
    
            log.debug("join() {}", begin);
            return begin;
        }
        if (end - begin == 1) {
    
    
            log.debug("join() {} + {} = {}", begin, end, end + begin);
            return end + begin;
        }
        //int mid = (end + begin) / 2;
        int mid = begin+(end-begin)/2;

        AddTask2 t1 = new AddTask2(begin, mid - 1);
        t1.fork();
        AddTask2 t2 = new AddTask2(mid + 1, end);
        t2.fork();
        log.debug("fork() {} + {} + {} = ?", mid, t1, t2);

        int result = mid + t1.join() + t2.join();
        log.debug("join() {} + {} + {} = {}", mid, t1, t2, result);
        return result;
    }
}

@Slf4j(topic = "c.AddTask")
class AddTask3 extends RecursiveTask<Integer> {
    
    

    int begin;
    int end;

    public AddTask3(int begin, int end) {
    
    
        this.begin = begin;
        this.end = end;
    }

    @Override
    public String toString() {
    
    
        return "{" + begin + "," + end + '}';
    }

    @Override
    protected Integer compute() {
    
    
        if (begin == end) {
    
    
            log.debug("join() {}", begin);
            return begin;
        }
        if (end - begin == 1) {
    
    
            log.debug("join() {} + {} = {}", begin, end, end + begin);
            return end + begin;
        }
        int mid = (end + begin) / 2;

        AddTask3 t1 = new AddTask3(begin, mid);
        t1.fork();
        AddTask3 t2 = new AddTask3(mid + 1, end);
        t2.fork();
        log.debug("fork() {} + {} = ?", t1, t2);

        int result = t1.join() + t2.join();
        log.debug("join() {} + {} = {}", t1, t2, result);
        return result;
    }
}

Ícone
Insira a descrição da imagem aqui

Acho que você gosta

Origin blog.csdn.net/e891377/article/details/109198812
Recomendado
Clasificación