多线程,了解-概念-实现方式-常见方法-安全问题-死锁-生产者消费者

news2024/11/26 16:26:27

了解

简单了解多线程

是指从软件或者硬件上实现多个线程并发执行的技术。 具有多线程能力的计算机因有硬件支持而能够在同一时间执行多个线程,提升性能。

简单了解多线程

                                               

 简单了解多线程

                                                  

 简单了解多线程

                                                  

 简单了解多线程

                                                  

概念

 线程相关的概念

                 并行:在同一时刻,有多个指令在多个CPU上同时执行。

                 并发:在同一时刻,有多个指令在单个CPU上交替执行。

并发和并行

                  并行:在同一时刻,有多个指令在多个CPU上同时执行。

 

 并发和并行

                 并发:在同一时刻,有多个指令在单个CPU上交替执行。

  并发和并行

                  并发:在同一时刻,有多个指令在单个CPU上交替执行。

 

 并发和并行

                  并发:在同一时刻,有多个指令在单个CPU上交替执行。

 

进程和线程

      进程:是正在运行的软件        

独立性:进程是一个能独立运行的基本单位,同时也是系统分配资源和调度的独立单位。

动态性:进程的实质是程序的一次执行过程,进程是动态产生,动态消亡的。

并发性:任何进程都可以同其他进程一起并发执行

 进程和线程

 线程:是进程中的单个顺序控制流,是一条执行路径。

                                            

                                           

                                           

                                          

                                           

  

进程和线程

      线程:是进程中的单个顺序控制流,是一条执行路径。  

单线程:一个进程如果只有一条执行路径,则称为单线程程序

多线程:一个进程如果有多条执行路径,则称为多线程程序

 小结

      并发和并行

并行:在同一时刻,有多个指令在多个CPU上同时执行。

并发:在同一时刻,有多个指令在单个CPU上交替执行。

       进程和线程

进程:就是操作系统中正在运行的一个应用程序。

线程:就是应用程序中做的事情。比如:360软件中的杀毒,扫描木马,清理垃圾。

实现方式

多线程的实现方式

多线程的实现方案

继承Thread类的方式进行实现

实现Runnable接口的方式进行实现

利用Callable和Future接口方式实现

package com.itheima.threaddemo1;

public class MyThread extends Thread{
    @Override
    public void run() {
        //代码就是线程在开启之后执行的代码
        for (int i = 0; i < 100; i++) {
            System.out.println("线程开启了" + i);
        }
    }
}

package com.itheima.threaddemo1;

public class Demo {
    public static void main(String[] args) {
        //创建一个线程对象
        MyThread t1 = new MyThread();
        //创建一个线程对象
        MyThread t2 = new MyThread();

        //t1.run();//表示的仅仅是创建对象,用对象去调用方法,并没有开启线程.
        //t2.run();
        //开启一条线程
        t1.start();
        //开启第二条线程
        t2.start();
    }
}
线程开启了90
线程开启了99
线程开启了91
线程开启了92
线程开启了93
线程开启了94
线程开启了95
线程开启了96
线程开启了97
线程开启了98
线程开启了99

Process finished with exit code 0

 

方案1:继承Thread类

定义一个类MyThread继承Thread类

在MyThread类中重写run()方法

创建MyThread类的对象

启动线程(如上)

两个小问题:

为什么要重写run()方法?

         因为run()是用来封装被线程执行的代码

run()方法和start()方法的区别?

         run():封装线程执行的代码,直接调用,相当于普通方法的调用,并没有开启线程。          start():启动线程;然后由JVM调用此线程的run()方法

多线程的实现方式

方式2:实现Runnable接口

定义一个类MyRunnable实现Runnable接口

在MyRunnable类中重写run()方法

创建MyRunnable类的对象

创建Thread类的对象,把MyRunnable对象作为构造方法的参数 启动线程

package com.itheima.threaddemo2;

public class MyRunnable implements Runnable{
    @Override
    public void run() {
        //线程启动后执行的代码
        for (int i = 0; i < 100; i++) {
            System.out.println(Thread.currentThread().getName() + "第二种方式实现多线程" + i);
        }
    }
}

 测试类:

