多线程案例

news2025/1/15 12:46:13

 日升时奋斗,日落时自省

目录

1、单例模式

1.1、饿汉模式

1.2、懒汉模式

1.3、饿汉和懒汉的线程安全

2、生产者消费者模型

2.1、理论解释

2.2、优势

2.3、阻塞队列代码解析

2.4、生产者消费者代码解析

2.5、简单实现阻塞队列代码解析

3、定时器

3.1、定时器的使用

3.2、自主实现简单的定时器

1、单例模式

单例模式:是设计模式的一种

模式如何理解,打过小游戏的友友们都知道“攻略”这个词应对不同的BOSS,其实模式也就是一种解决方法,也如下棋一样,会有棋谱如何解决不同的棋局

模式这里挺起来 有点类似于框架,仅仅是类似,因为限制不同

模式与框架的区别

(1)框架是硬性,大佬已经写好的,不按照框架来写,代码跑不起来(不得不遵守)

(2)模式是软性,可以不遵守也能跑起来,但是代码的可可读性,可维护性,可扩展性都比较低(可以不遵守)

单例如何解释:单个 实例对象 在某些情况下只能创建一个实例不应该创建多个实例,单例模式只是在这里限定,并不是说一定要用

创建一个实例,这件事情是程序员自己做的,也可以不用单例模式,自己创建一个,再也不创建了也行(这样的可以称为“君子协定”)

举一个例子解释:

在古代是可以一夫多妻,但是也有只爱一人的,只娶了一个妻子的,这就是君子协定,没有外界因素束缚,他可以娶多个,但是他以君子协定(口头协议)来约束自己(相当于程序员不用模式)

在现代是一夫一妻,只能娶一个人,不只是君子协定,还有法律要求(使用模式)

使用单例模式后,想在创建多个实例都难

单例模式,就是针对上述的需求场景进行了更强制的保证,通过巧用java的现有语法,达成了某个类,只能被创建出一个实例,这样的效果(创建多个实例就报错提醒)

之前的博客中写到JDBC数据库连接,DataSource这个就很适合使用单例模式,为什么呢?

因为只需要创建一个数据源,单例模式不是正好嘛

实现单例模式有很多种方式,这里介绍最常见的两种方式(饿汉模式和懒汉模式)

1.1、饿汉模式

为什么叫做饿汉:这里说的是单例模式,创建一个实例的先后来定义的名字,饿汉在类加载阶段创建实例,以饿汉定名

类加载阶段:运行一个java程序,就需要让java进程能够找到并读取对应的.class文件,就会读取文件内容,并解析,构造类对象...这一系列的过程操作称为 类加载(有点像预处理)

class Singleton{   //本身就是安全的, 因为只读就不会有线程安全问题
    //创建一个实例 ,这就是饿汉模式,为什么有static来修饰,是为了能够与类联系起来
    //static使这里的属性具有唯一性
    private static Singleton instance=new Singleton();
    
    //如果需要使用这个唯一的实例,统一通过Singleton.getInstance() 方式进行来获取
    public static Singleton getInstance(){   //静态方法是为了在main直接能够掉用
        return instance;
    }

    //为了避免Singleton 类不小心被复制出来多份
    //把构造方法设置为private 在类外就不能在通过new的方式创建出来Singleton类来创建
    private Singleton(){
    }
}
public class ThreadDemo_Single {
    public static void main(String[] args) {
//        Singleton s=new Singleton(); //这里就会报错哦
        Singleton s2=Singleton.getInstance();
        Singleton s1=Singleton.getInstance();
        System.out.println(s1==s2);
    }
}

这里我们先附一下代码,解释当前代码

 私有化的构造方法 :如果在外面创建了一个实例化对象(看一下效果)

 这里能看到报错的结果,因为私有化构造方法的限制,保证了只能通过调用方法的形式进行操作实例化对象。

private static Singleton instance=new Singleton();

这个属性和实例无关,而是和类相关

java代码中每个类,都会在编译完成后得到.class文件,JVM运动时就会加载这个.class文件读取其中的二进制指令,并且在内存中构造出对应的类对象(Singleton.class)

由于类对象 在java进程里,只是有唯一一份的,因为类对象内部的 类属性也是唯一一份了

1.2、懒汉模式

