【java基础系列】14- Java的内部类与常用类

news2025/1/24 15:26:47

Java的内部类与常用类

1、内部类

1.1 内部类的分类

  • 成员内部类
  • 静态内部类
  • 局部内部类
  • 匿名内部类

1.2 什么是内部类?

  • 概念:在一个类的内部再定义一个完整的类。
  • 特点:
    • 编译之后可生成独立的字节码文件。
    • 内部类可直接访问外部类的私有成员,而不破坏封装。
    • 可为外部类提供必要的内部功能组件。
1.2.1 案例(一)
  • 编译之后可生成独立的字节码文件。
package com.goshawk.chapter_02;

public class Outer {
    class Inner {

    }
}

将上述代码的文件进行编译:javac Outer.java

在这里插入图片描述

12.2 案例(二)
  • 内部类可直接访问外部类的私有成员,而不破坏封装。
package com.goshawk.chapter_02;

public class Outer {
    private String name;

    class Inner {
        public void show(){
            System.out.println(name);
        }
    }
}

1.3 成员内部类

  • 在类的内部定义,与实例变量、实例方法同级别的类。
  • 外部类的一个实例部分,创建内部类对象时,必须依赖外部类对象。
    • Outer out = new Outer();
    • Inner in = out.new Inner();
  • 当外部类、内部类存在重名属性时,会优先访问内部类属性。
  • 成员内部类不能定义静态成员,但是可以定义静态常量。
    • private static String country = "深圳"; ×
    • private static final String country = "深圳";
1.3.1 案例(一)
  • 创建内部对象
package com.goshawk.chapter_02;

/**
 * 外部类
 */
public class Outer {
    // 实例变量
    private String name = "李四";
    private int age = 21;

    // 内部类
    class Inner {
        private String address = "深圳";
        private String phone = "18100000001";
        // 方法
        public void show(){
            // 打印外部类的属性
            System.out.println(name);
            System.out.println(age);
            // 打印内部类的属性
            System.out.println(address);
            System.out.println(phone);
        }
    }
}

package com.goshawk.chapter_02;

import com.goshawk.chapter_02.Outer.Inner;

public class TestOuter {
    public static void main(String[] args) {
        // 创建内部对象的方式一
        // 1、创建外部类对象
        Outer outer = new Outer();
        // 2、创建内部类对象
        Inner inner = outer.new Inner();
        inner.show();
        // 创建内部对象的方式二
        Inner inner1 = new Outer().new Inner();
        inner1.show();
    }
}

1.3.2 案例(二)
  • 当外部类、内部类存在重名属性时,会优先访问内部类属性。
package com.goshawk.chapter_02;

/**
 * 外部类
 */
public class Outer {
    // 实例变量
    private String name = "李四";
    private int age = 21;

    // 内部类
    class Inner {
        private String address = "深圳";
        private String phone = "18100000001";

        private String name = "乌米";
        // 方法
        public void show(){
            // 内部类与外部类的属性名相同,优先打印内部类的属性
            System.out.println(name);
            // 内部类的属性名和外部类的属性名相同时,使用Outer.this来进行访问外部类的属性
            System.out.println(Outer.this.name);

        }
    }
}

1.4 静态内部类

  • 不依赖外部类对象,可直接创建或通过类名访问,可声明静态成员。
  • 只能直接访问外部类的静态成员(实例成员需实例化外部类对象)。
    • Outer.Inner inner = new Outer.Inner();
    • Outer.Inner.show();
  • 注意:只有在内部类才能使用static修饰类,外部类或者普通类是不能使用static关键字进行修饰的。
1.4.1 案例
package com.goshawk.chapter_02.section_2;

/**
 * 外部类
 */
public class Outer {
    // 实例变量
    private String name = "李四";
    private int age = 21;


    /**
     * 静态内部类,和外部类相同
     * 为什么要使用静态内部类?
     * 因为这个静态内部类要被外部类所使用,给外部提供一些功能
     */
    static class Inner {
        private String address = "上海";

        // 静态成员
        private static int count = 1000;

        public void show(){
            // 调用外部类的属性
            // 1、先创建外部类对象
            Outer outer = new Outer();
            // 2、调用外部类对象的属性
            System.out.println(outer.name);
            System.out.println(outer.age);

            // 调用静态内部类的属性
            System.out.println(address);

            // 调用静态内部类的静态属性
            System.out.println(Inner.count);
        }

    }
}

package com.goshawk.chapter_02.section_2;

public class TestOuter {
    public static void main(String[] args) {
        // 直接创建静态内部类对象
        Outer.Inner inner = new Outer.Inner();
        // 调用方法
        inner.show();
    }
}

1.5 局部内部类

  • 定义在外部类方法中,作用范围和创建对象范围仅限于当前方法。
  • 局部内部类访问外部类当前方法中的局部变量时,因无法保障变量的生命周期与自身相同,变量必须修饰为final
  • 限制类的使用范围。
1.5.1 案例
package com.goshawk.chapter_02.section_3;

/**
 * 外部类
 */
public class Outer {
    // 实例变量
    private String name = "李四";
    private int age = 21;