package com.itheima.threaddemo2;

public class Demo {
    public static void main(String[] args) {
        //创建了一个参数的对象
        MyRunnable mr = new MyRunnable();
        //创建了一个线程对象,并把参数传递给这个线程.
        //在线程启动之后,执行的就是参数里面的run方法
        Thread t1 = new Thread(mr);
        //开启线程
        t1.start();


        MyRunnable mr2 = new MyRunnable();
        Thread t2 = new Thread(mr2);
        t2.start();

    }
}
Thread-0第二种方式实现多线程96
Thread-1第二种方式实现多线程96
Thread-0第二种方式实现多线程97
Thread-1第二种方式实现多线程97
Thread-0第二种方式实现多线程98
Thread-1第二种方式实现多线程98
Thread-0第二种方式实现多线程99
Thread-1第二种方式实现多线程99

Process finished with exit code 0

方式3:Callable和Future

定义一个类MyCallable实现Callable接口

在MyCallable类中重写call()方法 创建MyCallable类的对象

创建Future的实现类FutureTask对象,把MyCallable对象作为构造方法的参数

创建Thread类的对象,把FutureTask对象作为构造方法的参数

启动线程

再调用get方法,就可以获取线程结束之后的结果。

package com.itheima.threaddemo3;

import java.util.concurrent.Callable;

public class MyCallable implements Callable<String> {
    @Override
    public String call() throws Exception {
        for (int i = 0; i < 100; i++) {
            System.out.println("跟女孩表白" + i);
        }
        //返回值就表示线程运行完毕之后的结果
        return "答应";
    }
}

 测试类:

package com.itheima.threaddemo3;

import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

public class Demo {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        //线程开启之后需要执行里面的call方法
        MyCallable mc = new MyCallable();

        //Thread t1 = new Thread(mc);

        //可以获取线程执行完毕之后的结果.也可以作为参数传递给Thread对象
        FutureTask<String> ft = new FutureTask<>(mc);

        //创建线程对象
        Thread t1 = new Thread(ft);

        String s = ft.get();
        //开启线程
        t1.start();

        //String s = ft.get();
        System.out.println(s);
    }
}

 正确的代码:

  public static void main(String[] args) throws ExecutionException, InterruptedException {

        //String s = ft.get();
        //开启线程
        t1.start();

        String s = ft.get();
        System.out.println(s);
    }

 三种方式的对比

优点

缺点

实现Runnable、

Callable接口

扩展性强,实现该接口的同时还可以继承其他的类。

编程相对复杂,不能直接使用Thread类中的方法

继承Thread类

编程比较简单,可以直接使用Thread类中的方法

可以扩展性较差,

不能再继承其他的类

常见方法

线程类的常见方法

获取和设置线程名称

获取线程的名字

  • String getName​():返回此线程的名称

Thread类中设置线程的名字

  • void setName​(String name):将此线程的名称更改为等于参数 name
  • 通过构造方法也可以设置线程名称
package com.itheima.threaddemo4;

public class MyThread extends Thread {

    public MyThread() {
    }

    public MyThread(String name) {
        super(name);
    }

    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            System.out.println(getName() + "@@@" + i);
        }
    }
}

 测试类:

package com.itheima.threaddemo4;

public class Demo {
    //1,线程是有默认名字的,格式:Thread-编号
    public static void main(String[] args) {
        MyThread t1 = new MyThread("小蔡");
        MyThread t2 = new MyThread("小强");

        //t1.setName("小蔡");
        //t2.setName("小强");

        t1.start();
        t2.start();
    }
}
小强@@@97
小蔡@@@98
小强@@@98
小蔡@@@99
小强@@@99

Process finished with exit code 0

获得当前线程的对象

  • public static Thread currentThread():返回对当前正在执行的线程对象的引用
package com.itheima.threaddemo5;

public class Demo {
    public static void main(String[] args) {
        String name = Thread.currentThread().getName();
        System.out.println(name);
    }
}

线程休眠

  • public static void sleep(long time):让线程休眠指定的时间,单位为毫秒。
package com.itheima.threaddemo6;