为什么叫做懒汉:以为在一开始并没有创建对象,等有人来调用方法的时候,如果没有创建对象,在进行创建;(代码大体上和饿汉模式有几分相似)

 那懒汉模式好嘛,当然嘛,懒汉不是说是贬义,而是褒义,效率更胜,懒在程序和代码里能提高效率,人因为懒才发明了计算机,为了便捷,才发明了更多方便的东西,才有现在的科技

1.3、饿汉和懒汉的线程安全

那友友们觉得谁是安全的呢?

饿汉线程更安全,为什么这么说?

 下面来看一下懒汉模式为啥不安全:

 那我们来加个锁,看一下对不对

 那我们知道要加锁,那如何进行加锁才能解决问题

 那现在线程安全问题也解决了,那是不是就安全,其实还有其他问题,内存可见性问题

假设有很多线程,都去进行getInstance,这个时候,是否就会有编译优化的风险,只有第一次读是真正的读内存,后续都是读寄存器内存可见性问题

也就是说每次在寄存器可能读的都是instance==null,仍然会有问题,相当于加锁没有起效

另外,还会涉及到指令重排序问题

instance=new Singleton();

拆分成三个步骤:

1、申请内存空间

2、调用构造方法,把这个内存空间初始化成一个合理的对象

3、把内存空间的地址赋值给instance引用

正常情况下,123按顺序走,编译器为了提升效率也可能调整代码执行顺序(单线程无所谓)

如果是多线程环境,t1按照132步骤执行的话,13刚刚执行结束,CPU就把t2调度来执行,t2就相当于直接返回instance引用并且可能会尝试使用引用中的属性。

但是t1的2还没有执行完,t2拿到了非法的对象,还没构造完成的不完整对象

volatile解决内存可见性问题

(1)解决内存可见性

(2)禁止指令重排序

下面是附的完整版代码(解决线程问题,解决内存可见性问题,解决指令重排序问题)

class SingletonLazy{   //可以写的就会出现线程安全问题
    //static 使所有的属性具有唯一性  ,懒汉模式就是不会直接创建对象,比较懒,但是懒有懒得好处,它节省资源
    private volatile static SingletonLazy instance=null;
    public static SingletonLazy getInstance(){
        if(instance==null){  //如果满足创建对象条件,就互加锁,
            synchronized (SingletonLazy.class) {
                if(instance==null){
                    instance=new SingletonLazy();
                }
            }
        }
        return instance;
    }
    //单例模式防止 在外面的类中new 实例化
    private SingletonLazy(){
    }
}
public class ThreadDemo_Single2 {
    public static void main(String[] args) {
        SingletonLazy s=SingletonLazy.getInstance();
        SingletonLazy s1=SingletonLazy.getInstance();
        System.out.println(s1==s);
    }
}

2、生产者消费者模型

2.1、理论解释

这个模型一下解释不清楚:在学习这个模型之前先要知道阻塞队列

阻塞队列也是一个队列(数据结构)

先进先出 是队列的一个特点 但是不是所有队列都遵循这个规则

数据结构中的优先级队列(PriorityQueue) 

还有就是这里要提到的阻塞队列,虽然是先进先出,但是同时存在其他的特点(顾名思义)

阻塞(针对多线程,如果是单线程就没有实际意义了):

(1)如果队列为空了,执行出队列的操作,就会阻塞等待,阻塞到另一个线程往队列中添加元素(队列不为空),阻塞结束当前线程继续执行

(2)如果队列满了,执行入队列的操作,就会阻塞等待,阻塞到另一个线程从队列取走元素(队列不满),阻塞结束当前线程继续执行

这里提一下消息队列:也是特殊队列,因为它就是在阻塞队列基础上进行的,加上一个“消息的类型”按照制定类别进行先进先出,此时就构成了一个消息队列,更是一个数据结构,消息队列的应用比较频繁,也被单独实现成了一个程序,该程序可以通过网络的方式和其他程序进行通信,单独部署到一组服务器上,存储能力和转发能力都大大提升了。

以上提及的消息队列也成了一个组件“中间件”

rabbit mq就是一种,还有active mq,rocket mq,kafka都是,因为都是消息队列的应用所以,使用大同小异。

基于阻塞队列的特性,可以实现“生产者和消费者模型”

