进程,线程,纤程的概念,以及线程和纤程对于高并发小案例的时间效率对比(小实验)

本文主要介绍Java中的纤程,以及一个测试的小案例,体现纤程的牛逼之处

1.进程和线程的区别?
进程是操作系统分配资源的基本单位,
线程是执行调度的基本单位。所谓的分配资源最重要的是为进程分配内存空间,线程是共享内存空间,没有自己独立的内存空间。
2.纤程:线程中的线程,纤程的调度和切换是不需要经过OS的,是在用户态中完成的,而线程的创建和调度都是需要OS操作系统,所以在执行效率上纤程要高效的多。

引申出一个小问题:为什么说跟内核打交道效率要低?
因为系统调用(我们这里说的是软中断),也就是 int 0x80,首先需要在bx,cx,dx,si,di不同的寄存器中放置参数,然后在ax寄存器中放置系统调用号(对应操作系统函数编号),在调用int 0x80中断,操作系统执行完之后,将返回值放入ax中,再由应用程序去ax中取值,这就应用程序调用操作系统内核的大致过程。所以与用户态上的调用相比,效率低下。

纤程的实现例子:
1.10000个线程,处理calc(随便写的计算的方法)

public class ThreadTest {
    public static void main(String[] args) throws InterruptedException{
        Long startTime = System.currentTimeMillis();
        Runnable runnable = new Runnable() {
            @Override
            public void run(){
                calc();
            }
        };
        int size =10000;
        Thread[] threads = new Thread[size];
        for(int i=0;i<size;i++){
            threads[i] = new Thread(runnable);
        }
        for (int i=0;i<size;i++){
            threads[i].start();
        }
        for(int i=0;i<size;i++){
            threads[i].join();
        }
        Long endtime = System.currentTimeMillis();
        System.out.println("10000个线程计算calc方法耗时:"+(endtime-startTime));
    }

    /**
     * 只是一个计算方法没有实际意义
     */
    private static void calc(){
        int result =0;
        for (int i =0;i<10000;i++){
            for(int j=0;j<200;j++){
                result +=i;
            }
        }
    }

2.使用纤程,创建10000个纤程
首先需要引入quasar.jar
最新版本是0.8的,但是我用的时候会出现问题。

<dependency>
	<groupId>co.paralleluniverse</groupId>
	<artifactId>quasar-core</artifactId>
	<version>0.7.9</version>
</dependency>
public class FiberTest {
    public static void main(String[] args) throws Exception{
        long start = System.currentTimeMillis();
        SuspendableRunnable suspendableRunnable = new SuspendableRunnable() {
            @Override
            public void run() throws SuspendExecution, InterruptedException {
                calc();
            }
        };
        int size = 10000;
        Fiber<Void>[] fibers = new Fiber[size];
        for(int i=0;i<size;i++){
            fibers[i] = new Fiber<>(suspendableRunnable);
        }
        for(int i=0;i<size;i++){
            fibers[i].start();
        }
        for(int i=0;i<size;i++){
            fibers[i].join();
        }
        long end = System.currentTimeMillis();
        System.out.println("10000个请求使用纤程耗费时间:"+(end-start));
    }
    /**
     * 只是一个计算方法没有实际意义
     */
    private static void calc(){
        int result =0;
        for (int i =0;i<10000;i++){
            for(int j=0;j<200;j++){
                result +=i;
            }
        }
    }
}

3.进一步压榨cpu的资源,创建100个线程,每个线程中创建1000个纤程,可以把第二步中的size设置为100000,在跟第3步进行比较。

public class FiberTest2 {
    public static void main(String[] args) throws InterruptedException{
        long start = System.currentTimeMillis();
        Thread[] t = new Thread[100];
        for(int i=0;i<10;i++){
            t[i] = new Thread(getThread());
        }
        for (int i=0;i<10;i++){
            t[i].start();
        }
        for (int i=0;i<10;i++){
            t[i].join();
        }
        long endtime = System.currentTimeMillis();
        System.out.println("使用线程池+纤程的方式耗时:"+(endtime-start));
    }
    public static Runnable getThread() {
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                //创建1000个纤程
                Fiber<Void>[] fibers = new Fiber[1000];
                SuspendableRunnable suspendableRunnable = new SuspendableRunnable() {
                    @Override
                    public void run() throws SuspendExecution, InterruptedException {
                        calc();
                    }
                };
                for (int i = 0; i < fibers.length; i++) {
                    fibers[i] = new Fiber<>(suspendableRunnable);
                }
                for (int i = 0; i < fibers.length; i++) {
                    fibers[i].start();
                }
                for (int i = 0; i < fibers.length; i++) {
                    try {
                        fibers[i].join();
                    } catch (Exception e) {

                    }

                }
            }
        };
        return runnable;
    }
    /**
     * 只是一个计算方法没有实际意义
     */
    private static void calc(){
        int result =0;
        for (int i =0;i<10000;i++){
            for(int j=0;j<200;j++){
                result +=i;
            }
        }
    }

经过测试,第三种方式效率更好,你如果电脑可以的话,可以创建10万个线程,看跟这种线程和纤程混合使用的时间进行比较。但是很遗憾,这种通过类库的支持,在做高并发的时候会存在问题,内置支持线程(Fiber)的语言:Kotlin,Scala,go,纤程适合计算的业务,如果是阻塞时间较长的IO,并不适合。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值