public class MyRunnable implements Runnable {
    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            System.out.println(Thread.currentThread().getName() + "---" + i);
        }
    }
}

测试类:(获取线程,休眠,获取名称)

package com.itheima.threaddemo6;

public class Demo {
    public static void main(String[] args) throws InterruptedException {
        /*System.out.println("睡觉前");
        Thread.sleep(3000);
        System.out.println("睡醒了");*/

        MyRunnable mr = new MyRunnable();

        Thread t1 = new Thread(mr);
        Thread t2 = new Thread(mr);

        t1.start();
        t2.start();
    }
}
Thread-0---97
Thread-1---98
Thread-0---98
Thread-0---99
Thread-1---99

Process finished with exit code 0

线程调度

多线程的并发运行:    

计算机中的CPU,在任意时刻只能执行一条机器指令。每个线程只有获得CPU的使用权才能执行代码。      

 各个线程轮流获得CPU的使用权,分别执行各自的任务。

线程有两种调度模型

  • 分时调度模型:所有线程轮流使用 CPU 的使用权,平均分配每个线程占用 CPU 的时间片
  • 抢占式调度模型:优先让优先级高的线程使用 CPU,如果线程的优先级相同,那么会随机选择一个,优先级高的线程      

        获取的 CPU 时间片相对多一些

 Java使用的是抢占式调度模型

                                CPU

 线程的优先级

        public final void setPriority(int newPriority)    设置线程的优先级

        public final int getPriority()                              获取线程的优先级

package com.itheima.threaddemo7;

import java.util.concurrent.Callable;

public class MyCallable implements Callable<String> {
    @Override
    public String call() throws Exception {
        for (int i = 0; i < 100; i++) {
            System.out.println(Thread.currentThread().getName() + "---" + i);
        }
        return "线程执行完毕了";
    }
}
import java.util.concurrent.FutureTask;

public class Demo {
    public static void main(String[] args) {
        //优先级: 1 - 10 默认值:5
        MyCallable mc = new MyCallable();
        FutureTask<String> ft = new FutureTask<>(mc);
        Thread t1 = new Thread(ft);
        t1.setName("飞机");
        t1.setPriority(10);
        //System.out.println(t1.getPriority());//5
        t1.start();

        
        MyCallable mc2 = new MyCallable();
        FutureTask<String> ft2 = new FutureTask<>(mc2);
        Thread t2 = new Thread(ft2);
        t2.setName("坦克");
        t2.setPriority(1);
        //System.out.println(t2.getPriority());//5
        t2.start();
    }
}

 后台线程/守护线程

public final void setDaemon(boolean on):设置为守护线程

package com.itheima.threaddemo8;

public class MyThread1 extends Thread {
    @Override
    public void run() {
        for (int i = 0; i < 10; i++) {
            System.out.println(getName() + "---" + i);
        }
    }
}
package com.itheima.threaddemo8;

public class MyThread2 extends Thread {
    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            System.out.println(getName() + "---" + i);
        }
    }
}
package com.itheima.threaddemo8;

public class Demo {
    public static void main(String[] args) {
        MyThread1 t1 = new MyThread1();
        MyThread2 t2 = new MyThread2();

        t1.setName("女神");
        t2.setName("备胎");

        //把第二个线程设置为守护线程
        //当普通线程执行完之后,那么守护线程也没有继续运行下去的必要了.
        t2.setDaemon(true);

        t1.start();
        t2.start();
    }
}
女神---8
备胎---8
女神---9
备胎---9
备胎---10
备胎---11
备胎---12
备胎---13
Process finished with exit code 0

 线程生命周期

安全问题

线程的安全问题

 需求:某电影院目前正在上映国产大片,共有100张票,而它有3个窗口卖票,请设计一个程序模拟该电影院卖票

思路:

①定义一个类Ticket实现Runnable接口,里面定义一个成员变量:private int ticketCount = 100;

② 在Ticket类中重写run()方法实现卖票,代码步骤如下

      A:判断票数大于0,就卖票,并告知是哪个窗口卖的

      B:票数要减1

      C:卖光之后,线程停止

