包装类、多线程的基本使用

news2025/1/24 2:26:25

包装类

1.基本数据类型对应的引用数据类型(包装类)

1.概述:所谓的包装类就是基本类型对应的类(引用类型),我们需要将基本类型转成包装类,从而让基本类型具有类的特性(说白了,就是将基本类型的数据转成包装类,就可以使用包装类中的方法来操作此数据)
    
2.为啥要学包装类:
  a.将来有一些特定操作,调用方法需要传递包装类
    比如:ArrayList集合,add(Object obj),我们只能传递Object的子类对象,此时如果我们想传递一些基本类型的数据,Object作为一个引用类型不能直接接受基本类型,所以我们需要先将基本类型转成包装类传递到Object3.将来我们需要让基本类型和包装类互相转换,为啥:
  a.基本类型为啥转成包装类:
    调用某个方法,方法参数传递Object类型,ArrayList集合,add(Object obj),我们只能传递Object的子类对象,此时如果我们想传递一些基本类型的数据,Object作为一个引用类型不能直接接受基本类型,所以我们需要先将基本类型转成包装类传递到Object中
              
  b.包装类为啥转成基本类型:
    包装类不能直接使用+ - * /做运算,所以需要先将包装类转成基本类型,才能直接使用+ - * /符号
基本类型包装类
byteByte
shortShort
intInteger
longLong
floatFloat
doubleDouble
charCharacter
booleanBoolean

2.Inteaer的介绍以及应用

2.1Integer基本使用

1.概述:Integerint的包装类
2.构造:
  a.Integer(String i):i必须是数字格式
  b.Integer(int i)    
public class Exercise {
    public static void main(String[] args) {
        Integer integer = new Integer("10");
        System.out.println("integer = " + integer);
        Integer integer1 = new Integer(100);
        System.out.println("integer1 = " + integer1);

        Boolean aBoolean = new Boolean(true);
        System.out.println("aBoolean = " + aBoolean);
        Boolean aBoolean1 = new Boolean("true");
        System.out.println("aBoolean1 = " + aBoolean1);
        Boolean aBoolean2 = new Boolean("True");
        System.out.println("aBoolean2 = " + aBoolean2);
    }
}
//输出
integer = 10
integer1 = 100
aBoolean = true
aBoolean1 = true
aBoolean2 = true
Boolean aBoolean = new Boolean(true);

Boolean包装类中 , 底层调用了parseBoolean(String s) , 在parseBoolean中调用了equals.IgnoreCase(s) , 忽略了大小写

public static boolean parseBoolean(String s) {
    return "true".equalsIgnoreCase(s);
}

2.2装箱and拆箱

1.装箱:
  将基本类型变成包装类
      
2.方法:
  static Integer valueOf(int i)  
  static Integer valueOf(String s) -> s需要是数字格式
1.拆箱:将包装类转成基本类型
2.方法:
  int intValue()  
public class Exercise03 {
    public static void main(String[] args) {
        //装箱
        Integer integer = Integer.valueOf(10);
        System.out.println("integer = " + integer);
        Integer integer1 = Integer.valueOf("15");
        System.out.println("integer1 = " + integer1);
        //拆箱
        int value = integer.intValue();
        System.out.println("value = " + value);
        int value1 = integer1.intValue();
        System.out.println("value1 = " + value1);
    }
}

2.3自动拆箱装箱

在操作的过程中,基本类型和包装类之间可以自动转换

public class Demo04Integer {
    public static void main(String[] args) {
        Integer i = 10;
        i+=1;
        System.out.println(i);
    }
}

反编译后如下图:

1694497269393

笔试题:

public class Demo05Integer {
public static void main(String[] args) {
    Integer i1 = 100;
    Integer i2 = 100;
    System.out.println(i1==i2);//true

    Integer i3 = 127;
    Integer i4 = 127;
    System.out.println(i3==i4);//true

    Integer i5 = 128;
    Integer i6 = 128;
    System.out.println(i5==i6);//false
}
}
包装类缓存对象
Byte-128-127
Short-128-127
Integer-128-127
Long-128-127
Float没有
Double没有
Character0-127
Booleantrue和false

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

