动态分配线程数目-保证系统负载

package com.zte.sunquan.demo.executor;

import com.google.common.util.concurrent.AtomicDouble;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import com.sun.management.OperatingSystemMXBean;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.management.ManagementFactory;
import java.lang.management.MemoryMXBean;
import java.lang.management.MemoryUsage;
import java.lang.management.ThreadMXBean;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;


/**
 * Created by sunquan on 2017/4/15.
 */
public final class DynamicExecutorFactory {
    private static final Logger logger = LoggerFactory.getLogger(DynamicExecutorFactory.class);
    private static final int DEFAULT_THRESHOLD = 4;
    private static AtomicDouble atomicDouble = new AtomicDouble(0L);
    private static final boolean USELOAD = false;
    private static final String GROUP_NAME = "dynamic-factory";

    private DynamicExecutorFactory() {
    }

    static {
        backgroundUpdate();
    }

    public static ExecutorService getExecutorService(int expiredNum) {
        return getExecutorService(USELOAD, expiredNum);
    }

    public static ExecutorService getExecutorService(boolean useLoad, int expiredNum) {
        int threads;
        if (useLoad) {
            threads = getThreadNum(expiredNum);
        } else {
            threads = expiredNum <= DEFAULT_THRESHOLD ? expiredNum : DEFAULT_THRESHOLD;
        }
        logger.info("create threads={} and expiredNum={}", threads, expiredNum);
        return Executors.newFixedThreadPool(threads, GroupedThreadFactory.groupedThreadFactory(GROUP_NAME));
    }

    private static int getThreadNum(int expiredNum) {
        if (expiredNum <= 2)
            return expiredNum;
        //注意第一次调用时,默认只给1个线程
        int threadNum = (int) (expiredNum * atomicDouble.get());
        return threadNum < 1 ? 1 : threadNum;
    }

    private static ThreadFactory getFactory(String groupName, String name, boolean daemon) {
        ThreadFactoryBuilder threadFactoryBuilder = new ThreadFactoryBuilder();
        threadFactoryBuilder.setNameFormat(name)
                .setThreadFactory(GroupedThreadFactory.groupedThreadFactory(groupName));
        threadFactoryBuilder.setDaemon(daemon);
        return threadFactoryBuilder.build();
    }

    private static void backgroundUpdate() {
        ScheduledExecutorService executorService = Executors.newSingleThreadScheduledExecutor(getFactory("dynamic-load-schedule", "s1", true));
        executorService.scheduleWithFixedDelay(new CalSysLoadTask(), 10, 3, TimeUnit.SECONDS);
    }

    private static class CalSysLoadTask implements Runnable {
        final OperatingSystemMXBean operatingSystemMXBean = ManagementFactory.getPlatformMXBean(OperatingSystemMXBean.class);
        final MemoryMXBean memoryMXBean = ManagementFactory.getMemoryMXBean();
        final ThreadMXBean thread = ManagementFactory.getThreadMXBean();

        @Override
        public void run() {
            double systemCpuLoad = 0;
            double processCpuLoad = 0;
            double usage = 0;
            double threadRadio = 0;
            double swapSpaceRadio = 0;
            try {
                systemCpuLoad = operatingSystemMXBean.getSystemCpuLoad();
                processCpuLoad = operatingSystemMXBean.getProcessCpuLoad();

                MemoryUsage memoryUsage = memoryMXBean.getHeapMemoryUsage();
                usage = (double) memoryUsage.getUsed() / (double) memoryUsage.getCommitted();

                double threadCount = (double) thread.getThreadCount();
                double peakThreadCount = (double) thread.getPeakThreadCount();
                threadRadio = 0L;
                if (threadCount < peakThreadCount) {
                    threadRadio = 1 - threadCount / peakThreadCount;
                } else {
                    threadRadio = 1 - peakThreadCount / threadCount;
                }

                long freeSwapSpaceSize = operatingSystemMXBean.getFreeSwapSpaceSize();
                long totalSwapSpaceSize = operatingSystemMXBean.getTotalSwapSpaceSize();
                swapSpaceRadio = (double) freeSwapSpaceSize / (double) totalSwapSpaceSize;
                atomicDouble.set((1 - systemCpuLoad) * 0.2 + (1 - processCpuLoad) * 0.2
                        + (1 - usage) * 0.1 + threadRadio * 0.3 + swapSpaceRadio * 0.2);
            } catch (Exception e) {
                logger.error("backgroundUpdate error =", e);
            } finally {
                StringBuilder stringBuilder = new StringBuilder();
                stringBuilder.append("systemCpuLoad:").append(systemCpuLoad)
                        .append("processCpuLoad:").append(processCpuLoad)
                        .append("usage:").append(usage)
                        .append("threadRadio:").append(threadRadio)
                        .append("swapSpaceRadio:").append(swapSpaceRadio)
                        .append("load:").append(atomicDouble.get());
                logger.info(stringBuilder.toString());
            }
        }
    }
}


package com.zte.sunquan.demo.executor;

import com.google.common.collect.Maps;

import java.util.Map;
import java.util.concurrent.ThreadFactory;

/**
 * Created by sunquan on 2017/4/15.
 */
public class GroupedThreadFactory implements ThreadFactory {
    private final ThreadGroup group;
    private static final Map<String, GroupedThreadFactory> FACTORIES = Maps.newConcurrentMap();

    public GroupedThreadFactory(ThreadGroup group) {
        this.group = group;
    }

    @Override
    public Thread newThread(Runnable r) {
        return new Thread(group, r);
    }