③ 定义一个测试类TicketDemo,里面有main方法,代码步骤如下

    A:创建Ticket类的对象

    B:创建三个Thread类的对象,把Ticket对象作为构造方法的参数,并给出对应的窗口名称

    C:启动线程

package com.itheima.threaddemo9;

public class Ticket implements Runnable {
    //票的数量
    private int ticket = 100;
    private Object obj = new Object();

    @Override
    public void run() {
        while(true){
            synchronized (obj){//多个线程必须使用同一把锁.
                if(ticket <= 0){
                    //卖完了
                    break;
                }else{
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    ticket--;
                    System.out.println(Thread.currentThread().getName() + "在卖票,还剩下" + ticket + "张票");
                }
            }
        }
    }
}
package com.itheima.threaddemo9;

public class Demo {
    public static void main(String[] args) {
        /*Ticket ticket1 = new Ticket();
        Ticket ticket2 = new Ticket();
        Ticket ticket3 = new Ticket();

        Thread t1 = new Thread(ticket1);
        Thread t2 = new Thread(ticket2);
        Thread t3 = new Thread(ticket3);*/

        Ticket ticket = new Ticket();

        Thread t1 = new Thread(ticket);
        Thread t2 = new Thread(ticket);
        Thread t3 = new Thread(ticket);

        t1.setName("窗口一");
        t2.setName("窗口二");
        t3.setName("窗口三");

        t1.start();
        t2.start();
        t3.start();
    }
}

 改造:

    @Override
    public void run() {
        while (true) {
            // 当前这个this表示的是Ticket的票对象,我们在测试类中创建了几个票对象?
            // 答: 1个,所以能保证多个线程使用的是同一个锁对象
            synchronized (this){
                if (ticket <= 0) {
                    //卖完了
                    break;
                } else {
                    System.out.println(Thread.currentThread().getName() + "正在卖" + ticket + "号票");
                    ticket--;
                }
            }
        }
    }

卖票案例的思考

刚才讲解了电影院卖票程序,好像没有什么问题。但是在实际生活中,售票时出票也是需要时间的, 所以,在出售一张票的时候,需要一点时间的延迟,接下来我们去修改卖票程序中卖票的动作: 每次出票时间100毫秒,用sleep()方法实现

卖票出现了问题

        相同的票出现了多次

        出现了负数的票

问题原因: 

线程执行的随机性导致的

卖票案例数据安全问题的解决

卖票案例数据安全问题的解决

为什么出现问题?(这也是我们判断多线程程序是否会有数据安全问题的标准)

        多线程操作共享数据

如何解决多线程安全问题呢?

        基本思想:让程序没有安全问题的环境

怎么实现呢?

         把多条语句操作共享数据的代码给锁起来,让任意时刻只能有一个线程执行即可

         Java提供了同步代码块的方式来解决

                                                                            

                                                                  

 

同步代码块

锁多条语句操作共享数据,可以使用[同步代码块]实现 

格式:        

synchronized(任意对象) {              

  多条语句操作共享数据的代码

 } 

使用 Runnable就可以了

 

 默认情况是打开的,只要有一个线程进去执行代码了,锁就会关闭  

 当线程执行完出来了,锁才会自动打开

同步的好处和弊端 

好处:解决了多线程的数据安全问题

弊端:当线程很多时,因为每个线程都会去判断同步上的锁,这是很耗费资源的,无形中会降低程序的运行效率   

同步方法 

同步方法:就是把synchronized关键字加到方法上

格式:

       修饰符 synchronized 返回值类型 方法名(方法参数) {    }

同步代码块和同步方法的区别:

同步代码块可以锁住指定代码,同步方法是锁住方法中所有代码

同步代码块可以指定锁对象,同步方法不能指定锁对象 

同步方法的锁对象是什么呢?

this 

package com.itheima.threaddemo011;
/*
    任务类,将来只会创建一个任务类对象,可以由多个线程,共同执行这个任务对象
 */
public class MyRunnable implements Runnable {
    private static int ticketCount = 100;