@IntrinsicCandidate
public static Integer valueOf(int i) {
   if (i >= IntegerCache.low && i <= IntegerCache.high)
       return IntegerCache.cache[i + (-IntegerCache.low)];
   return new Integer(i);
}

可以发现当数值超过了128就会开辟一个新空间 , 即new Integer(i)

此为享元设计模式

3.基本类型与String转换

3.1 基本类型往String转

1.方式1:
  +
2.方式2:String中的静态方法
  static String valueOf(int i)  
public class String_transform {
    public static void main(String[] args) {
        //基本数据类型转包装类
        //方式1: static String valueOf(int a)
        String s = String.valueOf(10);
        System.out.println(s+1);

        //方式2:+""
        int i  = 10;
        String s1 = i+"";
        System.out.println(s1+1);

    }
}

3.2 String转成基本数据类型

每一个包装类中都有一个parseXXX的方法
位置方法说明
Bytestatic byte parseByte(String s)将字符串转成byte
Shortstatic short parseShort(String s)将字符串转成short
Integerstatic int parseInt(String s)将字符串转成int
Longstatic long parseLong(String s)将字符串转成long
Floatstatic float parseFloat(String s)将字符串转成float
Doublestatic double parseDouble(String s)将字符串转成double
Booleanstatic boolean parseBoolean(String s)将字符串转成boolean
public class String_transform {
    public static void main(String[] args) {
        //基本数据类型转包装类
        //方式1: static String valueOf(int a)
        String s = String.valueOf(10);
        System.out.println(s+1);

        //方式2:+""
        int i  = 10;
        String s1 = i+"";
        System.out.println(s1+1);


        //将String转为基本数据类型
        int i1 = Integer.parseInt("101");
        System.out.println("i1+1 = " + i1+1);
        double parseDouble = Double.parseDouble("10.2");
        System.out.println("parseDouble+1 = " + parseDouble + 1);
    }
}

将来开发,我们定义javabean的时候,需要将基本类型的字段(成员变量)定义成包装类类型的

1.将来我们的javabean都是和表对应的
2.针对下面的Student,创建出来的表
id   -> 一般作为表的主键使用,而且主键会自增
name
age

如果主键为自增,我们添加数据时sql语句可以这么写:
insert into student (id,name,age) values (NULL,'柳岩',36);
我们可以用NULL占位(主键自增的列)
    
而我们将来需要将javabean对象中的数据添加到sql语句中,从而保存到数据库中
    
包装类型的属性默认值为NULL,所以到时候我们将javabean对象中的数据放到数据库中时,我们不需要单独为id属性赋值
    
如果不给javabean中的id赋值: Student s = new Student(null,"哈哈",36)
此时我们将javabean中的属性值获取出来放到sql中,正好是: 
insert into student (id,name,age) values (NULL,'哈哈',36);
public class User {
   private Integer uid;
   private String name;
   private Integer age;

   public User(Integer uid, String name, Integer age) {
       this.uid = uid;
       this.name = name;
       this.age = age;
   }

   public User() {
   }

   public Integer getUid() {
       return uid;
   }

   public void setUid(Integer uid) {
       this.uid = uid;
   }

   public String getName() {
       return name;
   }

   public void setName(String name) {
       this.name = name;
   }

   public Integer getAge() {
       return age;
   }

   public void setAge(Integer age) {
       this.age = age;
   }

   @Override
   public String toString() {
       return "User{" +
               "uid=" + uid +
               ", name='" + name + '\'' +
               ", age=" + age +
               '}';
   }
}

多线程

1.多线程_线程和进程

1694511947394

进程(Processor) : 程序的一次执行。由操作系统创建并分配资源,执行一个单独的任务。 进程是系统进行资源分配和调度的独立单位,每个进程都有自己的内存空间和系统资源。进程内所有线程共享堆存储空间,保存程序中定义的对象和常量池。

线程作用 : 负责当前进程中程序的运行 , 一个进程中至少有一个线程 . 一个进程中是可以有多个线程的,这样的应用程序就称之为多线程程序 . 线程是进程内的执行单元,不分配单独的资源,执行一个单独的子任务。线程是进程内调度和分派的基本单位,共享进程资源。每个线程有自己的独立的栈存储空间,保存线程执行的方法以及基本类型的数据。 运行的 Java 程序内含至少一个主线程 main ,用户可以在 Java 程序中自定义并调用多个线程。 JVM 垃圾回收线程也是一个独立的线程。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