什么是生产者消费者模型?

简单的举一个例子:

卖货这个流程 简介一下

(1)就是生产 出来  消费者买,生产出来的货正常量,但是恰逢过年大家买年货消费者就多,一购而空,这时候消费者还想要卖货就要等着生产商这边再产货才行,所以需要就阻塞等待.

(2)过年过完了,但是当时生产者囤货又囤的太多了,此时的生产量太大,消费者少了,这下生产者就需要等待了

其实这个例子与前面的阻塞队列在我理解以来是同一个意思。

2.2、优势

1、实现了发送方和接收方之间的“解耦”  

注:解耦就是降低耦合的过程(写代码尽量能够让代码低耦合 、高内聚)

低耦合: 就是将代码模块之间的联系降低 防止一个个方面的代码出现问题导致整个代码都瘫痪

高内聚:就是将代码元素之间的联系程度提高(如 在找衣服的时候,衣服就如元素,衣柜就如代模块,每一个人的衣服都应该在规定的衣柜里就容易找,这就高内聚,不同衣柜就是模块之间低耦合

在实现使用项目中,服务器之间相互调用可以看出(图解)

没有使用“生产者和消费者模型” 时 

使用模型的情况:

 总结:生产者和消费者模型降低了耦合性,同时可以在极端条件下阻塞进程,减少代码带来不必要的崩坏,在A服务器受损时不会影响到服务器B和C,如果B或者C服务器崩坏也不会影响服务器A。

2、生产者消费者模型 ,第二个可以做到“削峰填谷”,保证系统的稳定性

简单表示一下波峰填谷:

 在这里就是阻塞的意思:为了让整个服务器维持平衡,如果充值的数据量一次性暴增,阻塞队列就会在一定数据的时候进行阻塞,阻塞队列中的数据再在通过其他服务器慢慢取出,不会导致服务器直接崩溃,这里的阻塞等待相当于提供了一个缓冲的作用

注:在大型的项目上很实用,因为数据量是不可预知的,服务区要撑得住,所以阻塞队列就是可以满足该情况,就像如果某大型平台突然停止运行,就会导致同类小型的竞争平台数据量暴增导致,如果没有做好措施,服务器,平台,或者网页就可能崩溃,阻塞队列就是解决这种突发数据量增长问题。

2.3、阻塞队列代码解析

阻塞队列在java标准库中使用是BlockingQueue<> 创建对象有数组(ArrayBlockingQueue)、链表(LinkedBlockingQueue)和堆(PriorityBlockingQueue)类型的。

他们是队列自然也包含队列的基本方法offer(),peek(),poll()但是不具有阻塞功能

阻塞功能的主要存在于 put 入队列         take 出队列 这两个方法

以下代码可以尝试如果队列中空时,出队列会进行等待,程序并不会结束,当然这里表现的并不明显,之后,在写一个简单的生产者消费者模型,观察这种情况

public class BlockQueue {
    //阻塞队列
    public static void main(String[] args) throws InterruptedException {
        //阻塞队列  也分为三种不同的队列构成, 数组 链表 和 堆
        BlockingQueue<String> blockingQueue=new LinkedBlockingQueue<>();
        /*put   是入队的
        * take  是出队
        * */
        blockingQueue.put("hello");
        String res=blockingQueue.take();
        System.out.println(res);
        res=blockingQueue.take();
        //产生越界不会进行出问题  会进行等待  观察后可以看出 运行并没有结束
        System.out.println(res);
    }
}

2.4、生产者消费者代码解析

大体思路:由多线程构造出该模型,满足生产元素空时 消费者阻塞等待,消费者满足时 生产者阻塞等待

首先:创建一个阻塞队列 下来才能满足阻塞等待这个条件,这里给阻塞队列设置的是8个空间,便于测试生产速度如果快的话,后来仍然会与消费者的速度同步进行,因为会阻塞队列会进行阻塞

 BlockingQueue<Integer> blockingQueue=new LinkedBlockingQueue<>(8);

接下来创建先生产者或者消费者都行(代码上附加简单注释)

 这里先写生产者 线程 便于大家理解  

//便于理解先创生产者  线程
        Thread producer=new Thread(()->{
            int count=0;   //这里就简单记为  生产者生产的数据
            while(true){
                try {
                    //将生产的数据 放入 阻塞队列中 
                    blockingQueue.put(count);  
                    //这里就做一个标识作为验证
                    System.out.println("生成元素 :"+ count);
                    Thread.sleep(500);
                    count++; //表示新数据
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        });
        producer.start();  //线程执行

 后写消费者的线程

 //创建消费者
        Thread customer =new Thread(()->{
            while(true){
                try {
                    //将对阻塞队列的数据进行提取
                    Integer result=blockingQueue.take();
                    //打印出来检测作为验证
                    System.out.println("消费元素 :" + result);
                    Thread.sleep(500);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        });
        customer.start();

上述代码中生产者线程的代码Thread.sleep(500);没有删除,这里展示一下删除后的运行结果,友友们可以自己试试

 2.5、简单实现阻塞队列代码解析

阻塞队列也需要先写一个普通的队列,可以是链表来写,也可以是数组来写,阻塞队列更像是循环队列,我们这里就以数组的形式进行实现,两个指针,一个头指针,一个尾指针

 循环队列写出来,基础就打好了,但是线程调度是随机的,既然写多线程就会涉及到线程安全问题。阻塞队列的细节:

(1)这里有关++的都会涉及多线程安全问题,所以直接整个循环队列都套上一个锁。

(2)原来判满条件是不能返回的,现在需要的阻塞  所以把return 换成this.wait();

(3)既然有wait就 需要notify唤醒所以在notify唤醒都是在出队列和入队列的最后,出队列时为空就会阻塞,需要入队一次,队列不为空,此时唤醒才行,同理,入队列时为队列线程就会在此阻塞,需要出队一次,队列不为满,此时唤醒

(4)上面三个问题阻塞队列已经基本完成,但是有一个多线程中存在的缺陷,就是多线程调度的时候是可以切换的,所以判满或者判空被唤醒以后可能仍然为空或者为满,但是单单就一个if语句是不能连续判断当前队列是满的还是空的,所以if需要改成while

class  MYBlockingQueue{
    private  int[] items=new int[100];
    private  int head=0;
    private  int tail=0;
    private  int size=0;

    //入队列的 代码   线程安全 离不开锁  所以在以下的操作中需要用到锁的地方很多 ,所以当前位置都需要这些东西,直接给整体都加上一个锁
    public void put(int vaule) throws InterruptedException {
        synchronized (this){
            while (size==items.length){   //防止在过程中  被唤醒后仍然是 满的
                this.wait();// (1)
            }
            items[tail] = vaule;
            tail++;
            //这里为什么不会写成 tail = (tail + 1)%itmes.length
            // 因为当前值已经进行了加加  不在需要进行加1的 操作了
            if(tail>=items.length){
                tail=0;
            }
            size++;
             this.notify();// 唤醒(2)
        }
    }
    //出队列
    public Integer take() throws InterruptedException {
        int result=0;
        synchronized (this){
            while (size==0){   //防止在过程中  被唤醒后仍然是 空的
                this.wait();            //(2)
            }
            result=items[head];
            head++;
            if(head>=items.length){
                head=0;
            }
            size--;
            this.notify(); //唤醒 (1)
        }
        return result;
    }
}

上面附的就是阻塞队列的简单实现了,可以自己用一下main函数进行测试一下生产者消费者模型。

3、定时器

3.1、定时器的使用

一种就是 指定特点时刻 提醒

另一种是 指定特定时间段之后 提醒

这里提及的定时器不是提醒什么,而是执行一个实现准备好的方法或者代码

这个也是咱们开发中一个常用的组件,尤其是网络编程的时候,很容易卡连不上,但是要及时止损,不能一直卡也不说是什么问题,让客户一直等吧。

这里java标准库里也给我们提供了定时器(Timer)

这里粗略使用一下定时器,然后简单实现一个计时器(稍微有一点点繁琐,但是不要有心理负担)

主要使用的是 schedule这个方法其中有两个参数包含了 任务 和 时间

public class TimerTest {
    //简单的了解 定时器的使用凡是
    public static void main(String[] args) {
        System.out.println("启动程序");
        Timer timer=new Timer();    //定时器  java标准库定义
        //使用方法是 schedule 包含 有两个参数 第一个是任务   第二个是 时间 单位是ms 
        //这里每个任务的时间都是不一样的,  因为只有主线程一个线程在跑这些方法,所以会按照时间最短的先跑
        //所以先跑 的是任务3 然后是任务 2  最后是 任务1
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("运行定时器任务 1");
            }
        },3000);
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("运行定时任务 2");
            }
        },2000);
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("运行定时任务 3");
            }
        },1000);
    }
}