    @Override
    public void run() {
        // 希望通过这个run方法,获知当前是哪个线程正在执行任务,从而决定以什么方式保证数据安全;
        while(true){
            if("窗口一".equals(Thread.currentThread().getName())){
                //同步方法
                boolean result = synchronizedMthod();
                if(result){
                    break;
                }
            }

            if("窗口二".equals(Thread.currentThread().getName())){
                //同步代码块
                synchronized (MyRunnable.class){
                    if(ticketCount == 0){
                       break;
                    }else{
                        try {
                            Thread.sleep(10);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        ticketCount--;
                        System.out.println(Thread.currentThread().getName() + "在卖票,还剩下" + ticketCount + "张票");
                    }
                }
            }

        }
    }

    private static synchronized boolean synchronizedMthod() {
        if(ticketCount == 0){
            return true;
        }else{
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            ticketCount--;
            System.out.println(Thread.currentThread().getName() + "在卖票,还剩下" + ticketCount + "张票");
            return false;
        }
    }
}

 

同步方法

同步静态(static)方法:就是把synchronized关键字加到静态方法上

格式:

  修饰符 static synchronized 返回值类型 方法名(方法参数) {    }

同步静态方法的锁对象是什么呢?

类名.class

package com.itheima.day013_Teacher.demo02_MaiPiao;
/*
    这个类当成任务类
 */
public class Piao implements Runnable {

    // 票的总数
     private  int p=100;
    @Override
    public void run() {
       while (true){
           synchronized (this){
               if(p>0){
                   // 卖票的行为
                   System.out.println(Thread.currentThread().getName()+"卖了:"+p-- +"号票");
               }else {
                   System.out.println(Thread.currentThread().getName()+"卖票结束...");
                   break;
               }
           }
       }
    }
    /*@Override
    public void run() {
        while (true){
            // 人为规定,返回true表示可以继续卖票,返回false,表示票卖完了
           boolean r =  maiPiao();
           if(!r){
               break;
           }
        }
    }*/

    private synchronized boolean maiPiao() {
        if(p>0){
            // 卖票的行为
            System.out.println(Thread.currentThread().getName()+"卖了:"+p-- +"号票------------");
            return true;
        }else {
            System.out.println(Thread.currentThread().getName()+"卖票结束...");
            return false;
        }
    }
}

Lock锁

虽然我们可以理解同步代码块和同步方法的锁对象问题,但是我们并没有直接看到在哪里加上了锁,在哪里释放了锁, 为了更清晰的表达如何加锁和释放锁,JDK5以后提供了一个新的锁对象Lock

Lock实现提供比使用synchronized方法和语句可以获得更广泛的锁定操作

Lock中提供了获得锁和释放锁的方法

 void lock():获得锁

void unlock():释放锁

 Lock是接口不能直接实例化,这里采用它的实现类ReentrantLock来实例化

ReentrantLock的构造方法

ReentrantLock​():创建一个ReentrantLock的实例

死锁

死锁

线程死锁是指由于两个或者多个线程互相持有对方所需要的资源,导致这些线程处于等待状态,无法前往执行。

 

 

 

 生产者消费者

生产者消费者模式概述

生产者消费者理想情况 

 

生产者等待 

 等待和唤醒的方法

为了体现生产和消费过程中的等待和唤醒,Java就提供了几个方法供我们使用,这几个方法在Object类中 Object类的等待和唤醒方法:

方法名

说明

void wait​()

导致当前线程等待,直到另一个线程调用该对象的 notify()方法或 notifyAll()方法

void notify​()

唤醒正在等待对象监视器的单个线程

void notifyAll​()

唤醒正在等待对象监视器的所有线程 

 阻塞队列实现等待唤醒机制

阻塞队列继承结构

 

 阻塞队列实现等待唤醒机制

BlockingQueue的核心方法:     

                put(anObject):将参数放入队列,如果放不进去会阻塞。     
                take():取出第一个数据,取不到会阻塞。

常见BlockingQueue:     

          ArrayBlockingQueue:底层是数组,有界。     