并发:同一个时刻多个线程同时操作了同一个数据 , 多个任务交替使用 CPU 核心工作,以提高 CPU 利用率。

并行:同一个时刻多个线程同时执行不同的程序 , 多个CPU核心同时工作,处理不同的任务。

2.CPU调度

1.分时调度:指的是让所有的线程轮流获取CPU使用权,并且平均分配每个线程占用的cpu的时间片
2.抢占式调度:多个线程轮流抢占CPU使用权,哪个线程先抢到,哪个线程先执行,一般都是优先级高的抢到CPU使用权的概率大,我们java程序都是抢占式调度

线程的运行状态 : 线程除创建状态 New 和结束状态 Terminate 外,主要有以下几种运行状态:

NEW(新建) : 线程刚被创建,但是并未启动。还没调用start方法。

运行 : (Running) CPU 正在执行线程。

就绪 : (Runnable) 线程一切就绪,等待 CPU 执行。

运行/就绪状态 统称为可运行状态 Runnable。 Java 程序中,线程在 运行/就绪状态 之间的切换由 JVM 自动调度,开发者无法获知。线程之间的调度采用分优先级多队列时间片轮转算法。进程在执行完 CPU 时间片切换到就绪状态之前会先保存自己的状态,下次进入运行状态时再重新加载。

阻塞 : (Blocked) 线程因缺少其他资源,比如请求资源被上锁而暂停执行。在获得资源后进入就绪状态。 一个线程在等待另一个线程执行一个(唤醒)动作时,该线程进入Waiting状态。进入这个状态后是不能自动唤醒的,必须等待另一个线程调用notify或者notifyAll方法才能够唤醒。

等待 : (Waitting) 线程接受了等待指令,释放资源暂停执行。在超时/接受唤醒指令后进入就绪状态。 同waiting状态,有几个方法有超时参数,调用他们将进入Timed Waiting状态。这一状态将一直保持到超时期满或者接收到唤醒通知。带有超时参数的常用方法有Thread.sleep 、Object.wait。

Terminated(被终止) : 因为run方法正常退出而死亡,或者因为没有捕获的异常终止了run方法而死亡。或者调用过时方法stop()

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

3.主线程介绍

在CPU和内存之间专门为main开辟的通道叫做主线程

主线程专门运行main方法的

1694513606433

创建线程的方式(重点)

1.第一种方式_extends Thread

步骤 :

​ ①定义类 , 继承Thread类(Thread是一个线程类)

​ ②重写Thread中的run方法 , 设置线程任务(自定义的线程需要干什么 , 将业务写进run())

​ ③创建自定义线程类对象

​ ④调用Thread中的start方法(使线程开始执行 ; Java虚拟机会调用线程中的 run 方法)

public class MyThread extends Thread{
    @Override
    public void run() {
        for (int i = 0; i < 3; i++) {
            System.out.println("我是MyThread....");
        }
    }
}
public class Test01 {
    public static void main(String[] args) {
        MyThread myThread = new MyThread();
        myThread.start();	//开启线程 , jvm自动调用重写的run方法

        for (int i = 0; i < 3; i++) {
            System.out.println("我是主线程...");
        }
    }
}

2.多线程在内存中的运行原理

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

当有多个线程开启时 , 进程会为每个线程独自开启新线程的栈空间

注意 : 同一个线程对象 , 不可以连续调用多个start方法

如果需要开启多个线程 , 需要重新new

3.Thread类中的方法

1.void start() 开启线程,jvm会自动执行run方法
2.String getName() 获取线程名字
3.static Thread currentThread()  -> 获取的是当前正在执行的线程对象,此方法在哪个线程中用,获取的就是哪个线程对象 
4.void setName(String name) -> 给线程设置名字
5.static void sleep(long time)->线程睡眠,传递毫秒值,时间超时,线程自动醒来,自动继续执行   
public class MyThread extends Thread{
    @Override
    public void run() {
        for (int i = 0; i < 3; i++) {
            Thread.sleep(1000l);
            System.out.println(getName()+"线程正在执行....");   
            //getName()  获取线程名字   因为继承了Thread,可以直接调用
        }
    }
}
public class Test01 {
    public static void main(String[] args) {
        MyThread myThread = new MyThread();
        myThread.start();		//开启线程 , jvm自动调用重写的run方法
        myThread.setName("haha");	//修改线程名字
        for (int i = 0; i < 3; i++) {
            Thread.sleep(1000l);	//线程睡眠1秒
            System.out.println(Thread.currentThread().getName()+"线程正在执行..."); //static Thread currentThread()   获取的是当前正在执行的线程对象
        }
    }
}
//输出
main线程正在执行...
haha线程正在执行....
main线程正在执行...
haha线程正在执行....
main线程正在执行...
haha线程正在执行....