运行结果:

 定时器是前台线程,不会因为主线程结束,而运行结束所以这里的运行结果能明显看出运行的红色方块。

3.2、自主实现简单的定时器

思路:

(1)让被注册的任务,能够在指定事件,被执行

单独在定时器内部写一个线程,让着个线程周期性的扫描,判定任务是否到时间了,如果到时间了就执行,没到时间,就再等等。

(2)一个定时器是可以注册N个任务的,N个任务会被按照最初约定的时间,按顺序执行

但是这个顺序就需要点东西了,这里用的是阻塞队列的一种堆的阻塞队列

(1)有一个扫描线程,负责判定时间/执行任务

(2)还要有一个数据结构,来保存所有被注册的任务

使用一个优先级队列来表示,但是在多线程中就需要注意线程安全,可以加锁来解决线程安全问题,此处还有个选择,标准库提供的PriorityBlockingQueue

 一个类来实现定时器  首先要有一个自定义类来为阻塞队列做一个准备,因为这里需要任务 和 时间。

class MyTask{
    private Runnable runnable;   //定义一个任务
    private  long time;   //定义一个时间

    public MyTask (Runnable runnable, long time){
        this.runnable=runnable;
        this.time=time;
    }

    //获取当前时间
    public  long getTime(){
        return time;
    }
    public  void run(){
        runnable.run();
    }
   /* @Override
    public int compareTo(MyTask o) {
        return (int) (this.time - o.time);
    }*/
}

 第一步完成了,后面就先定义好一个类来写定时器,定时器的需要一个扫描线程,和一个前面提到的阻塞队列(以堆的形式),所以下面定义好了一个空线程,和一个阻塞队列,但是阻塞队列需要时间去比较时间不是吗,所以这里就在里面写了一个比较器,用来比较时间,谁的时间小就放在堆顶,这里就相当于是一个小根堆