          LinkedBlockingQueue:底层是链表,无界。但不是真正的无界,最大为int的最大值。

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

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

相关文章

【Linux进阶】-- 1.python脚本实现守护进程daemon调度,启停等

"怀斯曼不蒸汽啊"1. Linux守护进程介绍1.1 查看进程2. python指令for linux2.1 运行shell命令3. 代码讲解3.1 创建守护进程文件pre_deal.py3.2 用脚本运行3.3 用一个python程序来管理该任务任务介绍: 首先python创建文件1,作用是创建一个守护进程,内容是持续输出时间…

<C++>手撕搜索二叉树

目录 一、搜索二叉树的性质 二、搜索二叉树的结构定义 三、手撕搜索二叉树非递归 1&#xff09;Insert() 2&#xff09;Find() 3&#xff09;Erase() 4&#xff09;InOder() 5&#xff09;BSTree(const BSTree& t) 拷贝构造 6&#xff09;~BSTree()析构函数 四、手…

【学习笔记65】JavaScript的继承

一、认识继承 function Person(name) {this.name name}Person.prototype.sayName () > {console.log(name)}function Stu(age) {this.age age}const s new Stu(18)console.log(s) // {age: 18} 说明&#xff1a; 想让s这个对象内部有一个name属性, 并且可以使用 sayN…

双功能连接试剂:Alkyne hydrazide,炔烃-酰肼 主要特点进行分享

Alkyne hydrazide物理参数&#xff1a; CAS号&#xff1a;N/A |英文名&#xff1a;Alkyne hydrazide | 中文名&#xff1a;炔烃-酰肼 货号&#xff1a;X-CL-1132 分子式&#xff1a;C6H11ClN2O 分子量&#xff1a;162.62 纯度&#xff1a;95% 外形&#xff1a;淡黄色或白…

企业级的商用远程控制软件有哪些

现在远程控制软件&#xff0c;可选性还是比较大的。针对个人用户市场&#xff0c;也有不少免费软件。 企业用户的需求和个人用户的需求&#xff0c;差别较大。个人用户市场&#xff0c;主要诉求是免费、好用。企业用户&#xff0c;主要关注安全、管理功能、性能、价格等因素。…

为什么建议MySQL在2000W条左右记录分表

为什么建议MySQL在2000W条左右记录分表&#xff1f;这个数量问题并不是绝对的&#xff0c;这是一个相对的数量。在回答这个问题前我们应该先了解MySQL底层是用的什么数据结构来进行数据存储的。 MySQL底层采用的B树进行的数据存储 InnoDB存储引擎最小储存单元是页&#xff0c;…

【Java面试】异常常见面试题

文章目录1. Error 和 Exception 区别是什么&#xff1f;运行时异常和一般异常(受检异常)区别是什么&#xff1f;JVM 是如何处理异常的&#xff1f;throw 和 throws 的区别是什么&#xff1f;final、finally、finalize 有什么区别&#xff1f;NoClassDefFoundError 和 ClassNotF…

OpenMP 快速入门

学习《高性能计算&#xff1a;现代系统与应用实践》&#xff08;Thomas Sterling&#xff0c;Matthew Anderson&#xff0c;Maciej Brodowicz&#xff09;第 7 章 OpenMP 的基础 OpenMP OpenMP 是一个 API C、C、Fortran OpenMP 是共享内存的多线程编程模型 共享内存 默认所有…

22年11月-自研-面试题

目录背景题目Activiti回退功能条件分支功能&#xff0c;并行网关、包含网关有没有用到流程流转中&#xff0c;需知会其他人&#xff0c;这些人需同意/做处理&#xff08;有点流程的感觉&#xff09;&#xff0c;最后所有的意见都要汇总。你的实现思路Redis哪些数据结构&#xf…

STM32实战总结:HAL之低功耗

低功耗的含义不必过多解释&#xff0c;一听就能懂。 低功耗对电池供电产品尤其重要。 STM32的有三种低功耗模式&#xff0c;即睡眠模式、停止模式和待机模式。 在我的印象中&#xff0c;停止不就是关机吗&#xff1f;但并不是。 在系统或电源复位以后&#xff0c;微控制器处于运…

基于最小二乘插值(Least-Squares Interpolation)图像超分辨率重构算法研究-附Matlab代码

⭕⭕ 目 录 ⭕⭕✳️ 一、引言✳️ 二、最小二乘图像插值理论与Matlab处理✳️ 三、基于最小二乘插值超分辨重构实验验证✳️ 四、参考文献✳️ 五、Matlab程序获取与验证✳️ 一、引言 图像超分辨率重构技术起源于上世纪60年代Harris和Goodman构造的单帧图像重构方法&#xf…

AutoCAD Electrical 2022—元件的绘制

原理图——图标菜单 选择要插入的元件&#xff1b; 根据实际情况&#xff0c;选择垂直放置还是水平放置&#xff0c;比例大小&#xff1b; 选择一个三极断路器&#xff0c;垂直放置&#xff1b; 点击确定后&#xff0c;点击一根导线&#xff0c;选择断路器另外两个符号是的方向…

相控阵天线(十一):阵列天线有源驻波分析

目录简介有源驻波概念和计算公式平面阵列天线的有源驻波平面阵列有源驻波计算公式平面阵列有源驻波仿真示例不同耦合系数/隔离度的有源驻波分析简介 有源相控阵最大的特点是每一个收发天线后均连接一个独立的T/R组件&#xff0c;每一个T/R组件相当于一个常规雷达的高频前端&am…

【信息检索与数据挖掘期末笔记】(二) IR Evaluation

文章目录测试集无序检索结果集合的评价Precision & RecallAccuarcy?F值有序检索结果评价方法二值相关&#xff08;相关/不相关&#xff09;PrecisionK&#xff08;PK&#xff09;Mean Average Precision&#xff08;MAP&#xff09;Mean Reciprocal Rank多级相关CG&#x…

LeetCode542. 01 矩阵(C++中等题)

题目 给定一个由 0 和 1 组成的矩阵 mat &#xff0c;请输出一个大小相同的矩阵&#xff0c;其中每一个格子是 mat 中对应位置元素到最近的 0 的距离。 两个相邻元素间的距离为 1 。 示例 1&#xff1a; 输入&#xff1a;mat [[0,0,0],[0,1,0],[0,0,0]] 输出&#xff1a;[[…

(三) 共享模型之管程【共享带来的问题】

一、共享带来的问题 1. 临界区 &#xff08;1&#xff09;一个程序运行多个线程本身是没有问题的 &#xff08;2&#xff09;问题出在多个线程访问共享资源 1️⃣多个线程读共享资源其实也没有问题 2️⃣在多个线程对共享资源读写操作时发送指令交错&#xff0c;就会出现问题 …

git@github.com: Permission denied (publickey).

本地虚拟机ubuntu上安装git&#xff0c;想从github上拉取项目到ubuntu上的过程。 1、在ubuntu上安装git 更新apt指令 sudo apt update 安装git sudo apt install git 查看安装git版本 git --version 2、ssh认证 首先已经安装了ssh指令 先执行 ssh -T gitgithub.com 执行之…

3.11 怎么增加小红书评论区的互动?【玩赚小红书】

今天就为大家总结了一下&#xff0c;关于小红书粉丝互动的一些小技巧&#xff0c;来供大家参考。 ​ ​ 一、 固好“真爱粉” 经常会在笔记下面评论、点赞、浏览笔记内容的粉丝&#xff0c;也就是所谓的“真爱粉”、“铁粉”&#xff0c;我们就需要用心维护这一部分粉丝。 ​…

虹科分享|硬件加密U盘|居家办公的网络安全:远程员工可以采取的步骤

新冠肺炎的流行迫使数以百万计的人在家工作&#xff0c;而当时他们对这一概念知之甚少&#xff0c;甚至完全没有经验。虽然许多员工已经重返办公室&#xff0c;但最近的一项研究发现&#xff0c;72%的受访者希望每周至少有两天在家工作&#xff0c;32%的人表示他们希望永久在家…

全波形反演的深度学习方法: 第 4 章 基于正演的 FWI (草稿)

本章论述经典的 FWI, 它基于正演方法. 本贴仅供内部培训. 4.1 FWI 问题 图 4.1 FWI 的输入与输出 [1].图 4.2 FWI 的数学式子.正演问题是建立从速度模型到地震数据的映射. 一般认为是单解的, 即一个速度模型只能生成一个地震数据 (如果不考虑噪声).反演问题是建立从地震数据到…