4.第二种方式_实现Runnable接口

步骤 :

​ ①定义一个类 , 实现Runnable接口

​ ②重写run方法 , 设置线程任务

​ ③创建自定义的线程类对象

​ ④利用Thread中的构造方法 : Thread(Runnable target) , 创建实现类对象 , 放到Thread中

​ ⑤调用start方法 , 启动线程

public class MyThread_implements implements Runnable{
    @Override
    public void run() {
        try {
            Thread.sleep(100L);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName()+"线程正在执行....");
        }
    }
}
public class Test02 {
    public static void main(String[] args) throws InterruptedException {
        MyThread_implements myThread_implements = new MyThread_implements();
        Thread thread = new Thread(myThread_implements);
        thread.start();

        for (int i = 0; i < 3; i++) {
            Thread.sleep(100l);
            System.out.println(Thread.currentThread().getName()+"线程正在执行...");
        }
    }
}

问题 : 为什么再run方法中处理异常只能使用try_catch?

因为Thread中的run方法没有抛异常 , 子类重写父类之后也不抛 , 只能try_catch自己解决

5.两种实现多线程的方式区别

继承extends Thread : 由于继承只能单继承 , 如果某个场景它有父类也需要继承 , 就无法继承Thread了 , 所以有局限性

实现implement Runnable : 接口可以多继承多实现 , 解决了继承的局限性 , 一个类继承一个父类的同 时可以实现一个或多个接口 -----> 推荐使用

6.第三种方式_匿名内部类创建多线程

匿名内部类回顾:一种格式代表子类对象或实现类对象

使用:
	new 接口/抽象父类(){
        重写方法
    }.重写的方法();

	接口/抽象类 对象名 = new 接口/抽象父类(){
        重写方法
    }
	对象名.重写的方法();
public class Test03Niming {
    public static void main(String[] args) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 3; i++) {
                    System.out.println(Thread.currentThread().getName()+"匿名线程正在执行...");
                }
            }
        }).start();

        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName()+"主线程正在执行...");
        }
    }
}

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

线程安全

问 : 什么时候出现线程安全问题 ?

答 : 当多个线程访问同一个资源时 , 可能会出现线程安全问题

1.线程安全问题 —>线程不安全代码

public class MyTicket implements Runnable{
    int ticket = 100;
    @Override
    public void run() {
        while(true){
            try {
                Thread.sleep(100L);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            if (ticket>0){
                System.out.println(Thread.currentThread().getName()+"买了第"+ticket+"张票");
            }
            ticket--;
        }
    }
}

public class Test01 {
    public static void main(String[] args) {
        MyTicket myTicket = new MyTicket();

        Thread thread1 = new Thread(myTicket,"柳岩");
        Thread thread2 = new Thread(myTicket,"涛哥");
        Thread thread3 = new Thread(myTicket,"金莲");

        thread1.start();
        thread2.start();
        thread3.start();
    }
}

原因 : CPU在多个线程中做高速切换

2.解决线程安全问题的第一种方式(使用同步代码块)

格式:
	synchronized(任意对象->锁对象){
        可能出现的线程不安全的代码
    }

执行流程:
	某一个线程抢到锁,进入内部执行,其他线程就抢不到锁了,只能等待,等着执行的线程出了代码块,将锁释放,其他线程才有可能抢到锁对象,去执行
public class Ticket implements Runnable{
    Integer ticket = 50;