    public static GroupedThreadFactory groupedThreadFactory(String groupName) {
        //oscp/a/b/c
        GroupedThreadFactory groupedThreadFactory = FACTORIES.get(groupName);
        if (groupedThreadFactory != null) {
            return groupedThreadFactory;
        }

        int i = groupName.lastIndexOf("/");
        if (i > 0) {
            String parentGroupName = groupName.substring(0, i);
            ThreadGroup parentGroup = groupedThreadFactory(parentGroupName).getGroup();
            groupedThreadFactory = new GroupedThreadFactory(new ThreadGroup(parentGroup, groupName));
        } else {
            groupedThreadFactory = new GroupedThreadFactory(new ThreadGroup(groupName));
        }
        FACTORIES.putIfAbsent(groupName,groupedThreadFactory);
        return groupedThreadFactory;
    }

    public ThreadGroup getGroup() {
        return group;
    }
}

测试:

package com.zte.sunquan.demo.executor;

import com.google.common.util.concurrent.ThreadFactoryBuilder;
import org.junit.Test;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadFactory;

/**
 * Created by sunquan on 2017/4/15.
 */
public class ExecutorTest {

    public ThreadFactory getFactory(String groupName, String name) {
        ThreadFactoryBuilder threadFactoryBuilder = new ThreadFactoryBuilder();
        threadFactoryBuilder.setNameFormat(name)
                .setThreadFactory(GroupedThreadFactory.groupedThreadFactory(groupName));
        return threadFactoryBuilder.build();
    }

    @Test
    public void test() throws InterruptedException {
        //对每一个事件都要定义其所在的事件组,事件名,并保持相关继承关系
        CountDownLatch countDownLatch = new CountDownLatch(1);
        ExecutorService executorService =
                Executors.newFixedThreadPool(2, getFactory("sqGroup/a/b", "sqThread%d"));
        executorService.execute(() -> {
            System.out.println("hello,world");
            System.out.println("ThreadParentGroup:" + Thread.currentThread().getThreadGroup().getParent().getName());
            System.out.println("ThreadGroup:" + Thread.currentThread().getThreadGroup().getName());
            System.out.println("ThreadName:" + Thread.currentThread().getName());
            countDownLatch.countDown();
        });
        countDownLatch.await();
        System.out.println(Thread.currentThread().getThreadGroup().getName());

    }

    @Test
    public void dTest() throws InterruptedException {
        ExecutorService executorService = DynamicExecutorFactory
                .getExecutorService(true, 50);//只有1个线程
        Thread.sleep(15000);
        DynamicExecutorFactory.getExecutorService(true, 50);//压力计算得26个线程
//        Thread.currentThread().join();
    }

}


### 回答1: Linux多线程矩阵相乘是一种利用Linux操作系统的多线程技术来实现矩阵相乘的方法。通过将矩阵分成多个小块,每个小块分配给一个线程进行计算,可以大大提高计算效率。同时,Linux操作系统提供了丰富的多线程API,如pthread库,可以方便地实现多线程编程。 ### 回答2: 矩阵相乘是矩阵运算中最常见的一个操作,它在多种科学计算和工程计算应用中都有广泛的应用。在实际应用场景中,矩阵相乘的规模通常都非常大,因此优化矩阵相乘的效率对于提升运算速度和效率至关重要。在linux系统中,使用多线程技术可以有效地优化矩阵相乘的效率。 多线程是指在同一程序内部,同时存在多个线程,每个线程完成不同的任务,从而提高程序的同时处理多个任务的能力,提高运算效率。在矩阵相乘程序中,可以将矩阵按照区域进行划分,将计算任务分给不同的线程执行,从而实现多线程计算。 在使用多线程技术的矩阵相乘程序中,需要注意以下几个点: 1.矩阵分块技术。将矩阵按照分块的方式进行划分,可以使得每个线程只需要处理部分数据,从而减少了计算量,提高了程序的运行速度。 2.线程数目的选择。线程数目过多会增加线程切换的开销,降低程序效率;线程数目过少会导致计算和通信负载较大,也会影响程序的执行效率。因此,在多线程矩阵相乘程序中,需要合理选择线程数目,以达到最佳的程序效率。 3.线程同步技术。由于多个线程会同时访问共享数据,为避免数据竞争或冲突产生,需要采用线程同步技术来保证数据的正确性。 总之,利用多线程技术可以优化矩阵相乘的效率,提高程序的运算速度和效率。在实际应用中,需要根据具体情况合理选择线程数目,并采用适当的线程同步技术来保证程序的正确性和稳定性。 ### 回答3: 在Linux系统中,利用多线程技术来进行矩阵相乘可以大大提高程序的效率。多线程是指在一个进程中有多个线程同时运行,而每个线程都有自己的执行序列和程序执行状态。对于矩阵相乘问题,我们可以将矩阵分成多个块,每个块可以由一个线程进行运算。 在具体实现上,首先需要将要运算的矩阵读入内存中,并将其分成n个块,然后创建n个线程来分配任务,每个线程分别计算一个块的结果,最终合并结果得到最终的矩阵。在合并结果时,可以借助另外的线程来完成,这样可以进一步提高程序的效率。 在进行多线程编程时,需要注意线程间的同步问题。常见的同步方式有信号量、互斥锁、条件变量等,需要根据具体情况选择合适的同步机制。同时,应该尽可能避免资源竞争问题,一方面可以提高程序的效率,另一方面可以避免出现不可预期的错误。 需要注意的一点是,在Linux系统中使用多线程技术进行矩阵相乘并不一定比使用单线程技术效率更高。具体的效率取决于矩阵的大小、分块的大小、线程数量等因素。因此,在使用多线程技术时需要进行充分的测试和优化,以达到最佳的效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值