//定时器  需要 一个 执行方法 和 一个扫描线程
    private Thread t=null;

    //有一个阻塞优先级队列 ,来保存任务

    private PriorityBlockingQueue<MyTask> queue=new PriorityBlockingQueue<>(8,new Comparator<MyTask>(){
        @Override
        public int compare(MyTask o1, MyTask o2) {
            return (int) (o1.getTime()- o2.getTime());
        }
    });
    

 那接下来写什么呢,那就是构造方法了,指定两个参数,一个是任务 第二个是 时间,但是步骤稍微多了那么一点点(这里有一个阻塞等待,现在不说为什么,先写剩余的代码,后面在解释wait的作用)

public MyTimer(){
        t=new Thread(()->{
            while(true){
                try {
                    //取出首元素 检查队列首元素是否满足时间条件
                    //如果时间没到 就把任务塞回去
                    //另外时间到了就正常执行
                    synchronized (this){
                        MyTask myTask=queue.take();
                        Long curTime=System.currentTimeMillis();
                        //如果时间大于了我们当前的时间 就再把当前时间装进队列里
                        if(curTime<myTask.getTime()){
                            queue.put(myTask);
                            this.wait(myTask.getTime()-curTime);
                        }else{
                            //如果时间小于或者等于当前时间就把该时间的任务执行了
                            myTask.run();
                        }
                    }
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        });
        t.start();
    }

扫描线程已经写完了,思路的第一个步骤也算是走完了,但是第二步骤就是用一个方法将任务装进队列中,

//指定两个参数
    //第一就参数是  任务 内容
    //第二参数 是 任务执行多 少秒
    public void schedule(Runnable runnable,long after){
        MyTask task=new MyTask(runnable,System.currentTimeMillis() + after);
        queue.put(task);
        synchronized (this){
            this.notify();
        }
    }

这里遗留了一个问题,其实是两个问题,为什么要在线程上加锁,为什么要用wait等待。

(1)先解决wait等待问题,前面的博客中提到wait是可以有自己的时间限制的,因为如果时间不到,不能一直执行if语句从队列中弹出,再塞回去也是要有消耗的呀,所以并不能单单的用if语句再这里解决问题,所以这里使用wait确定一个等待时间自动开始,或者有新任务入队列,就会进行一次notify唤醒,从新计算等待时间

(2)然后就是加锁问题,因为线程是会进行随机调度的,所以可能线程在计算wait等待时间前就已经调度走了,在这样的情况下入队列是一个最小时间,那再回来的时候wait会进行更新吗,答案是当然不会,因为我们之前已经take出队列了,所以wait等待的时间是上一次的时间,所以加锁的目的就是为了能够保证wait能够执行完,notify唤醒通知不会被放鸽子,同时也不会导致新任务添加入队列时wait时间计算错误

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/145455.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java中String、StringBuffer 和 StringBuilder 的区别

1. String 字符串常量&#xff0c;字符串长度不可变。Java 中 String 是 immutable&#xff08;不可变&#xff09;的。 2. StringBuffer 1.如果要频繁对字符串内容进行修改&#xff0c;出于效率考虑最好使用 StringBuffer&#xff0c;如果想转成 String 类型&#xff0c;可以调…

论文笔记Neural Ordinary Differential Equations

论文笔记Neural Ordinary Differential Equations概述参数的优化连续标准化流&#xff08;Continuous Normalizing Flows&#xff09;生成式的隐轨迹时序模型&#xff08;A generative latent function time-series model&#xff09;这篇文章有多个版本&#xff0c;在最初的版…

嵌入式系统IO体系简述

前言&#xff1a; CPU的主要职责是负责运算&#xff0c;而计算机是需要各种外设的&#xff0c;否则无法和人进行交互。早期x86体系的CPU&#xff0c;需要使用前端总线&#xff08;fsb&#xff09;和北桥芯片相连&#xff0c;北桥再和南桥相连。南北桥是一种架构的划分&#xff…

数据结构与算法——算法分析(3)

算法的时间复杂度计算 算法基本操作执行的次数还会随着问题输入的数据集不同而不同 最坏时间复杂度&#xff1a;在最坏的情况下&#xff0c;算法的时间复杂度 平均时间复杂度&#xff1a;所有可能输入在等概率的情况下算法的期望运行时间 最好时间复杂度&#xff1a;在最好的…

企业内训方案|数据治理/项目管理/敏捷项目管理/产品管理

企业内训方案|数据治理/项目管理/敏捷项目管理/产品管理 》》数据治理 数据管理基础 数据处理伦理 数据治理 数据架构 数据建模和设计 数据安全 数据集成和互操作 文件和内容管理 参考数据和主数据 数据仓库和商务智能 元数据管理 数据质量 大数据和数据科学 数据管理成熟度评…

网络协议HTTP:了解Web及网络基础

文章整理自图书图解Http第一章&#xff1a;使用Http协议访问Web第二章&#xff1a;Http的诞生第三章&#xff1a;网络基础TCP/IP协议一&#xff1a;应用层二&#xff1a;传输层三&#xff1a;网络层四&#xff1a;链路层五&#xff1a;TCP/IP通信传输流第四章&#xff1a;IP、T…

http与https的区别我真的知道吗

之前每次看到类似“http与https的区别&#xff1f;”的问题时&#xff0c;都会自己思考一下答案&#xff0c;好像只是浅显地知道https比http安全&#xff0c;但究竟为什么更安全&#xff0c;却又似乎说不出个所以然&#xff0c;或者说很多细节地方自己都是不清楚的。为了搞清楚…

Linux权限shell命令以及运行原理

文章目录一、Linux权限的概念二、Linux权限管理2.1.文件访问者的分类&#xff08;角色&#xff09;2.2文件属性2.3文件访问权限的相关设置方法2.4访问者角色的修改2.5目录权限含义2.6默认权限三、粘滞位四、 shell命令以及运行原理一、Linux权限的概念 权限的概念通常是指行事…

ffmpeg录制H265格式的桌面视频

ffmpeg本身不支持H265&#xff0c;如果需要支持&#xff0c;需要事先编译出libx265&#xff0c;读者可以到libx265的官方网站https://www.videolan.org/developers/x265.html上找到下载地址&#xff0c;本人下载的是x265_3.5.tar.gz。 编译libx265时&#xff0c;定位到其目录下…

java基础 网络编程

网络编程概念&#xff1a; 让程序可以和网络上的其他设备中的程序进行数据交互。 网络通信基本模式&#xff1a; CS&#xff1a;Client-Server 自己写客户端和服务器交流 BS&#xff1a;Browser/Server 通过浏览器和服务器交流 实现网络编程关键的三要素…

python中的函数与变量

一、函数python中函数的基本格式则为:def函数名参数名函数体返回&#xff0c;python作为一门面向对象的语言&#xff0c;同样可分为类函数、实例函数。 # 定义一个函数 def add(x, y):"""函数的说明:param x: 参数x的作用:param y: 参数y的作用:return: 函数返…

碱性环境吸钯树脂技术

汞和贵金属的选择性去除回收离子交换树脂 Tulsimer CH-95S 是一款为了从工业废水中去除回收汞和贵金属而开发的螯合树脂。 Tulsimer CH-95S是一款拥有聚乙烯异硫脲官能基的大孔树脂&#xff0c;这种树脂对汞有的选择性。它也选择其他的贵金属&#xff0c;如黄金&#xff0c;铂…

消息收发弹性——生产集群如何解决大促场景消息收发的弹性降本诉求

作者&#xff1a;宸罡 产品介绍—什么是消息收发弹性 大家好&#xff0c;我是来自阿里云云原生消息团队的赖福智&#xff0c;花名宸罡&#xff0c;今天来给大家分享下阿里云 RocketMQ5.0 实例的消息弹性收发功能&#xff0c;并且通过该功能生产集群是如果解决大促场景消息收发…

JS in CSS:一键支持响应式布局

前言 如今网速不再成为适配移动端时选择响应式设计的限制因素&#xff0c;在资源充足的条件下&#xff0c;针对各端各自设计应用界面能达到应用最佳用户体验&#xff0c;毕竟不同类型的设备交互体验是不同的&#xff0c;但在团队前端资源拮据时&#xff0c;相比无脑自适应&…

基于Java+jquery+SpringMVC校园网站平台设计和实现

基于JavajquerySpringMVC校园网站平台设计和实现 博主介绍&#xff1a;5年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java毕设项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言 文末获取源码联…

临时白名单

临时白名单介绍 相关常量 临时白名单列表介绍 前两个临时白名单可以豁免后台启动Service、豁免uid后台1min后进入idle状态等&#xff0c;最后一个临时白名单可以后台启动FGS。 // 由于高优先级消息而暂时允许逃避后台检查的一组应用程序 ID&#xff0c;短信/彩信 Composite…

【Vue路由】路由守卫、生命周期钩子、路由器工作模式

文章目录生命周期钩子案例实现总结路由守卫全局路由守卫独享守卫组件内守卫总结路由器的两种工作模式总结生命周期钩子 我们在News组件列表中的第一行加一个渐变文字。同时原来的路由缓存功能也要保存。 案例分析&#xff1a; 我们实现这个渐变的效果&#xff0c;是使用周期定…

Go select底层原理

在对Channel的读写方式上&#xff0c;除了我们通用的读 i <- ch, i, ok <- ch&#xff0c;写 ch <- 1 这种阻塞访问方式&#xff0c;还有select关键字提供的非阻塞访问方式。 在日常开发中&#xff0c;select语句还是会经常用到的。可能是channel普通读写的使用频率比…

基于Node.js和vue的博客系统的设计与实现

摘要随着互联网技术的高速发展&#xff0c;人们生活的各方面都受到互联网技术的影响。现在人们可以通过互联网技术就能实现不出家门就可以在线发布博客文章&#xff0c;简单、快捷的方便了人们的日常生活。同样的&#xff0c;在人们的工作生活中&#xff0c;也需要互联网技术来…

【Java寒假打卡】Java基础-日期类对象

【Java寒假打卡】Java基础-日期类对象Date概述Date类常用成员方法SimpleDateFormat案例:秒杀活动案例&#xff1a;在当前时间加上一天时间JDK8新增日期类获取时间中的一个值LocalDateTime转换方法LocalDateTime格式化和解析LocalDateTime 增加或者减少时间的方法修改时间的方法…