    @Override
    public void run() {
        while (true){
            try {
                Thread.sleep(100L);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            synchronized (this){
                if(ticket > 0){
                    System.out.println(Thread.currentThread().getName()+"抢到了第"+ticket+"张票");
                    ticket--;
                }else{
                    return ;
                }
            }
        }
    }
}
public class Test04 {
    public static void main(String[] args) {
        Ticket ticket = new Ticket();
        Thread t1 = new Thread(ticket, "张三");
        Thread t2 = new Thread(ticket, "李四");
        Thread t3 = new Thread(ticket, "王五");

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

3.解决线程安全问题的第二种方式:同步方法

3.1.普通同步方法

1.格式:
  public synchronized 返回值类型 方法名(参数){
      方法体
      return 结果
  }

  public 返回值类型 方法名(参数){
      synchronized(this){
          方法体
          return 结果
      }
  }

2.默认锁:this
public class Ticket1 implements Runnable{
    Integer ticket = 50;
    @Override
    public void run() {
        while (true){
            try {
                Thread.sleep(100l);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            method();
            if(ticket == 0){
                return;
            }
        }
    }
    /*
    //采用synchronized同步方法
    public synchronized void method(){
        if(ticket > 0){
            System.out.println(Thread.currentThread().getName()+"抢到了第"+ticket+"张票");
            ticket--;
        }
    }*/

    public void method(){
        synchronized (this){
            if(ticket > 0){
                System.out.println(Thread.currentThread().getName()+"抢到了第"+ticket+"张票");
                ticket--;
            }
        }
    }
}
public class Test05 {
    public static void main(String[] args) {
        Ticket1 ticket1 = new Ticket1();
        Thread t1 = new Thread(ticket1, "张三");
        Thread t2 = new Thread(ticket1, "李四");
        Thread t3 = new Thread(ticket1, "王五");
        t1.start();
        t2.start();
        t3.start();
    }
}

3.2.静态同步方法

1.格式:
  public static synchronized 返回值类型 方法名(参数){
      方法体
      return 结果
  }

2.默认锁:
  当前类.class
public class Ticket2 implements Runnable{
    static Integer ticket = 50;
    @Override
    public void run() {
        while (true){
            try {
                Thread.sleep(100l);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            method();
            if(ticket == 0){
                return;
            }
        }
    }
    /*
    //采用synchronized同步方法
    public static synchronized void method(){
        if(ticket > 0){
            System.out.println(Thread.currentThread().getName()+"抢到了第"+ticket+"张票");
            ticket--;
        }
    }*/

    public static void method(){
        synchronized (Ticket2.class){
            if(ticket > 0){
                System.out.println(Thread.currentThread().getName()+"抢到了第"+ticket+"张票");
                ticket--;
            }
        }
    }
}
public class Test06 {
    public static void main(String[] args) {
        Ticket2 ticket2 = new Ticket2();
        Thread t1 = new Thread(ticket2, "张三");
        Thread t2 = new Thread(ticket2, "李四");
        Thread t3 = new Thread(ticket2, "王五");

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

死锁

1.死锁介绍(锁嵌套就有可能产生死锁)

​ 指的是两个或者两个以上的线程在执行的过程中,由于竞争同步锁而产生的一种阻塞现象;如果没有外力的作用,他们将无法继续执行下去,这种情况就称之为死锁.

1694519025452
根据上图所示:线程T1正在持有R1,但是T1线程必须再拿到R2,才能继续执行
而线程T2正在持有R2,但是T2线程需要再拿到R1,才能继续执行
此时两个线程处于互相等待的状态,就是死锁,在程序中的死锁将出现在同步代码块的嵌套中

2.死锁的分析

1694519069437

3.代码实现

public class LockA {
    public static LockA lockA = new LockA();
}
public class LockB {
    public static LockB lockB = new LockB();
}
public class DieLock implements Runnable {
    private boolean flag;

    public DieLock(boolean flag) {
        this.flag = flag;
    }

    @Override
    public void run() {
        if (flag) {
            synchronized (LockA.lockA) {
                System.out.println("if...lockA");
                synchronized (LockB.lockB) {
                    System.out.println("if...lockB");
                }
            }
        } else {
            synchronized (LockB.lockB) {
                System.out.println("else...lockB");
                synchronized (LockA.lockA) {
                    System.out.println("else...lockA");
                }
            }
        }

    }
}
public class Test01 {
    public static void main(String[] args) {
        DieLock dieLock1 = new DieLock(true);
        DieLock dieLock2 = new DieLock(false);

        new Thread(dieLock1).start();
        new Thread(dieLock2).start();
    }
}

lag = flag;
}

@Override
public void run() {
    if (flag) {
        synchronized (LockA.lockA) {
            System.out.println("if...lockA");
            synchronized (LockB.lockB) {
                System.out.println("if...lockB");
            }
        }
    } else {
        synchronized (LockB.lockB) {
            System.out.println("else...lockB");
            synchronized (LockA.lockA) {
                System.out.println("else...lockA");
            }
        }
    }

}

}


~~~java
public class Test01 {
    public static void main(String[] args) {
        DieLock dieLock1 = new DieLock(true);
        DieLock dieLock2 = new DieLock(false);

        new Thread(dieLock1).start();
        new Thread(dieLock2).start();
    }
}

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

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

相关文章

springboot上传文件到后台指定文件夹

第一步&#xff0c;在application.yml做一下配置&#xff0c;预设下载目录 files:upload:path: D:/SpringBootItem/springboot/files/ 其中有用到hutool工具依赖&#xff0c;如下在pom.xml中添加依赖&#xff0c;也可以选择不添加&#xff0c;自己修改下Controller中的代码即可…

批量采集头条号文章的工具-让我们更好地辨别信息好坏

大家好&#xff0c;今天我想和大家聊一聊一个在互联网时代备受瞩目的话题&#xff0c;那就是批量采集头条号的文章。作为一个热衷于信息获取和分享的人&#xff0c;我深知这一领域的挑战和机遇。 让我们来谈谈批量采集头条号的文章所面临的挑战。随着互联网的迅猛发展&#xff…

亚马逊API接口解析,实现获得AMAZON商品详情

要解析亚马逊API接口并实现获取亚马逊商品详情&#xff0c;你需要按照以下步骤进行操作&#xff1a; 了解亚马逊开发者中心&#xff1a;访问亚马逊开发者中心&#xff0c;并了解相关的API文档、开发者指南和规定。注册开发者账号&#xff1a;在亚马逊开发者中心上注册一个开发…

Java“牵手”京东商品详情数据,京东商品详情接口,京东API接口申请指南

京东商品详情API是京东平台提供给开发者的应用程序编程接口&#xff0c;通过API可以获取京东平台上商品详情信息。 京东商品详情API可以获取到商品的标题、价格、销量、评价、详情页等信息。开发者在京东开放平台注册开发者账号&#xff0c;并获得访问API接口的密钥后&#xf…

代码随想录算法训练营day46|139.单词拆分|多重背包基础理论| 背包总结

139.单词拆分 力扣题目链接 给定一个非空字符串 s 和一个包含非空单词的列表 wordDict&#xff0c;判定 s 是否可以被空格拆分为一个或多个在字典中出现的单词。 说明&#xff1a; 拆分时可以重复使用字典中的单词。 你可以假设字典中没有重复的单词。 示例 1&#xff1a…

关于批量安装多个apk

for %i in (apks地址/*.apk); do adb install %i https://www.cnblogs.com/lihongtaoya/p/15084378.html adb install -r apks地址/1.apk && adb install -r apks地址/2.apk install-multi-package - 暂时nok https://adbshell.com/commands 最新版本的platform-tool…

Dinky上路之旅

1、部署flink集群 1.1、flink-conf.yaml cat > flink-conf.yaml << EOF jobmanager.rpc.address: boshi-146 jobmanager.rpc.port: 6123 jobmanager.bind-host: 0.0.0.0 jobmanager.memory.process.size: 1600m taskmanager.bind-host: 0.0.0.0 # 修改为本机ip tas…

今日宜分享:科技十足主页面的高校官网颜值排行榜

科技十足主页面的高校官网颜值排行榜 全国985名单&#xff08;最新&#xff09;1. 北京&#xff08;8所&#xff09;2. 上海&#xff08;4所&#xff09;3. 湖南&#xff08;3所&#xff09;4. 陕西&#xff08;3所&#xff09;5. 湖北&#xff08;2所&#xff09;6. 山东&…

postgresql如何关闭自动提交设置

问题&#xff1a;代码运行报错&#xff1a; Caused by: org.postgresql.util.PSQLException: Cannot commit when autoCommit is enabled. 解决方法&#xff1a;修改postgresql配置文件&#xff0c;关闭自动提价设置

机器学习:PCA(Principal Component Analysis主成分)降维

参考&#xff1a;PCA降维原理 操作步骤与优缺点_TranSad的博客-CSDN博客 PCA降维算法_偶尔努力翻身的咸鱼的博客-CSDN博客 需要提前了解的数学知识&#xff1a; 一、PCA的主要思想 PCA&#xff0c;即主成分分析方法&#xff0c;是一种使用最广泛的数据降维算法。PCA的主要思想…

centerOS连不上网解决办法

1、检查路由 route -n如果你缺失第一个路由&#xff0c;是肯定无法ping通外网的。 2、添加dns # vim /etc/resolv.conf nameserver 8.8.8.83、在/etc/resolv.conf文件添加路由 route add default gw 192.168.0.14、重启网卡 service network restart

KT142C-sop16语音芯片的4个IO口如何一对一触发播放_配置文件详细说明

目录 KT142C是一个提供串口的SOP16语音芯片&#xff0c;完美的集成了MP3的硬解码。内置330KByte的空间&#xff0c;最大支持330秒的语音长度&#xff0c;支持多段语音&#xff0c;支持直驱0.5W的扬声器无需外置功放 如上图&#xff0c;芯片有4个IO口可以一对一&#xff0c;详…

服务器数据恢复-热备盘同步过程中硬盘离线的RAID5数据恢复案例

服务器数据恢复环境&#xff1a; 华为OceanStor某型号存储&#xff0c;11块硬盘组建了一组RAID5阵列&#xff0c;另外1块硬盘作为热备盘使用。基于RAID5阵列的LUN分配给linux系统使用&#xff0c;存放Oracle数据库。 服务器故障&#xff1a; RAID5阵列1块硬盘由于未知原因离线…

线程详细解析

本文重点: 目录 什么是线程? 线程共享和非共享资源 线程的优缺点 多线程 线程池 Java创建线程池 什么是线程? 线程是操作系统调度和执行的基本单位,线程和进程一样也有PCB 一个进程必定会有一个线程 在Linux内核中是不会区分进程和线程的,只在用户层面区分 线程共享…

新手询问想要成功学好嵌入式开发有什么建议吗?

今日话题&#xff0c;想要成功学好嵌入式开发有什么建议吗&#xff1f;想要学好的话选择一门合适的编程语言是关键。虽然嵌入式开发支持多种语言&#xff0c;但C和C仍然是最常用的。如果你是初学者&#xff0c;从学习C语言开始是一个不错的选择。它相对容易学习&#xff0c;而且…

基于Python开发的智能停车场车牌识别计费系统(源码+可执行程序+程序配置说明书+程序使用说明书)

一、项目简介 本项目是一套基于Python开发的智能停车场车牌识别计费系统&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Python学习者。 包含&#xff1a;项目源码、项目文档等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试…

教你制作作业查询系统

嗨&#xff0c;各位老师们&#xff0c;今天我要给你们介绍一个超级方便的工具——易查分&#xff01;你知道吗&#xff0c;利用易查分&#xff0c;我们可以轻松制作一个便捷高效的作业查询系统哦&#xff01; 是不是想有个自己的分班or成绩查询页面&#xff1f;博主给老师们争取…

​重生奇迹失落之塔三-四层刷怪攻略​

重生奇迹中的失落之塔一共是八层&#xff0c;每一层均需玩家达到相应的等级后方可进入&#xff0c;等级不到&#xff0c;谢绝入内&#xff0c;甚至有时即便你的等级达标&#xff0c;在里面刷一会怪之后&#xff0c;便自动地退去&#xff0c;因为这里面的怪物绝非等闲之辈&#…

再也不想去阿里巴巴面试了,6年测开面试遭到这样打击.....

前几天我朋友跟我吐苦水&#xff0c;这波面试又把他打击到了&#xff0c;做了快6年软件测试员。。。为了进大厂&#xff0c;也花了很多时间和精力在面试准备上&#xff0c;也刷了很多题。但题刷多了之后有点怀疑人生&#xff0c;不知道刷的这些题在之后的工作中能不能用到&…