    public void show(){
        // 定义局部变量
        /**
         * 在JDK1.8之后则自动加final
         *
         * 为什么要加final呢?
         * 因为如果不加final的话,在show这个方法执行完毕后,局部变量就会释放(消失),
         *
         * 这里注意此时用于存放创建局部内部类对象的inner变量会消失,
         * 但是show方法内部创建的局部内部类对象(new Inner())不会消失,
         * 以及局部内部类(Inner)也不会消失。
         * 如果此时的局部变量不存在,那么将导致局部内部类找不到这个局部变量address,
         * 所以这里必须要使其变成一个常量,而局部内部类引用的也不再是address,而是深圳。
         *
         * 所以说要想在局部内部类引用局部变量,则必须要使用final对局部变量进行修饰。
         *
         * 还要注意一点就是局部内部类不能声明静态变量,但是可以声明静态常量。
         */
        String address = "深圳";
        // final String address1 = "深圳";

        // 局部内部类:注意不能加任何访问修饰符
        class Inner{
            // 局部内部类的属性
            private String phone = "13100000001";
            private String email = "lisi@163.com";
            private final static int count = 1000;

            public void show2(){
                // 访问外部类的属性
                System.out.println(name);
                System.out.println(Outer.this.age);
                // 访问内部类的属性
                System.out.println(phone);
                System.out.println(this.email);
                // 访问局部变量,(在JDK1.7要求变量必须是常量,也就是要加上final,但是在JDK1.8之后则自动加final)
                System.out.println(address);
            }
        }
        // 创建局部内部类对象
        Inner inner = new Inner();
        inner.show2();
    }
}

package com.goshawk.chapter_02.section_3;

public class TestOuter {
    public static void main(String[] args) {
        Outer outer = new Outer();
        outer.show();
    }
}

1.6 匿名内部类

  • 没有类名的局部内部类(一切特征都与局部内部类相同)。
  • 必须继承一个父类或者实现一个接口。
  • 定义类、实现类、创建对象的语法合并,只能创建一个该类的对象。
  • 优点:减少代码量。
  • 缺点:可读性较差。
1.6.1 案例(一)
  • 实现接口的匿名类
package com.goshawk.chapter_02.section_4;

/**
 * 接口
 */
public interface Usb {
    // 服务
    void service();
}

package com.goshawk.chapter_02.section_4;

/**
 * Mouse类实现了Usb接口
 */
public class Mouse implements Usb{
    @Override
    public void service() {
        System.out.println("连成功连接电脑,鼠标开始工作");
    }
}

package com.goshawk.chapter_02.section_4;

public class TestUsb {
    public static void main(String[] args) {
        // 创建接口类型的变量
        Usb usb = new Mouse();

        usb.service();

        // 局部内部类
        class Fan implements Usb{
            @Override
            public void service() {
                System.out.println("成功连接电脑,风扇开始工作");
            }
        }
        // 使用局部内部类创建对象
        Usb usb1 = new Fan();
        usb1.service();

        // 使用匿名内部类进行优化(相当于创建了一个局部内部类),new Usb() 可以是父类,也可以是接口
        Usb usb2 = new Usb() {
            @Override
            public void service() {
                System.out.println("成功连接电脑,风扇2开始工作");
            }
        };
        usb2.service();
    }

}

1.6.2 案例(二)
  • 匿名子类(继承父类)
package com.goshawk.chapter_02.section_4;

public class Animal {
    private String name;
    // 品种
    private String breed;
    // 年龄
    protected int age;
    // 性别
    public String sex;

    public Animal(){}

    public Animal(String name) {
        this.name = name;
    }

    public void eat(){
        System.out.println("动物吃东西");
    }

    public void sleep(){
        System.out.println("动物睡觉");
    }
}

package com.goshawk.chapter_02.section_4;

public class Bird extends Animal{
    @Override
    public void eat() {
        System.out.println("cuckoo - 杜鹃吃虫子");
    }

    public void show() {
        System.out.println("大家好,我是杜鹃");
    }
}

package com.goshawk.chapter_02.section_4;

public class TestBird {


    public static void main(String[] args) {
        Bird bird = new Bird();
        bird.eat();

        // 匿名子类,继承自Animal类,可以重写父类方法
        Animal bird1 = new Animal(){
            @Override
            public void eat() {
                System.out.println("Magpie - 喜鹊鸟吃虫子");
                super.sleep();
                TestBird testBird = new TestBird();
                System.out.println("访问外部类final修饰的变量:");
            }
        };
        bird1.eat();


    }
}

1.6.3 案例(三)
  • 匿名内部类访问其外部类方法
package com.goshawk.chapter_02.section_4;

public class Dog {
    private final String GOGS = "狗类";

    public void sleep(){
        System.out.println("匿名内部类访问其外部类方法");
        System.out.println("狗在睡觉");
    }

    class Poodle {
        private String name = "小帽";
        private String breed = "贵宾犬";

        public Poodle(String name, String breed){
            this.name = name;
            this.breed = breed;
        }

        public void eat(){
            System.out.println(breed + " " + name + "在吃狗粮");
        }

    }
    Poodle poodle = new Poodle("小华", "哈士奇"){
        @Override
        public void eat() {
            System.out.println("访问外部类final修饰的变量:" + GOGS);
            System.out.println("访问外部类方法");
            sleep();

            super.eat();
        }
    };

    public void print() {
        poodle.eat();
    }

    public static void main(String[] args) {
        Dog dog = new Dog();
        dog.print();
    }

}

  • 注意:使用javac Dog.java进行编译后,匿名内部类编译之后可生成独立的字节码文件。
    • 匿名内部类:Dog$1.class
    • 内部类:Dog$Poodle.class

在这里插入图片描述

1.7 为什么要使用内部类?

  • 每个内部类都能独立的继承一个接口的实现,所以无论外部类是否已经继承了某个(接口的)实现,对于内部类都没有影响。
  • 方便将存在一定逻辑关系的类组织在一起,又可以对外界隐藏。
  • 方便编写事件驱动程序。
  • 方便编写线程代码。

1.8 内部类的作用

  • 实现了更好的封装,普通类(非内部类)的访问修饰符不能为privateprotected,而内部类可以。当我们将内部类声明为private时,只有外部类可以访问内部类,很好地隐藏了内部类。
  • 内部类可以继承(extends)或实现(implements)其他的类或接口,而不受外部类的影响。
  • 内部类可以直接访问外部类的字段和方法,即使是用private修饰的,相反的,外部类不能直接访问内部类的成员。

2、Object

  • 超类。基类,所有类的直接或间接父类,位于继承树的最顶层。
  • 任何类,如没有书写extends显示继承某个类,都默认直接继承Object类,否则为间接继承。
  • Object类中所定义的方法,是所有对象都具备的方法。
  • Object类型可以存储任何对象。
    • 作为参数,可接收任何对象。
    • 作为返回值,可返回任何对象。

3、Object类常用方法

3.1 getClass()方法

  • public final Class<?> getClass(){}
  • 返回引用中存储的实际对象类型。
  • 应用:通常用于判断两个引用中实际存储对象类型是否一致。
3.1.1 案例
package com.goshawk.chapter_02.section_5;

public class Student {
    private String name;
    private int age;

    public Student(){}

    public Student(String name, int age){
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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

package com.goshawk.chapter_02.section_5;

public class TestStudent {
    public static void main(String[] args) {
        Student s1 = new Student("小明", 20);
        Student s2 = new Student("小赵", 21);
        // 判断s1和s2是不是同一个类型
        Class class1 = s1.getClass();
        Class class2 = s2.getClass();
        if (class1==class2){
            System.out.println("s1和s2属于同一个类型");
        }else {
            System.out.println("s1和s2不属于同一个类型");
        }
    }
}

3.2 hashCode()方法

  • public int hashCode(){}
  • 返回该对象的哈希码值。
  • 哈希值根据对象的地址或字符串或数字使用hash算法计算出来的int类型的数值。
  • 一般情况下相同对象返回相同哈希码。
3.2.1 案例
package com.goshawk.chapter_02.section_5;

public class Student {
    private String name;
    private int age;

    public Student(){}

    public Student(String name, int age){
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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


package com.goshawk.chapter_02.section_5;

public class TestStudent {
    public static void main(String[] args) {

        Student s1 = new Student("小明", 20);
        Student s2 = new Student("小赵", 21);

        // hashCode()
        System.out.println(s1.hashCode());
        System.out.println(s2.hashCode());
        Student s3 = s1;
        System.out.println(s3.hashCode()); // s3.hashCode() == s1.hashCode()
    }
}

3.3 toString()方法

  • public String toString(){}
  • 返回该对象的字符串表示(表现形式)。
  • 可以根据程序需求覆盖该方法,如:展示对象各个属性值。
3.3.1 案例
package com.goshawk.chapter_02.section_5;

public class Student {
    private String name;
    private int age;

    public Student(){}

    public Student(String name, int age){
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
		// 重写toString()
    public String toString() {
        return name + ": " + age;
    }
    
    
}

package com.goshawk.chapter_02.section_5;

public class TestStudent {
    public static void main(String[] args) {

        Student s1 = new Student("小明", 20);
        Student s2 = new Student("小赵", 21);

        // toString()
        System.out.println("-----------3、toString()-----------");
        System.out.println(s1.toString());
        System.out.println(s2.toString());
    }
}

3.4 equals()方法

  • public boolean equals(Object obj){}
  • 默认实现为(this == obj),比较两个对象地址是否相同。
  • 可进行覆盖,比较两个对象的内容是否相同。
3.4.1 equals()方法覆盖步骤
  • 比较两个引用是否指向同一个对象。
  • 判断obj是否为null
  • 判断两个引用指向的实际对象类型是否一致。
  • 强制类型转换。
  • 依次比较各个属性值是否相同。
package com.goshawk.chapter_02.section_5;

import java.util.Objects;

public class Student extends Object{
    private String name;
    private int age;

    public Student(){}

    public Student(String name, int age){
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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


    // 重写equals()
    @Override
    public boolean equals(Object obj) {
        // 判断两个对象是否是同一个引用
        if (this==obj){
            return true;
        }
        // 判断obj是否null
        if (obj==null){
            return false;
        }
        // 判断是否是同一个类型
//        if (this.getClass()==obj.getClass()){
//
//        }
        // instanceof判断对象是否是某种类型
        if (obj instanceof Student){
            // 强制转换类型
            Student s = (Student) obj;
            // 比较属性
            if (this.name.equals(s.getName())&&this.age==getAge()){
                return true;
            }
        }
        return false;
    }
}

package com.goshawk.chapter_02.section_5;

public class TestStudent {
    public static void main(String[] args) {
        Student s1 = new Student("小明", 20);
        Student s2 = new Student("小赵", 21);
        System.out.println("-----------1、getClass()-----------");

        // equals(),判断两个对象是否相等
        System.out.println("-----------4、equals()-----------");
        System.out.println(s1.equals(s2));

        Student s4 = new Student("小李", 23);
        Student s5 = new Student("小李", 23);
        System.out.println(s4.equals(s5));
    }
}

3.5 finalize()方法

  • 当对象被判定为垃圾对象时,由JVM自动调用此方法,用以标记垃圾对象,进入回收队列。
  • 垃圾对象:没有有效引用指向此对象时,为垃圾对象。
  • 垃圾回收:由GC销毁垃圾对象,释放数据存储空间。
  • 自动回收机制:JVM的内存耗尽,一次性回收所有垃圾对象。
  • 手动回收机制:使用System.gc();通知JVM执行垃圾回收。
3.5.1 案例
package com.goshawk.chapter_02.section_5;

import java.util.Objects;

public class Student extends Object{
    private String name;
    private int age;

    public Student(){}

    public Student(String name, int age){
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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

    @Override
    protected void finalize() throws Throwable {
        System.out.println(this.name + "对象被回收");
    }
}

package com.goshawk.chapter_02.section_5;

public class TestStudent2 {
    public static void main(String[] args) {
//        Student s1 = new Student("小明", 20);
//        Student s2 = new Student("小赵", 21);
//        Student s3 = new Student("小李", 23);
//        Student s4 = new Student("小甘", 24);
        new Student("小明", 20);
        new Student("小赵", 21);
        new Student("小李", 23);
        new Student("小甘", 24);

        // 回收垃圾
        System.gc();
        System.out.println("回收垃圾");


    }
}

4、包装类

4.1 什么是包装类?

  • 基本数据类型所对应的引用数据类型。
  • Object可统一所有数据,包装类的默认值是null

4.2 包装类对应

基本数据类型包装类型
byteByte
shortShort
intInteger
longLong
floatFloat
doubleDouble
booleanBoolean
charCharacter

4.3 类型转换与装箱、拆箱

4.3.1 什么是装箱?
  • 装箱:将基本数据类型转换为包装类类型(引用类型)

    Integer integer1 = new Integer(num1);
    Integer integer2 = Integer.valueOf(num1);
    
4.3.2 什么是拆箱?
  • 拆箱:将包装类类型转换为基本数据类型

    Integer integer3 = new Integer(100);
    int num2 = integer3.intValue();
    
4.3.3 自动装箱和自动拆箱
package com.goshawk.chapter_02.section_6;

public class Demo {
    public static void main(String[] args) {
        // 类型转换:装箱 --> 基本类型转成引用类型的过程

        // 基本类型
        int num1 = 18;
        // 使用Integer类创建对象

        // JDK1.5之前
        Integer integer1 = new Integer(num1);
        Integer integer2 = Integer.valueOf(num1);
        System.out.println("手动装箱");
        System.out.println(integer1);
        System.out.println(integer2);

        // 类型转换:拆箱 --> 引用类型转成基本类型的过程
        Integer integer3 = new Integer(100);
        int num2 = integer3.intValue();
        System.out.println("手动拆箱");
        System.out.println(num2);

        // JDK1.5之后,提供自动装箱和拆箱
        int age = 30;
        // 自动装箱
        Integer integer4 = age;
        System.out.println("自动装箱");
        System.out.println(integer4);
        // 自动拆箱
        int age2 = integer4;
        System.out.println("自动拆箱");
        System.out.println(age2);
    }
}

  • 使用自动装箱或者自动拆箱后,将编译之后的.class文件进行反编译,会自动转成手动装箱或者手动拆箱的格式。

    在这里插入图片描述

4.3.4 基本类型和字符串转换
  • 8种包装类提供不同类型间的转换方式:
    • Number父类中提供的6个共性方法。
    • parseXXX()静态方法。XXX - > 基本类型
    • valueOf()静态方法。
  • 注意:需保证类型兼容,否则抛出NumberFormatException异常。
package com.goshawk.chapter_02.section_6;

public class Demo {
    public static void main(String[] args) {

        // 基本类型和字符串之间转换
        // 1、基本类型转成字符串
        int n1 = 15;
        // 1.1> 使用 + 号
        String s1 = n1 + "";
        // 1.2> 使用Integer中的toString()方法
        String s2 = Integer.toString(n1);
        // 1.3> 使用使用Integer中的toString()的重载方法
        String s3 = Integer.toString(n1, 16); // radix 表示基数,我这里表示转成16进制

        System.out.println(s1);
        System.out.println(s2);
        System.out.println(s3);

        // 2、字符串转成基本类型

        String str = "150"; // 这里注意不能出现非数字类型
        // 使用Integer.parseXXX();
        int n2 = Integer.parseInt(str);
        System.out.println(n2);

        // boolean字符串转成基本类型 "true" -> true ,非"true" ->false(如果给的字符串不是true,那么将转成false)
        String str2 = "true";
        boolean b1 = Boolean.parseBoolean(str2);
        System.out.println(b1);

    }
}

4.3.5 整数缓冲区
  • Java预先创建256个常用的整数包装类型对象。
  • 在实际应用当中,对已创建的对象进行复用。
package com.goshawk.chapter_02.section_6;

public class Demo1 {
    public static void main(String[] args) {
        Integer integer1 = new Integer(100);
        Integer integer2 = new Integer(100);
        System.out.println(integer1==integer2); // 因为这里比较的是引用类型的地址,所以这里打印出false

        Integer integer3 = 100; // 自动装箱,等同于调用Integer.valueOf方法
        Integer integer4 = 100;
        // Integer integer4 = Integer.valueOf(100);
        System.out.println(integer3 == integer4); // true 因为100在:-128 ~ 127之间内(为什么是这个范围呢?参考Integer.valueOf的源码)

        Integer integer5 = 200; // 自动装箱
        Integer integer6 = 200;
        System.out.println(integer5 == integer6); // false  因为200不在:-128 ~ 127之间内
    }
}

5、String

5.1 String的概述

  • 字符串是常量,创建之后不可改变。
  • 字符串字面值存储在字符串池中,可以共享。(字符串池在方法区中;方法区在JDK1.8之前也称为永久代,JDK1.8之后称为元空间。)
  • String s = "Hello";产生一个对象,字符串池中存储。
  • String s = new String("Hello"); // 产生两个对象,堆、字符串池各存储一个
package com.goshawk.chapter_02.section_6;

public class Demo2 {
    public static void main(String[] args) {
        String name = "hello"; // 常量存储在字符串池中
        System.out.println(name.hashCode());
        name = "lisi"; // 李四赋值给name变量,给字符串赋值时,并没有修改数据,而是重新开辟一个空间。
        System.out.println(name.hashCode());
        String name1 = "lisi";
        System.out.println(name1.hashCode());

        // 字符串的创建方式
        String str = new String("Java");
        String str2 = new String("Java");
        System.out.println(str==str2); // -> false
        System.out.println(str.equals(str2)); // -> true
    }
}

在这里插入图片描述

5.2 String的常用方法

  • public int length():返回字符串的长度。
  • public char charAt(int index):根据下标获取字符。
  • public boolean contains(String str):判断当前字符串中是否包含str
  • public char[] toCharArray():将字符串转换成数组。
  • public int indexOf(String str):查找str首次出现的下标,如果存在返回该下标,否则返回-1
  • public int lastIndexOf(String str):查找字符串在当前字符串中最后一次出现的下标索引。
  • public String trim():去掉字符串前后的空格。
  • public String toUpperCase():将小写转成大写。
  • public String toLowerCase():将大写转成大写。
  • public boolean endWith(String str):判断字符串是否以str结尾。
  • public boolean startWith(String str):判断字符串是否以str开头。
  • public String replace(char oldChar, char newChar):将旧字符串替换成新字符串。
  • public String[] split(String str):根据str做拆分。
5.2.1 案例(一)
package com.goshawk.chapter_02.section_6;

import java.util.Arrays;

public class Demo2 {
    public static void main(String[] args) {

        // 字符串方法的使用
        String content = "java是一种编程语言";

        // length();返回字符串的长度
        // charAt(int index);返回某个位置的字符
        // contains(String str);判断是否包含某个子字符串
        System.out.println(content.length());
        System.out.println(content.charAt(content.length()-1));
        System.out.println(content.contains("java"));

        // toCharArray()返回字符串对应的数组
        // indexOf() 返回字符串首次出现的位置
        // lastIndexOf() 返回字符串最后一次出现的位置
        System.out.println(Arrays.toString(content.toCharArray()));
        System.out.println(content.indexOf("java"));
        System.out.println(content.indexOf("java", 4));
        System.out.println(content.lastIndexOf("java"));

        // trim() 去掉字符串前后的空格
        // toUpperCase() 将小写转成大写
        // toLowerCase() 将大写转成大写
        // endWith() 判断字符串是否以`str`结尾
        // startWith() 判断字符串是否以`str`开头
        String content2 = " hello word ";
        System.out.println(content2.trim());
        System.out.println(content2.toUpperCase());
        System.out.println(content2.toLowerCase());
        String filename = "hello.java";
        System.out.println(filename.endsWith(".java"));
        System.out.println(filename.startsWith("hello"));

        // replace() 用新的字符或字符串 替换 旧的字符或字符串
        // split() 对字符串进行拆分
        System.out.println(content.replace("java", "py"));

        String say = "java,python is the best   programing language";
        String[] arr = say.split("[ ,]+"); // + 表示可能出现多个
        System.out.println(arr.length);
        for (String string : arr){
            System.out.println(string);
        }

        // equals() 、 compareTo() 比较大小
        String s1 = "hello";
        String s2 = "HELLO";
        System.out.println(s1.equalsIgnoreCase(s2)); // 忽略大小写比较

        String s3 = "abc";
        String s4 = "def";
        // 将a的ASCII码值97与d的ASCII码值100 -> 97-100。如果第一个都是a,则开始比较第二个字符,依次类推。
        System.out.println(s3.compareTo(s4));

        String s5 = "abc";
        String s6 = "abcdef";
        // s5的长度为3,s6的长度为6 -> 3-6=-3。
        System.out.println(s5.compareTo(s6));
    }
}

5.2.2 案例(二)
  • 要求
    • 已知String str = "this is a text";
    • str中的单词单独获取出来
    • str中的text替换成practice
    • text前面插入一个easy
    • 将每个单词的首字母改成大写
package com.goshawk.chapter_02.section_6;

public class Demo3 {
    public static void main(String[] args) {
        String str = "this is the text";
        String[] arr = str.split(" ");
        for (String string : arr){
            System.out.println(string);
        }
        
        String str2 = str.replace("text", "practice");
        System.out.println(str2);
        
        String str3 = str.replace("text", "easy text");
        System.out.println(str3);
        
        for (int i=0;i<arr.length;i++){
            char first = arr[i].charAt(0);
            // 把第一个字符转成大写
            char upperfirst = Character.toUpperCase(first);
            
            String newStr = upperfirst+arr[i].substring(1); // 截取
            System.out.println(newStr);
        }
    }
}

5.2.3 可变字符串
  • StringBuffer:可变长字符串,`JDK1.0提供,运行效率慢、线程安全。
  • StringBuilder:可变长字符串,JDK5.0提供,运行效率快,线程不安全。
5.2.3.1 案例(一)
package com.goshawk.chapter_02.section_6;

public class Demo4 {
    public static void main(String[] args) {
        StringBuffer str = new StringBuffer();
        // StringBuilder str = new StringBuilder();
        
        // append追加
        str.append("动物园有大象");
        System.out.println(str.toString());
        str.append("有猴子");
        System.out.println(str.toString());
        // insert添加
        str.insert(0, "先看到长颈鹿,");
        System.out.println(str.toString());
        // replace替换
        str.replace(0, 3, "孔雀");
        System.out.println(str.toString());
        // delete删除
        str.delete(0, 2);
        System.out.println(str.toString());
        // 清空
        str.delete(0, str.length());
        System.out.println(str.length());
    }
}

5.2.3.2 案例(二)
  • 验证StringBuilder效率高于String
package com.goshawk.chapter_02.section_6;

public class Demo5 {
    public static void main(String[] args) {
        // 开始时间
        long startTime = System.currentTimeMillis();
//        String string = "";
//        for (int i=0;i<99999;i++){
//            string+=i;
//        }
//        System.out.println(string);
        
        StringBuilder str = new StringBuilder();
        for (int i=0;i<99999;i++){
            str.append(i);
        }
        System.out.println(str.toString());

        long endTime = System.currentTimeMillis();
        System.out.println("用时:"+(endTime-startTime));
    }
}

6、BigDecimal

6.1 案例(一)

public class TestBigDecimal{
  public static void main(String[] args){
    double d1 = 1.0;
    double d2 = 0.9;
    System.out.println(d1-d2);
    
    double result = (1.4-0.5)/0.9;
    System.out.println(result);
  }
}

输出:

0.09999999999999998
0.9999999999999999
  • 很多实际应用中需要精确运算,而double是近似值存储,不在符合要求,需要借助BigDecimal

6.2 BigDecimal的使用

  • 位置:java.math包中

  • 作用:精确计算浮点数。

  • 创建方式:BigDecimal bd = new BigDecimal("1.0");

  • 除法:divide(BigDecimal bd, int scal, RoundingMode mode)

    • 参数scal:指定精确到小数点后几位。
    • 参数mode
      • 指定小数部分的取舍模式,通常采用四舍五入的模式。
      • 取值为BigDecimal.ROUND_HALF_UP

6.3 案例(二)

package com.goshawk.chapter_02.section_6;

import java.math.BigDecimal;

public class Demo6 {
    public static void main(String[] args) {

        BigDecimal bd1 = new BigDecimal("1.0");
        BigDecimal bd2 = new BigDecimal("0.9");
        // 减法
        BigDecimal result1 = bd1.subtract(bd2);
        System.out.println(result1);
        // 加法
        BigDecimal result2 = bd1.add(bd2);
        System.out.println(result2);
        // 乘法
        BigDecimal result3 = bd1.multiply(bd2);
        System.out.println(result3);
        // 除法
        BigDecimal result4 = new BigDecimal("1.4")
                .subtract(new BigDecimal("0.5"))
                .divide(new BigDecimal("0.9"));
        System.out.println(result4);
        // 除法除不尽
        BigDecimal result5 = new BigDecimal("10")
                // 保留两位小数,且四舍五入
                .divide(new BigDecimal("3"), 2, BigDecimal.ROUND_HALF_UP);

    }

}

7、Date 类

  • Date表示特定的瞬间,精确到毫秒。Date类中的大部分方法都已被Calendar类中的方法所取代(JDK1.1之后)。
  • 时间单位:
    • 1秒=1000毫秒。
    • 1毫秒=1000微秒。
    • 1微秒=1000纳秒。

7.1 案例(一)

package com.goshawk.chapter_02.section_6;

import java.util.Date;

public class Demo7 {
    public static void main(String[] args) {
        // 今天
        Date date1 = new Date();
        System.out.println(date1.toString());
        System.out.println(date1.toLocaleString());
        // 昨天
        Date date2 = new Date(date1.getTime()-(60*60*24*1000));
        System.out.println(date2.toLocaleString());

        boolean b1 = date1.after(date2); // date1在date2之后,返回ture或者false
        System.out.println(b1);

        boolean b2 = date1.before(date2); // date1在date2之前,返回ture或者false
        System.out.println(b2);

        // 比较 compareTo
        int d1 = date1.compareTo(date2);
        System.out.println(d1);
        int d2 = date2.compareTo(date1);
        System.out.println(d2);
        // 比较是否相等equals()
        boolean b3 = date1.equals(date2);
        System.out.println(b3);
    }
}

8、Calendar 类

  • Calendar提供了获取或设置各种日历字段的方法。

  • 构造方法

    • protected Calendar():由于修饰符是protected,所以无法直接创建该对象。
  • 其他方法

    方法名说明
    Static Calendar getlnstance()使用默认时区和区域获取日历
    Void set(int year, int month, int date, int hourofday, int minute, int second)设置日历的年、月、日、时、分、秒。
    int get(int field)返回给定日历字段的值。字段比如年、月、日等。
    Void setTime(Date date)用给定的Date设置此日历的时间。Date-Calendar
    Date getTime()返回一个Date表示此日历的时间。Calendar-Date
    Void add(int field, int amount)按照日历的规则,给指定字段添加或减少时间量
    long getTimeMillies()毫秒为单位返回该日历的时间值。

8.1 案例(一)

package com.goshawk.chapter_02.section_6;

import java.util.Calendar;

public class Demo8 {
    public static void main(String[] args) {
        // 创建Calendar对象
        Calendar calendar = Calendar.getInstance();
        System.out.println(calendar.getTime().toLocaleString());
        System.out.println(calendar.getTimeInMillis());

        // 获取时间信息
        // 获取年
        int year = calendar.get(Calendar.YEAR);
        // 获取月 0-11月,要想正常显示则+1
        int month = calendar.get(Calendar.MONTH);
        // 获取日
        int day = calendar.get(Calendar.DAY_OF_MONTH);
        // 获取小时
        int hour = calendar.get(Calendar.HOUR_OF_DAY); // HOUR 12小时; HOUR_OF_DAY 24小时
        // 获取分钟
        int minute = calendar.get(Calendar.MINUTE);
        // 获取秒
        int second = calendar.get(Calendar.SECOND);
        System.out.println(year+"年"+(month+1)+"月"+day+"日"+hour+":"+minute+":"+second);

        // 修改时间
        Calendar calendar1 = Calendar.getInstance();
        calendar1.set(Calendar.DAY_OF_MONTH, 5);
        System.out.println(calendar1.getTime().toLocaleString());

        // add方法修改时间
        calendar1.add(Calendar.HOUR, 1);
        System.out.println(calendar1.getTime().toLocaleString());
        calendar1.add(Calendar.HOUR, -1);
        System.out.println(calendar1.getTime().toLocaleString());

        int max = calendar1.getActualMaximum(Calendar.DAY_OF_MONTH);
        int min = calendar1.getActualMinimum(Calendar.DAY_OF_MONTH);
        System.out.println(max);
        System.out.println(min);

    }
}

9、SimpleDateFormat 类

  • SimpleDateFormat是一个以与语言环境有关的方式来格式化和解析日期的具体类。

  • 进行格式化(日期 -> 文本)、解析(文本 -> 日期)。

  • 常用的时间模式字母

    字母日期或时间示例
    y2022
    M年中月份08
    d月中天数22
    H1天中小时数(0-23)16
    M分钟13
    s59
    S毫秒367

9.1 案例(一)

package com.goshawk.chapter_02.section_6;

import java.text.SimpleDateFormat;
import java.util.Date;

public class Demo9 {
    public static void main(String[] args) throws Exception{
        // 创建SimpleDateFormat对象
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
        SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy/MM/dd");
        // 创建Date
        Date date = new Date();
        // 格式化date 把日期转换成字符串
        String str = sdf.format(date);
        System.out.println(str);

        // 解析 把字符串转成日期
        Date date1 = sdf1.parse("1990/05/01");
        System.out.println(date1);

    }
}

10、System 类

  • System系统类,主要用于获取系统的属性数据和其他操作,构造方法私有的。
方法名说明
static void arraycopy(…)复制数组
Static long currentTimeMillis();获取当前系统时间,返回的是毫秒值
Static void gc();建议JVM赶快启动垃圾回收器回收垃圾
static void exit(int status);退出jvm,如果参数是0表示正常退出jvm,非0表示异常退出jvm

10.1 案例(一)

package com.goshawk.chapter_02.section_6;

public class Demo10 {
    public static void main(String[] args) {
        // arraycopy 数组复制
        // src 源数组
        // srcPos:从哪个位置开始赋值 0
        // dest:目标数组
        // destPos: 目标数组的位置
        // length: 赋值的长度
        int[] arr = {20, 18, 16, 14, 26, 45, 89, 22};
        int[] dest = new int[8];

        System.arraycopy(arr, 0, dest, 0, arr.length);
        // System.arraycopy(arr, 4, dest, 4, 4);
        for (int i=0;i<dest.length;i++){
            System.out.println(dest[i]);
        }

        // Arrays.copyOf(original, newLength)
        System.out.println(System.currentTimeMillis());

        long start = System.currentTimeMillis();
        for (int i = 0; i<9999999;i++){
            for (int j=0;j<9999999;j++){
                int result=i+j;
            }
        }
        long end = System.currentTimeMillis();
        System.out.println("用时:"+(end-start));

        Student s1 = new Student("aaa", 17); // 对象被变量使用是不回收的
        Student s2 = new Student("bbb", 18);
        Student s3 = new Student("ccc", 19);

        new Student("aaa", 17);
        new Student("bbb", 18);
        new Student("ccc", 19);
        // System.gc() 告诉垃圾回收器回收垃圾
        System.gc();

        // 退出jvm
        System.exit(0);
        System.out.println("结束了"); // 此行代码不打印

    }
}

11、总结

  • 内部类
    • 在一个类的内部再定义一个完整的类。
    • 成员内部类、静态内部类、局部内部类、匿名内部类。
  • Object
    • 所有类的直接或间接父类,可存储任何对象。
  • 包装类
    • 基本数据类型所对应的引用数据类型,可以使Object统一所有数据。
  • String
    • 字符串是常量,创建之后不可改变,字面值保存在字符串池中,可以共享。
  • BigDecimal
    • 可精确计算浮点数。
  • Date
    • 特定时间
  • Calendar
    • 日历
  • System
    • 系统类

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

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

相关文章

【HIT-OSLAB-实验报告】

文章目录前言实验 0 环境的搭建实验原理&材料实验流程建议实验1 输出硬件参数实验内容基础知识实验代码实验结果实验2 实现系统调用实验内容whoami()评分标准基础知识实验代码实验结果实验3 进程运行轨迹的跟踪实验内容基础知识实验代码实验结果结合自己的体会 从程序设计者…

Elasticsearch

一、Spring Data 1、简介 Spring Data 是一个用于简化数据库、非关系型数据库、索引库访问&#xff0c;并支持云服务的开源框架。Spring Data 可以极大的简化JPA的写法&#xff0c;可以在几乎不用写实现的情况下&#xff0c;实现对数据库的访问和操作。除了 CRUD 之外&#xff…

作业-11.29

将txt中的单词转到数据库中 #include <stdio.h> #include <sqlite3.h> #include <stdlib.h> #include <string.h> void do_insert(sqlite3* db, int id, char word[], char jieshi[]); void txt_todatabase(sqlite3* db); int main(int argc, const ch…

DevExpress FMX Data Grid全面编辑和定制

DevExpress FMX Data Grid全面编辑和定制 FMX数据网格(CTP)FireMonkey(FMX)的高性能数据网格组件&#xff0c;具有集成的主细节和数据分组支持。它被优化并构建为与RAD Studio/Delphi/CBuilder一起使用。它支持Windows、Android和macOS平台。 DevExpress FMX数据网格功能强大&a…

redis介绍和理解

官网 介绍: https://www.bilibili.com/video/BV1Fd4y1T7pD/?spm_id_from333.337.search-card.all.click&vd_source4c263677a216945c0d21ca65ee15a5f9 Redis是一个key value的数据库&#xff0c;基于内存、分布式、可选持久性的键值对(Key-Value)存储数据库。 https://ww…

【Java+LeetCode训练】binarySearch源码解析

二分搜索Arrays.binarySearch(int[] a,int key)源码分析【LeetCode】209. 长度最小的子数组解法1&#xff1a;前缀和 暴力解法解法2&#xff1a;前缀和 二分搜索序&#xff1a;使用Arrays工具类中的binarySearch方法进行二分搜索时&#xff0c;我们知道搜索成功会返回其下标&…

数字化餐饮| 刘大厨湘菜馆进杭州,开场及巅峰

盼了几年的刘大厨辣椒炒肉终于来杭州了&#xff0c;但我却没有吃到&#xff0c;小钱对雨科网说&#xff1a;驱车三十里&#xff0c;排队三小时都没吃上&#xff0c;原来他们是每天10点开始放号&#xff0c;11点开餐&#xff0c;去的晚就吃不到。 5月20日&#xff0c;刘大厨在杭…

5G无线技术基础自学系列 | 5G上行物理信道和信号

素材来源&#xff1a;《5G无线网络规划与优化》 一边学习一边整理内容&#xff0c;并与大家分享&#xff0c;侵权即删&#xff0c;谢谢支持&#xff01; 附上汇总贴&#xff1a;5G无线技术基础自学系列 | 汇总_COCOgsta的博客-CSDN博客 5G上行的物理信道包括PRACH、PUCCH、PU…

产品经理要不要考PMP?进化你能力的阶梯!(附:新版考纲及教材)

产品经理和项目经理看起来是毫不相关的两个专业&#xff0c;那么产品经理要不要考PMP呢&#xff1f;其实是非常有必要的。 以前去面试产品经理&#xff0c;HR只会问1个问题&#xff1a;会用axure吗&#xff1f;一开始对产品经理的定义就是设计产品原型的。能设计产品原型&…

【附源码】计算机毕业设计JAVA中小学教务管理平台

【附源码】计算机毕业设计JAVA中小学教务管理平台 目运行 环境项配置&#xff1a; Jdk1.8 Tomcat8.5 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; JAVA …

【北京迅为】RK3568开发板android11系统固件讲解

脚本里面写入这些内容&#x1f446;&#xff0c; apt-get install uuid 后面就是包名&#xff0c;比如说安装了这些内容uuid 在安装之前先执行这个命令增加下载源&#x1f447; 这里会提示&#xff0c;需要输入 回车继续&#xff0c;还是输入 Ctrl-c取消 当然要输入回车继续…

51单片机学习笔记4 新建工程及点亮LED实战

51单片机学习笔记4 新建工程及点亮LED实战一、使用keil新建工程二、项目设置1. 点击魔术棒&#xff0c;钩选Output-Create Hex File2. 设置仿真器三、编写代码1. 尝试编译代码2. 点亮LED的代码3. GPIO引脚介绍4. GPIO内部结构P0端口&#xff1a;P1 端口四、软件仿真一、使用kei…

aws cloudformation 理解自定义资源的使用

资料 AWS::CloudFormation::CustomResourcecfn-response module 自定义资源的逻辑 cloudformation只能对aws service进行部署和配置&#xff0c;但是用户可能需要使用第三方产品&#xff0c;此时需要通过自定义资源将其纳入到cloudformation的管理中。通过编写自定义逻辑&am…

为什么我们提供了新的公共镜像库

众所周知&#xff0c;建木在项目初期就已经完成了“自举”&#xff0c;就是使用建木完成自身的全部CI/CD/CO等自动化流程。 另外&#xff0c;由于建木本身和官方支持的节点都是打包为镜像发布到Docker Hub上&#xff0c;结果最近半年我们频繁碰到如下场景。 场景一 “CI服务的…

flink程序执行管理-1.13

1. 版本说明 本文档内容基于 flink-1.13.x&#xff0c;其他版本的整理&#xff0c;请查看本人博客的 flink 专栏其他文章。 2. 执行配置 StreamExecutionEnvironment 包含 ExecutionConfig 对象&#xff0c;该对象允许程序指定运行时的配置值。改变默认值可以影响所有的任务…

【Nginx 原理】进程模型、HTTP 连接建立和请求处理过程、高性能、高并发、事件处理模型、模块化体系结构

Nginx 原理 Nginx 以其高性能&#xff0c;稳定性&#xff0c;丰富的功能&#xff0c;简单的配置和低资源消耗而闻名。 Nginx进程模型 Nginx 是一个多进程的模型&#xff0c;主要分为一个 Master 进程、多个 Worker 进程。 Master 进程&#xff1a; 管理 Worker 进程。 对外…

TKE 超级节点,Serverless 落地的最佳形态

陈冰心&#xff0c;腾讯云产品经理&#xff0c;负责超级节点迭代与客户拓展&#xff0c;专注于 TKE Serverless 产品演进。 背景 让人又爱又恨的 Serverless Serverless 炙手可热&#xff0c;被称为云原生未来发展的方向。信通院报告显示&#xff1a;在核心业务中使用 Server…

[oeasy]python0022_ python虚拟机_反编译_cpu架构_二进制字节码_汇编语言

程序本质 回忆上次内容 ​python3​​ 的程序是一个 5.3M 的可执行文件 我们通过which命令找到这个python3.8的位置将这个python3.8复制到我们的用户目录下这个文件还是能够执行的 将这个文件转化为字节形态 确实可以转化但是这个文件我们看不懂啊&#xff01;&#xff01;&a…

【应用多元统计分析】上机四五——主成分分析因子分析

目录 一、主成分分析 1.princomp命令 2.screeplot命令 3.【例7.3.3】对【例6.3.3】中的数据从相关矩阵出发进行主成分分析 ​编辑&#xff08;1&#xff09;代码 &#xff08;2&#xff09;碎石图 &#xff08;3&#xff09;散点图 二、因子分析 1.载荷矩阵求解 &…

考CISAW的N个理由!

随着信息科技的飞速发展&#xff0c;互联网的普及&#xff0c;面对信息安全的严峻局势&#xff0c;网络信息安全显得尤为重要&#xff0c;同时近些年来&#xff0c;国家也相继出台一些政策&#xff0c;并推出一些国家认证的资格证书&#xff0c;CISAW认证就是专门针对信息安全保…