Java高级-常用类-String、Date、Compare、Other

news2024/9/29 3:32:56

5cb056d96f727

本篇讲解java常用类

String类

String:字符串,使用一对""引起来表示。

  1. String类被声明为final的,不可被继承。

  2. String实现了Serializable接口:表示字符串是支持序列化的。

    ​ 实现了Comparable接口:表示String可以比较大小

  3. String内部定义了final char[] value用于存储字符串数据

  4. String:代表不可变的字符序列。简称:不可变性。

    体现:

    • 当对字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值。
    • 当对现有的字符串进行连接操作时,同上。
    • 当调用String的replace()方法修改指定字符或字符串时,同上。
  5. 通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。

  6. 字符串常量池中是不会存储相同内容的字符串的。

String的实例化方式:

方式一:通过字面量定义的方式,如下图左

方式二:通过new + 构造器的方式,如下图右

String的实例化方式对比

面试题:String s = new String(“abc”);方式创建对象,在内存中创建了几个对象?

答:两个:一个是堆空间中new结构,另一个是char[]对应的常量池中的数据:“abc”

字符串拼接

  1. 常量与常量的拼接结果在常量池。 且常量池中不会存在相同内容的常量。
  2. 只要其中有一个是变量, 结果就在堆中,相当于new出来的对象对象中有一个value指向常量池的地址
  3. 如果拼接的结果调用intern()方法, 返回值就是常量池的地址
image-20230203155553307

面试题

public class StringTest {

    String str = new String("good");
    char[] ch = { 't', 'e', 's', 't' };

    public void change(String str, char ch[]) {
        str = "test ok";
        ch[0] = 'b';
    }
    public static void main(String[] args) {
        StringTest ex = new StringTest();
        ex.change(ex.str, ex.ch);
        System.out.println(ex.str);//good
        System.out.println(ex.ch);//best
    }
}

首先,初始化str和ch时的内存模型为:

image-20230203162706600

进入change()方法后:

声明了一个形参str,形参传入的是地址,相当于直接指向value。ch则是数组,直接指向数组地址。

str通过字面量赋值修改,由于String值的不可变性,直接在常量池中重新开辟空间赋值。现在,两个str就有区别了,一个是change方法中的str,一个是ex对象中的str属性,它们分别指向不同的地址。

ch也进行了修改,数组可以进行直接修改,所以test被修改为了best。

image-20230203163507479

注意

为了进一步说明字符串的常量与变量间的区别,我们有下面的例子

String s = "hellojava";
final String s1 = "hello";//现在他是一个常量了
String s2 = "hello";
System.out.println(s1 + "java" == s);//true
System.out.println(s2 + "java" == s);//false

s存储在常量池中,s1、s2相同,正常情况下参与运算会被看作是变量,变量地址加常量地址,当然不等于常量的地址了。但现在我们把s1标为常量,再去计算,就是常量间的运算(先将字符串拼接,然后发现常量池中已经有"hellojava"了,就直接使用之前的地址。),所以结果是true。

常用方法

字符串处理:
int length():返回字符串的长度: return value.length
char charAt(int index): 返回某索引处的字符return value[index]
boolean isEmpty():判断是否是空字符串:return value.length == 0
String toLowerCase():使用默认语言环境,将 String 中的所有字符转换为小写
String toUpperCase():使用默认语言环境,将 String 中的所有字符转换为大写
String trim():返回字符串的副本,忽略前导空白和尾部空白
boolean equals(Object obj):比较字符串的内容是否相同
boolean equalsIgnoreCase(String anotherString):与equals方法类似,忽略大小写
String concat(String str):将指定字符串连接到此字符串的结尾。 等价于用“+”
int compareTo(String anotherString):比较两个字符串的大小
String substring(int beginIndex):返回一个新的字符串,它是此字符串的从beginIndex开始截取到最后的一个子字符串。
String substring(int beginIndex, int endIndex) :返回一个新字符串,它是此字符串从beginIndex开始截取到endIndex(不包含)的一个子字符串。

注意,只要涉及到开始索引和结束索引的方法,基本都是左闭右开。顾头不顾腚~

String s = " ab bc ";
System.out.println(s.length());//7
System.out.println(s.charAt(2));//b
System.out.println(s.isEmpty());//false
System.out.println(s.toUpperCase());// AB BC 
System.out.println(s.trim());//ab bc
System.out.println(s.equals("ac"));//false
System.out.println(s.equalsIgnoreCase(" A Bc "));//false
System.out.println(s.concat("def"));// ab bc def
System.out.println(s.compareTo("def"));//-68
System.out.println(s.substring(1));//ab bc

前缀后缀:
boolean endsWith(String suffix):测试此字符串是否以指定的后缀结束
boolean startsWith(String prefix):测试此字符串是否以指定的前缀开始
boolean startsWith(String prefix, int toffset):测试此字符串从指定索引开始的子字符串是否以指定前缀开始

String s = "While the world sleep, you dream.";
System.out.println(s.startsWith("Wh"));//true
System.out.println(s.startsWith("ile", 2));//true
System.out.println(s.endsWith("eam."));//true

查找字符:
boolean contains(CharSequence s):当且仅当此字符串包含指定的 char 值序列时,返回 true
int indexOf(String str):返回指定子字符串在此字符串中第一次出现处的索引
int indexOf(String str, int fromIndex):返回指定子字符串在此字符串中第一次出现处的索引,从指定的索引开始
int lastIndexOf(String str):返回指定子字符串在此字符串中最右边出现处的索引
int lastIndexOf(String str, int fromIndex):返回指定子字符串在此字符串中最后一次出现处的索引,从指定的索引开始反向搜索

注:indexOflastIndexOf方法如果未找到都是返回-1

String s = "While the world sleep, you dream.";
System.out.println(s.contains("sleep"));//true
System.out.println(s.indexOf("the"));//6
System.out.println(s.indexOf("l", 4));//13
System.out.println(s.lastIndexOf("l"));//17
System.out.println(s.lastIndexOf("h", 6));//1

替换
String replace(char oldChar, char newChar):返回一个新的字符串,它是通过用newChar替换此字符串中出现的所有oldChar得到的。
String replace(CharSequence target, CharSequence replacement):使用指定的字面值替换序列替换此字符串所有匹配字面值目标序列的子字符串。
String replaceAll(String regex, String replacement):使用给定的replacement替换此字符串所有匹配给定的正则表达式的子字符串。
String replaceFirst(String regex, String replacement):使用给定的replacement替换此字符串匹配给定的正则表达式的第一个子字符串。

String str = "Tom is Tommy.";
System.out.println(str.replace('T', 'J'));//Jom is Jommy.
System.out.println(str.replace("Tom", "Jim"));//Jim is Jimmy.
str = "12Tom34Tommy5";
//把字符串中的数字替换成,,如果结果中开头和结尾有,的话去掉
String string = str.replaceAll("\\d+", ",").replaceAll("^,|,$", "");
System.out.println(string);//Tom,Tommy

匹配
boolean matches(String regex):告知此字符串是否匹配给定的正则表达式。

String str = "12345";
//判断str字符串中是否全部有数字组成,即有1-n个数字组成
boolean matches = str.matches("\\d+");
System.out.println(matches);//true
String tel = "0571-4534289";
//判断这是否是一个杭州的固定电话
boolean result = tel.matches("0571-\\d{7,8}");
System.out.println(result);//true

切片
String[] split(String regex):根据给定正则表达式的匹配拆分此字符串。
String[] split(String regex, int limit):根据匹配给定的正则表达式来拆分此字符串,最多不超过limit个,如果超过了,剩下的全部都放到最后一个元素中。

str = "Tom|Tommy|Jimmy";
String[] strs = str.split("\\|");
for (int i = 0; i < strs.length; i++) {
    System.out.println(strs[i]);
}
System.out.println();
str = "Tom.Tommy.Jimmy";
String[] strs2 = str.split("\\.");
for (int i = 0; i < strs2.length; i++) {
    System.out.println(strs2[i]);
}
/*
Tom
Tommy
Jimmy

Tom
Tommy
Jimmy
*/

String 与基本数据类型、包装类之间的转换

String --> 基本数据类型、包装类:调用包装类的静态方法:parseXxx(str)

基本数据类型、包装类 --> String:调用String重载的valueOf(xxx)

String s = "123";
int i = Integer.parseInt(s);
System.out.println(i);//123
System.out.println(String.valueOf(i));//123

String 与 char[]之间的转换

String --> char[]:调用String的toCharArray()

char[]–> String:调用String的构造器

String s = "hello";
char[] chars = s.toCharArray();
for (int i = 0; i < chars.length; i++) {
    System.out.println(chars[i]);
}
String s1 = new String(chars);
System.out.println(s1);

String 与 byte[]之间的转换

编码:String --> byte[]:调用String的getBytes()

​ 字符串 -->字节 (看得懂 —>看不懂的二进制数据)

解码:byte[] --> String:调用String的构造器

​ 编码的逆过程,字节 --> 字符串 (看不懂的二进制数据 —> 看得懂)

说明:解码时,要求解码使用的字符集必须与编码时使用的字符集一致,否则会出现乱码。

String s = "abc123爱你";
byte[] bytes = s.getBytes();
System.out.println(Arrays.toString(bytes));//默认使用UTF-8编码
//[97, 98, 99, 49, 50, 51, -25, -120, -79, -28, -67, -96]
String s1 = new String(bytes);//解码,默认使用UTF-8
System.out.println(s1);
//abc123爱你

byte[] gbks = s.getBytes("gbk");//使用gbk字符编码
System.out.println(Arrays.toString(gbks));
//[97, 98, 99, 49, 50, 51, -80, -82, -60, -29]
String s2 = new String(gbks);//解码,默认使用UTF-8
System.out.println(s2);
//abc123����
String gbk = new String(gbks, "gbk");//解码,使用GBK
System.out.println(gbk);
//abc123爱你

练习题

模拟一个trim方法,去除字符串两端的空格。

public String trim(String str){
    if (str.isEmpty()){
        return null;
    }
    //先把String转为Char数组
    char[] chars = str.toCharArray();

    //遍历数组,返回开头和末尾字符的索引
    int foreIndex = 0, rearIndex = 0;
    boolean flag = false;

    //查找首位字符索引
    for (int i = 0; i < chars.length; i++) {
        if (chars[i] == ' ') {
            System.out.println("出现空格!现在记录空格索引:" + i);
            flag = true;//出现过空格了
        } else {
            System.out.println("出现了非空格内容。");
            if (flag == true) {
                foreIndex = i;
                break;//检测到出现非空格内容前有空格,记录该位置,并退出遍历
            }
        }
    }
    System.out.println("--------------");
    //查找末位字符索引
    for (int i = chars.length - 1;i >= 0; i--) {
        if (chars[i] == ' ') {
            System.out.println("出现空格!现在记录空格索引:" + i);
            flag = true;//出现过空格了
        } else {
            System.out.println("出现了非空格内容。");
            if (flag == true) {
                rearIndex = i;
                break;
            }
        }
    }
    //如果首末位本身就有字符,直接返回
    if (chars[0] != ' '){
        foreIndex = 0;
    }
    if (chars[chars.length - 1] == ' '){
        rearIndex = chars.length - 2;
    }
    System.out.println("字符首位索引是:" + foreIndex);
    System.out.println("字符末位索引是:" + rearIndex);
    //取出新的字符串        
    return str.substring(foreIndex, ++rearIndex);
}

源码对比分析:

Returns a string whose value is this string, with any leading and trailing whitespace removed.

If this String object represents an empty character sequence, or the first and last characters of character sequence represented by this String object both have codes greater than ‘\u0020’ (the space character), then a reference to this String object is returned.

Otherwise, if there is no character with a code greater than ‘\u0020’ in the string, then a String object representing an empty string is returned.

Otherwise, let k be the index of the first character in the string whose code is greater than ‘\u0020’, and let m be the index of the last character in the string whose code is greater than ‘\u0020’. A String object is returned, representing the substring of this string that begins with the character at index k and ends with the character at index m-that is, the result of this.substring(k, m + 1).

This method may be used to trim whitespace (as defined above) from the beginning and end of a string.

Returns:

A string whose value is this string, with any leading and trailing white space removed, or this string if it has no leading or trailing white space.

External and inferred annotations:

@org.jetbrains.annotations.NotNull @org.jetbrains.annotations.Contract(pure = true)

public String trim() {
    int len = value.length;
    int st = 0;//记录字串首位索引
    char[] val = value;    /* avoid getfield opcode 避免获取字段操作码 */

    while ((st < len) && (val[st] <= ' ')) {
        st++;
    }
    while ((st < len) && (val[len - 1] <= ' ')) {//此时st < len表示判断到子串开头即可
        len--;
    }
    return ((st > 0) || (len < value.length)) ? substring(st, len) : this;
}

来看第一个循环,st < len的条件表示首位索引小于字符串长度,保证循环整个字符串。val[st] <= ' '的条件表示字符的编码小于空格(空格编码为32),如果两个条件都满足说明出现空格等字符,不满足则要么是开头没有出现空格,要么是整个字符串都是空格。

第二个循环是从后向前遍历,同时查看是否为空格等字符,两个条件都满足,说明出现空格,不满足要么末尾没有出现空格,要么是整个字符串都是空格。

最后(st > 0) || (len < value.length)条件通过(st和len发生了改变)表示:

  • 前后任意位置发现了空格 --> substring(st, len)取子串
  • 字符串全为空格 --> substring(st, len)取子串,会返回空的字符串

将一个字符串进行反转。将字符串中指定部分进行反转。

比如“abcdefg”反转为”abfedcg”

/**
* 将一个字符串进行反转。将字符串中指定部分进行反转。比如“abcdefg”反转为”abfedcg”
* @param startIndex 反转字符串开头索引
* @param endIndex  反转字符串末尾索引
* @return 反转后的数组
*/
public String reverse(String str, int startIndex,int endIndex){
    char[] chars = str.toCharArray();
    //反转数组
    char[] reverse = new char[chars.length];
    reverse = Arrays.copyOf(chars, chars.length);//先把之前的数组复制一份
    for (int i = startIndex,j = endIndex; i < endIndex + 1; i++,j--) {
        reverse[i] = chars[j];
    }
    return new String(reverse);
}
//改进后
public String reverse(String str, int startIndex,int endIndex){
    char[] chars = str.toCharArray();
    //反转数组
    for (int i = startIndex,j = endIndex; i < j; i++,j--) {//i < j是该算法的精髓
        char temp = chars[j];
        chars[j] = chars[i];
        chars[i] = temp;
    }
    return new String(chars);
}
//方式三:使用String的拼接
public String reverse(String str, int startIndex,int endIndex){
    if (str == null){
        return null;
    }
    //第一部分
    String s = str.substring(0, startIndex);
    //第二部分
    for (int i = endIndex;i >= startIndex;i--){
        s += str.charAt(i);
    }
    //第三部分
    s += str.substring(endIndex+1);
    return s;
}
//方式四:使用StringBuilder的拼接
public String reverse(String str, int startIndex,int endIndex){
    if (str == null){
        return null;
    }
    //第一部分
    StringBuilder s = new StringBuilder(str.substring(0, startIndex));
    //第二部分
    for (int i = endIndex;i >= startIndex;i--){
        s.append(str.charAt(i));
    }
    //第三部分
    s.append(str.substring(endIndex+1));
    return s.toString();
}

获取一个字符串在另一个字符串中出现的次数。

比如:获取“ ab”在 “abkkcadkabkebfkabkskab” 中出现的次数

/**
* 获取一个字符串在另一个字符串中出现的次数。
*       比如:获取“ ab”在 “abkkcadkabkebfkabkskab” 中出现的次数
* @param str 要受检测的字符串
* @param checkStr 要检测的子串
* @return 如果出现,返回次数,如果没出现返回0
*/
public int getCount(String str,String checkStr){
    int count = 0;
    char[] chars = str.toCharArray();
    char[] checkChars = checkStr.toCharArray();
    outter:for (int i = 0; i < chars.length; i++) {
        if (chars[i] != checkChars[0]){
            continue;
        }
        for (int j = 0,k = i; j < checkChars.length; j++,k++) {
            if (chars[k] != checkChars[j]){
                continue outter;
            }
        }
        count++;
    }
    return count;
}
//方式二:利用indexOf()检查字符串
public int getCount(String str,String checkStr){
    if (str.length() < checkStr.length()){
        return 0;
    }
    int count = 0,index = 0;
    //循环字符串直到找不到为止
    while ((index = str.indexOf(checkStr,index)) != -1){
        count++;
        index += checkStr.length();//每次开始寻找的位置==上次找到的字符出现位置+字符长度
    }
    return count;
}

获取两个字符串中最大相同子串。

比如:str1 = "abcwerthelloyuiodef“;``str2 = "cvhellobnm"
提示:将短的那个串进行长度依次递减的子串与较长的串比较。
前提:两个字符串中只有一个最大相同字串。

public String getMaxSubString(String str1,String str2){
    String longerStr = (str1.length() > str2.length())? str1:str2;
    String shortStr = (str1.length() <= str2.length())? str1:str2;
    int length = shortStr.length();
    for (int i = 0; i < length; i++) {
        for (int j = 0,k = length - i; k <= length; j++,k++) {//k <= length k现在是索引数,要能取到才行
            String subString = shortStr.substring(j,k);
            if (longerStr.contains(subString)){
                return subString;
            }
        }
    }
    return null;
}

对字符串中字符进行自然顺序排序。

提示:

  1. 字符串变成字符数组。

  2. 对数组排序,选择,冒泡,Arrays.sort();

  3. 将排序后的数组变成字符串。

public String sort(String str){
    char[] chars = str.toCharArray();
    Arrays.sort(chars);//方式一:系统快排
    //方式二:手写冒泡
    boolean is_swap = false;
    for (int i = 0; i < chars.length - 1; i++) {
        for (int j = 0; j < chars.length - 1; j++) {
            if (chars[j] > chars[j+1]){
                char temp = chars[j];
                chars[j] = chars[j+1];
                chars[j+1] = temp;
                is_swap = true;
            }
            if (is_swap){
                is_swap = false;
                break;
            }
        }
    }
    return new String(chars);
}

StringBuffer、StringBuilder类

与String比较

String、StringBuffer、StringBuilder三者的异同?

  • String:不可变的字符序列;底层使用char[]存储
  • StringBuffer:可变的字符序列;线程安全的,效率低;底层使用char[]存储
  • StringBuilder:可变的字符序列;jdk5.0新增的,线程不安全的,效率高;底层使用char[]存储

源码分析:

String str = new String();//char[] value = new char[0];
String str1 = new String("abc");//char[] value = new char[]{'a','b','c'};

StringBuffer sb1 = new StringBuffer();//char[] value = new char[16];底层创建了一个长度是16的数组。
System.out.println(sb1.length());//0,显示的是实际写入的字符长度。
sb1.append('a');//value[0] = 'a';
sb1.append('b');//value[1] = 'b';

StringBuffer sb2 = new StringBuffer("abc");//char[] value = new char["abc".length() + 16];

问题1. System.out.println(sb2.length());输出结果为?

答案:3。只写入了3个字符。而底层char[]数组的长度为19。

问题2. 扩容问题:如果要添加的数据底层数组盛不下了,那就需要扩容底层的数组。如何进行扩容

答案:默认情况下,扩容为原来容量的2倍 + 2,同时将原有数组中的元素复制到新的数组中。

指导意义:

​ 开发中建议大家使用:StringBuffer(int capacity)StringBuilder(int capacity),因为它们可以改变长度随时修改数据,再根据是否需要解决线程安全,选其一(需要线程安全选StringBuffer(int capacity))。

其常用方法

有以下方法:
StringBuffer append(xxx):提供了很多的append()方法,用于进行字符串拼接
StringBuffer delete(int start,int end):删除指定位置的内容
StringBuffer replace(int start, int end, String str):把[start,end)位置替换为str
StringBuffer insert(int offset, xxx):在指定位置插入xxx
StringBuffer reverse() :把当前字符序列逆转
public int indexOf(String str)
public String substring(int start,int end):返回一个从start开始到end索引结束的左闭右开区间的子字符串
public int length()
public char charAt(int n )
public void setCharAt(int n ,char ch)

总结:
增:append(xxx)
删:delete(int start,int end)
改:setCharAt(int n ,char ch) / replace(int start, int end, String str)
查:charAt(int n)
插:insert(int offset, xxx)
长度:length();
⭐遍历:for() + charAt() / toString()

StringBuilder str = new StringBuilder("0123456");
str.append(7);
System.out.println(str);//01234567
str.delete(6,8);
System.out.println(str);//012345
str.setCharAt(5,'a');
System.out.println(str);//01234a
str.replace(4,6,"45");
System.out.println(str);//012345
str.insert(3,"hello");
System.out.println(str);//012hello345
System.out.println(str.length());//11
System.out.println(str.toString());//012hello345

Date类

System类时间

使用System类的currentTimeMillis()方法获取当前时间

long time = System.currentTimeMillis();
System.out.println(time);//1675588908461

JDK 8之前日期和时间的API

java.util.Date类及其子类

java.util.Date
|—java.sql.Date

  1. 两个构造器的使用
    • 构造器一:Date():创建一个对应当前时间的Date对象
    • 构造器二:创建指定毫秒数的Date对象
Date date = new Date();
Date date1 = new Date(1675563865916L);//java.util.Date有两个
  1. 两个方法的使用
    • toString():显示当前的年、月、日、时、分、秒
    • getTime():获取当前Date对象对应的毫秒数。(时间戳)
System.out.println(date.getTime());//1675564133278
System.out.println(date.toString());//Sun Feb 05 10:28:53 CST 2023
  1. java.sql.Date对应着数据库中的日期类型的变量

    • 如何实例化

      java.sql.Date date2 = new java.sql.Date(1675563865916L);
      //java.sql.Date只有这一个构造器可用,必须填上时间戳
      System.out.println(date2.toString());//2023-02-05	java.sql.Date只会输出年月日
      System.out.println(date2.getTime());//1675563865916
      
    • 如何将java.util.Date对象转换为java.sql.Date对象

      通过时间戳

      Date date1 = new Date(1675563865916L);
      java.sql.Date date3 = new java.sql.Date(date1.getTime());
      System.out.println(date3);
      

SimpleDateFormat

Date类的API不易于国际化,大部分被废弃了,java.text.SimpleDateFormat类是一个不与语言环境有关的方式来格式化和解析日期的具体类。

它允许进行

  • 格式化:日期–>文本

    SimpleDateFormat():默认的模式和语言环境创建对象

    public SimpleDateFormat(String pattern):该构造方法可以用参数pattern指定的格式创建一个对象,该对象调用–>public String format(Date date)方法格式化时间对象date

  • 解析:文本–>日期

    public Date parse(String source):从给定字符串的开始解析文本,以生成一个日期。

//使用默认构造器
SimpleDateFormat sdf = new SimpleDateFormat();
String format = sdf.format(new Date());
System.out.println(format);//2/6/23 12:11 PM

//解析
String s = "2/6/23 11:25 AM";
Date date = sdf.parse(s);
System.out.println(date);//Mon Feb 06 11:25:00 CST 2023

//按照指定的方法格式化和解析,调用带参的构造器
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy年MM月dd日 HH时mm分ss秒");
//格式化
String format1 = dateFormat.format(new Date());
System.out.println(format1);//2023年02月06日 12时19分47秒
//解析,解析时必须使用与SimpleDateFormat相同的日期格式
Date parse1 = dateFormat.parse("2023年02月06日 12时18分56秒");
System.out.println(parse1);//Mon Feb 06 12:18:56 CST 2023

日期格式的字母含义:

image-20230206150552876

Calendar

Calendar是一个抽象基类,主用用于完成日期字段之间相互操作的功能。

获取Calendar实例的方法

  • 使用Calendar.getInstance()方法

    Calendar calendar = Calendar.getInstance();即可实例化

  • 调用它的子类GregorianCalendar的构造器。(不常用)

一个Calendar的实例是系统时间的抽象表示,通过get(int field)方法来取得想要的时间信息。比如YEARMONTHDAY_OF_WEEKHOUR_OF_DAYMINUTESECOND

  • public void get(int field,int value)

    int days = calendar.get(Calendar.DAY_OF_MONTH);
    System.out.println(days);//6
    
  • public void set(int field,int value)

    calendar.set(Calendar.DAY_OF_MONTH,22);
    int day = calendar.get(Calendar.DAY_OF_MONTH);
    System.out.println(day);//22
    
  • public void add(int field,int amount)

    calendar.add(Calendar.DAY_OF_MONTH,-1);
    System.out.println(calendar.get(Calendar.DAY_OF_MONTH));//21
    
  • 与Date转换

    • public final Date getTime()

      Date time = calendar.getTime();
      System.out.println(time);//21
      
    • public final void setTime(Date date)

      calendar.setTime(new Date());
      System.out.println(calendar.get(Calendar.DAY_OF_MONTH));//6
      

注意:

  • 获取月份时:一月是0,二月是1,以此类推,12月是11
  • 获取星期时:周日是1,周二是2 。。。。。周六是7

JDK 8中新的日期和时间API

为啥要引入新的时间api?来看看旧版api有多变态

用我们之前的Date类生成2023年2月6日的日期

Date date = new Date(2023, 2, 6);//你以为是这样?
System.out.println(date);//Tue Mar 06 00:00:00 CST 3923

错了,这样给出的是3923年3月6日,只有日是对的。这是因为Date中的年份是从1900开始的,而月份都从0开始。所以要想得到正确的日期,必须要使用偏移量。像这样:

Date date = new Date(2023 - 1900, 2 - 1, 6);
System.out.println(date);//Mon Feb 06 00:00:00 CST 2023

太变态了,不过幸好有外部的包可以使用,它叫做:Joda-Time,java8之前我们通过maven引入即可,java8之后官方引入了这个包,我们可以直接使用。

java.time

java新的日期包有:

java.time – 包含值对象的基础包
java.time.chrono – 提供对不同的日历系统的访问
java.time.format – 格式化和解析时间和日期
java.time.temporal – 包括底层框架和扩展特性
java.time.zone – 包含时区支持的类

LocalDate、LocalTime、LocalDateTime 类是其中较重要的几个类,它们的实例是不可变的对象,分别表示使用 ISO-8601日历系统的日期、时间、日期和时间。它们提供了简单的本地日期或时间,并不包含当前的时间信息,也不包含与时区相关的信息。

  • LocalDate代表IOS格式(yyyy-MM-dd)的日期,可以存储 生日、纪念日等日期。
  • LocalTime表示一个时间,而不是日期。
  • LocalDateTime是用来表示日期和时间的,这是一个最常用的类之一。

这三个类拥有相同的方法,不同的是只包含日期或时间而已。LocalDateTime最常用。

方法列举:

方法描述
now() / now(ZoneId zone)静态方法,根据当前时间创建对象/指定时区的对象
of()静态方法,根据指定日期/时间创建对象
getDayOfMonth()/getDayOfYear()获得月份天数(1-31) /获得年份天数(1-366)
getDayOfWeek()获得星期几(返回一个 DayOfWeek 枚举值)
getMonth()获得月份, 返回一个 Month 枚举值
getMonthValue() / getYear()获得月份(1-12) /获得年份
getHour()/getMinute()/getSecond()获得当前对象对应的小时、分钟、秒
withDayOfMonth()/withDayOfYear()/
withMonth()/withYear()
将月份天数、年份天数、月份、年份
修改为指定的值并返回新的对象
plusDays()/plusWeeks()
plusMonths()/plusYears()/plusHours()
向当前对象添加几天、几周、
几个月、几年、几小时
minusMonths() / minusWeeks()/
minusDays()/minusYears()/minusHours()
从当前对象减去几月、几周、几天、
几年、几小时
//now()方法获取当前时间
LocalDate localDate = LocalDate.now();
System.out.println(localDate);//2023-02-06
LocalTime localTime = LocalTime.now();
System.out.println(localTime);//22:35:26.178
LocalDateTime localDateTime = LocalDateTime.now();
System.out.println(localDateTime);//2023-02-06T22:35:26.178

//of()
LocalDateTime dateTime = LocalDateTime.of(2023, 2, 6, 22, 53, 23);
System.out.println(dateTime);//2023-02-06T22:53:23

//getXxx()
LocalDateTime now = LocalDateTime.now();
System.out.println(now.getDayOfMonth());//6
System.out.println(now.getDayOfWeek());//MONDAY
System.out.println(now.getMonth());//FEBRUARY
System.out.println(now.getMinute());//2
System.out.println(now.getHour());//23

//withXxx(),修改时间。由于不可变性,所以返回一个日期对象。
LocalDateTime now = LocalDateTime.now();
System.out.println(now.withHour(3));//修改时间为3点:2023-02-06T03:13:50.929
System.out.println(now.withDayOfMonth(22));//修改日期为22号:2023-02-22T23:13:50.929
System.out.println(now.withDayOfYear(234));//修改日期为一年中的第234天:2023-08-22T23:13:50.929

//plusXxx(),加上一段时间。由于不可变性,所以返回一个日期对象。
System.out.println(now.plusMonths(2));//2023-04-06T23:33:21.147
System.out.println(now.plusDays(3));//2023-02-09T23:33:21.147

//minusXxx(),减去一段时间。由于不可变性,所以返回一个日期对象。
System.out.println(now.minusDays(2));//2023-02-04T23:34:58.746
System.out.println(now.minusHours(3));//2023-02-06T20:34:58.746

瞬时:Instant

Instant:时间线上的一个瞬时点。 这可能被用来记录应用程序中的事件时间戳。

java.time包通过值类型Instant提供机器视图,不提供处理人类意义上的时间单位。Instant表示时间线上的一点,而不需要任何上下文信息,例如,时区。概念上讲,它只是简单的表示自1970年1月1日0时0分0秒(UTC)开始的秒数。因为java.time包是基于纳秒计算的,所以Instant的精度可以达到纳秒级。

方法描述
now()静态方法,返回默认UTC时区的Instant类的对象
ofEpochMilli(long epochMilli)静态方法,返回在1970-01-01 00:00:00基础上
加上指定毫秒数之后的Instant类的对象
atOffset(ZoneOffset offset)结合即时的偏移来创建一个 OffsetDateTime
toEpochMilli()返回1970-01-01 00:00:00到当前时间的毫秒数,即为时间戳

使用示例:

Instant instant = Instant.now();
System.out.println(instant);//2023-02-06T15:58:40.700Z
//默认提供的是本初子午线时间,北京时间加上8小时即可
OffsetDateTime beijingTime = instant.atOffset(ZoneOffset.ofHours(8));//atOffset()方法调整时差
System.out.println(beijingTime);//2023-02-06T23:58:40.700+08:00

long milli = instant.toEpochMilli();//获取时间戳(自1970年1月1日0时0分0秒(UTC)开始的秒数)
System.out.println(milli);//1675699348497

Instant instant = Instant.ofEpochMilli(1675700191827L);//以某一时刻shi'li
System.out.println(instant.atOffset(ZoneOffset.ofHours(8)));
//2023-02-07T00:16:31.827+08:00	为了更好展示添加了时区

DateTimeFormatter

java.time.format.DateTimeFormatter类:该类提供了三种格式化方法:

  • 预定义的标准格式。(不常用)

    如:ISO_LOCAL_DATE_TIME;ISO_LOCAL_DATE;ISO_LOCAL_TIME

    DateTimeFormatter formatter = DateTimeFormatter.ISO_LOCAL_DATE_TIME;
    
    //为了显示出来,我们进行格式化
    LocalDateTime now = LocalDateTime.now();
    String format = formatter.format(now);
    System.out.println(format);//2023-02-07T10:36:22.118
    

    此时生成的是一个格式,将实现java.time.temporal.TemporalAccessor接口的时间对象放进它的format()方法就可以实例化一个标准格式的时间。

  • 本地化相关的格式。如:ofLocalizedDateTime(FormatStyle.LONG)

    DateTimeFormatter formatter = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.LONG).withZone(ZoneId.systemDefault());
    LocalDateTime now = LocalDateTime.now();
    String formated_time = formatter.format(now);
    System.out.println(formated_time);//February 7, 2023 11:01:28 AM CT
    

    注意:使用FormatStyle.LONG格式时可能会碰到unchecked异常,加上时区即可。

    根据社区中的回答,这大概率是LONG格式在输出时需要时区信息,而formatter中默认没有这个信息,导致了错误。

  • 自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)(最常用)

    DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss");
    String formated_time = formatter.format(LocalDateTime.now());
    System.out.println(formated_time);//2023-02-07 11:10:24
    

方法列表:

方法描述
ofPattern(String pattern)静 态 方 法 , 返 回 一 个 指 定
字 符 串 格 式 的DateTimeFormatter
format(TemporalAccessor t)格式化一个日期、时间,返回字符串
parse(CharSequence text)将指定格式的字符序列解析为一个日期、时间

其它API

  • ZoneId:该类中包含了所有的时区信息,一个时区的ID,如 Europe/Paris
  • ZonedDateTime:一个在ISO-8601日历系统时区的日期时间,如 2007-12-03T10:15:30+01:00 Europe/Paris。
  • 其中每个时区都对应着ID,地区ID都为“{区域}/{城市}”的格式,例如:Asia/Shanghai等
  • Clock:使用时区提供对当前即时、日期和时间的访问的时钟。
  • 持续时间:Duration,用于计算两个“时间”间隔
  • 日期间隔:Period,用于计算两个“日期”间隔
  • TemporalAdjuster : 时间校正器。有时我们可能需要获取例如:将日期调整到“下一个工作日”等操作。
  • TemporalAdjusters : 该类通过静态方法firstDayOfXxx()/lastDayOfXxx()/nextXxx()提供了大量的常用TemporalAdjuster的实现。

参考:与传统日期处理的转换

To 遗留类From 遗留类
java.time.Instantjava.util.DateDate.from(instant)Date.from(instant)
java.time.Instant java.sql.TimestampTimestamp.from(instant)timestamp.toInstant()
java.time.ZonedDateTimejava.util.GregorianCalendarGregorianCalendar.from(zonedDateTime)cal.toZonedDateTime()
java.time.LocalDate/LocalTimejava.sql.TimeDate.valueOf(localDate)date.toLocalDate()
java.time.LocalDateTimejava.sql.TimestampTimestamp.valueOf(localDateTime)timestamp.toLocalDateTime()
java.time.ZoneIdjava.util.TimeZoneTimezone.getTimeZone(id)timeZone.toZoneId()
java.time.format.DateTimeFormatterjava.text.DateFormatformatter.toFormat()

比较类

在Java中经常会涉及到对象数组的排序问题,那么就涉及到对象之间的比较问题。

Java实现对象排序的方式有两种:

  • 自然排序:java.lang.Comparable
  • 定制排序:java.util.Comparator

Comparable接口-自然排序

对于String类、包装类

像String、包装类等实现了Comparable接口,重写了compareTo(obj)方法,给出了比较两个对象大小的方式。重写compareTo(obj)方法以后,进行了从小到大的排列。我们直接使用即可

重写compareTo(obj)的规则:

如果当前对象this大于形参对象obj,则返回正整数

如果当前对象this小于形参对象obj,则返回负整数

如果当前对象this等于形参对象obj,则返回零。

String[] arr = new String[]{"AA","CC","KK","MM","GG","JJ","DD"};
Arrays.sort(arr);//内部调用了重写的compareTo(obj)方法
System.out.println(Arrays.toString(arr));//[AA, CC, DD, GG, JJ, KK, MM]

对于自定义类

对于自定义类来说,如果需要排序,我们可以让自定义类实现Comparable接口,重写compareTo(obj)方法。在compareTo(obj)方法中指明如何排序。

public class Person implements Comparable{
    private String name;
    private int age;

    /*
    省略构造器,String等方法
    */

    @Override
    public int compareTo(Object o) {
        if (!(o instanceof Person)){
            throw new RuntimeException("传入的类型不是Person!");
        }
        Person person = (Person) o;
        if (this.age > person.age){
            return this.age - person.age;
        }else if (this.age < person.age){
            return this.age - person.age;
        }else {
            return this.name.compareTo(person.name);//年龄相同比较姓名
        }
    }
}

接下来就可以进行比较

Person[] people = new Person[5];
{
    people[0] = new Person("恒星", 22);
    people[1] = new Person("芽衣", 18);
    people[2] = new Person("琪亚娜", 18);
    people[3] = new Person("布洛尼亚", 16);
    people[4] = new Person("无量塔 姬子", 22);
}
Arrays.sort(people);
for (int i = 0; i < people.length; i++) {
    System.out.println(people[i]);
}
/*
Person{name='布洛尼亚', age=16}
Person{name='琪亚娜', age=18}
Person{name='芽衣', age=18}
Person{name='恒星', age=22}
Person{name='无量塔 姬子', age=22}
*/

Comparator接口-定制排序

当元素的类型没有实现java.lang.Comparable接口而又不方便修改代码,或者实现了java.lang.Comparable接口的排序规则不适合当前的操作,那么可以考虑使用Comparator的对象来排序。

String[] arr = new String[]{"AA","CC","KK","MM","GG","JJ","DD"};
Arrays.sort(arr, new Comparator() {//自己定制一个排序方法,这个方法可能只是临时用一下

    @Override
    public int compare(Object o1, Object o2) {
        if (!(o1 instanceof String && o2 instanceof String)){
            throw new RuntimeException("传入的参数不是String类型。");
        }
        String s1 = (String) o1;
        String s2 = (String) o2;
        return -(s1.compareTo(s2));//从高到低排列
    }
});
System.out.println(Arrays.toString(arr));

对于自定义对象

Person[] people = new Person[5];
{
    people[0] = new Person("恒星", 22);
    people[1] = new Person("芽衣", 18);
    people[2] = new Person("琪亚娜", 18);
    people[3] = new Person("布洛尼亚", 16);
    people[4] = new Person("无量塔 姬子", 22);
}
//之前的排序不符合我们的要求,我们自己重新定制一个
Arrays.sort(people, new Comparator() {

    @Override//先按照姓名升序排列,再按照年龄升序排列
    public int compare(Object o1, Object o2) {
        if (!(o1 instanceof Person && o2 instanceof Person)){
            throw new RuntimeException("传入的类型不是Person!");
        }
        Person person1 = (Person) o1;
        Person person2 = (Person) o2;
        //方式一:
        if (person1.getName().compareTo(person2.getName()) > 0){
            return 1;
        }else if(person1.getName().compareTo(person2.getName()) < 0){
            return -1;
        }else {
            return person2.getAge() - person1.getAge();
        }
        //方式二:这样似乎更好?
        if (person1.getName().equals(person2.getName())){
            return person2.getAge() - person1.getAge();
        }else {
            return person1.getName().compareTo(person2.getName());
        }
    }
});
System.out.println(Arrays.toString(people));
/*[Person{name='布洛尼亚', age=16},
Person{name='恒星', age=22},
Person{name='无量塔 姬子', age=22},
Person{name='琪亚娜', age=18},
Person{name='芽衣', age=18}]*/

Comparable接口与Comparator的使用的对比

Comparable接口的方式一旦确定,保证Comparable接口实现类的对象在任何位置都可以比较大小。Comparator接口属于临时性的比较。

其他类

System类

System类代表系统,系统级的很多属性和控制方法都放置在该类的内部。该类位于java.lang包。

  • 由于该类的构造器是private的,所以无法创建该类的对象,也就是无法实
    例化该类。其内部的成员变量和成员方法都是static的,所以也可以很方便
    的进行调用。

  • 成员变量

    • System类内部包含in、out和err三个成员变量,分别代表标准输入流
      (键盘输入),标准输出流(显示器)和标准错误输出流(显示器)。
  • 成员方法

    • native long currentTimeMillis()
      该方法的作用是返回当前的计算机时间,时间的表达格式为当前计算机时
      间和GMT时间(格林威治时间)1970年1月1号0时0分0秒所差的毫秒数。

    • void exit(int status)
      该方法的作用是退出程序。其中status的值为0代表正常退出,非零代表
      异常退出。使用该方法可以在图形界面编程中实现程序的退出功能等。

    • void gc()
      该方法的作用是请求系统进行垃圾回收。至于系统是否立刻回收,则
      取决于系统中垃圾回收算法的实现以及系统执行时的情况。

    • String getProperty(String key)
      该方法的作用是获得系统中属性名为key的属性对应的值。系统中常见
      的属性名以及属性的作用如下表所示:

      image-20230207154150702

String javaVersion = System.getProperty("java.version");
System.out.println("java的version:" + javaVersion);//java的version:1.8.0_181

String javaHome = System.getProperty("java.home");
System.out.println("java的home:" + javaHome);//java的home:C:\Program Files\Java\jdk1.8.0_181\jre

String osName = System.getProperty("os.name");
System.out.println("os的name:" + osName);//os的name:Windows 10

String osVersion = System.getProperty("os.version");
System.out.println("os的version:" + osVersion);//os的version:10.0

String userName = System.getProperty("user.name");
System.out.println("user的name:" + userName);//user的name:HengxingStu

String userHome = System.getProperty("user.home");
System.out.println("user的home:" + userHome);//user的home:C:\Users\HengxingStu

String userDir = System.getProperty("user.dir");
System.out.println("user的dir:" + userDir);//user的dir:C:\Users\HengxingStu\Documents\CodeSpace\JavaSenior\day04

Math类

java.lang.Math提供了一系列静态方法用于科学计算。其方法的参数和返回值类型一般为double型。

abs绝对值
acos,asin,atan,cos,sin,tan三角函数
sqrt平方根
pow(double a,doble b)a的b次幂
log自然对数
expe为底指数
max(double a,double b)
min(double a,double b)
random()
返回0.0到1.0的随机数
long round(double a)double型数据a转换为long型(四舍五入)
toDegrees(double angrad)弧度—>角度
toRadians(double angdeg)角度—>弧度

BigInteger类 和 BigDecimal类

Integer类作为int的包装类,能存储的最大整型值为(2^31) -1,Long类也是有限的,最大为(2^63) -1。如果要表示再大的整数,不管是基本数据类型还是他们的包装类都无能为力,更不用说进行运算了。

  • java.math包的BigInteger可以表示不可变的任意精度的整数。BigInteger 提供
    所有 Java 的基本整数操作符的对应物,并提供java.lang.Math的所有相关方法。
    另外,BigInteger 还提供以下运算:模算术、GCD 计算、质数测试、素数生成、
    位操作以及一些其他操作。

  • 构造器

    • BigInteger(String val):根据字符串构建BigInteger对象
  • 常用方法

    • public BigInteger abs():返回此BigInteger的绝对值的BigInteger。
    • BigInteger add(BigInteger val) :返回其值为 (this + val) 的 BigInteger
    • BigInteger subtract(BigInteger val):返回其值为 (this - val) 的 BigInteger
    • BigInteger multiply(BigInteger val):返回其值为 (this * val) 的 BigInteger
    • BigInteger divide(BigInteger val):返回其值为 (this / val) 的 BigInteger。整数相除只保留整数部分。
    • BigInteger remainder(BigInteger val):返回其值为 (this % val) 的 BigInteger。
    • BigInteger[] divideAndRemainder(BigInteger val):返回包含 (this / val) 后跟(this % val) 的两个 BigInteger 的数组。
    • BigInteger pow(int exponent):返回其值为 (this exponent ) 的 BigInteger。

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

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

相关文章

微搭低代码从入门到精通09-数据容器

我们已经用了两篇的篇幅介绍了微搭的布局组件&#xff0c;包括普通容器、文本、图片、轮播容器。 微搭中还有粗粒度的组件&#xff0c;今天介绍的数据容器就是粗粒度的组件。所谓粗粒度的组件&#xff0c;一般包括基础组件、样式还有默认的事件。数据容器一共包含三种分别是数…

vscode sftp从linux服务器下载文件至本地:No such file or dictionary【已解决】

在服务器跑完程序需要下载数据的时候报错&#xff1a; [warn] ENOENT: no such file or directory, open /home/LIST_2080Ti/.ssh/config load /home/LIST_2080Ti/.ssh/config failed 完整报错内容如下&#xff1a; [02-10 08:38:47] [info] config at /home/LIST_2080Ti {&q…

Arm-Linux子系统的互相Notify

前言&#xff1a; Linux下面不同的子系统一个个的组成了整个系统的运行环节&#xff0c;为了让这些子系统能够互相通讯&#xff0c;有一种叫做&#xff1a;notify chain(通知链)的东西。本篇看下。 概括 所谓通知链&#xff0c;有通知&#xff0c;就有执行的地方。比如A子系统通…

无题

&#xff08;1&#xff09;风国产化替代&#xff1f;全球化&#xff1f;新一代数字化技术升级&#xff1f;云化&#xff08;公有云化&#xff09;&#xff1f;业务线上化&#xff1f;产业互联整合&#xff1f;私有云原生技术可以支撑&#xff1a;国产化替代-新一代数字化技术升…

WPS底层逻辑串讲

文章目录wps页面基本介绍演示文稿功能讲解框架介绍具体功能讲解&#xff1a;1. 另存为2. 输出图片3. 文件打包4. 演示文稿打印5. 文档加密两种方式 ❤文件打开密码 &#xff1a;文档加密--->密码加密--》输入密码即可 ❤文档编辑密码&#xff1a;输出为PDF--->设置即可6.…

【老卫拆书】009期:Vue+Node肩挑全栈!《Node.js+Express+MongoDB+Vue.js全栈开发实战》开箱

今天刚拿到一本新书&#xff0c;叫做《Node.jsExpressMongoDBVue.js全栈开发实战》&#xff0c;做个开箱。 外观 先从外观上讲&#xff0c;这本是全新的未开封的&#xff0c;膜还在。 这本书介绍从技术原理到整合开发实战&#xff0c;以丰富的项目展现全栈开发的一个技巧。 …

ChatGPT如何注册,如何使用(个人版)文末送账号

您好&#xff0c;我是码农飞哥&#xff08;wei158556&#xff09;&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f4aa;&#x1f3fb; 1. Python基础专栏&#xff0c;基础知识一网打尽&#xff0c;9.9元买不了吃亏&#xff0c;买不了上当。 Python从入门到精…

ES8——Generator函数的使用

babel工具插件下载&#xff1a;npm i --save babel-polyfill 引入&#xff1a;polyfill.js进行转码&#xff08;es8->es5&#xff09; 介绍 Generator函数用于生成迭代器 function * (){} yeild: 作用同return类似 {const obj function* () {yield "a";yield 12…

分享! opensource.builders——为您喜爱的应用程序查找开源替代方案

今天和大家分享一个非常有意思的网站, 我们可以从这个网站中找到一些常用软件的开源替代方案, 比如说之前很火Notion, 我们可以找到它的开源替代方案——Appflowy, 并且在 github 上学习部署它的本地版本.好啦, 话不多说, 上地址:网址地址: https://opensource.builders/github…

数据库(三):多版本并发控制MVCC,行锁的衍生版本,记录锁,间隙锁, Next-Key锁(邻键锁)

文章目录前言一、MVCC以及MVCC的缺点1.1 MVCC可以为数据库解决什么问题1.2 MVCC的基本思想1.3 版本号1.4 Undo日志1.5 ReadView1.6 快照读和当前读1.6.1 快照读1.6.2 当前读二、记录锁三、间隙锁四、邻键锁总结前言 一、MVCC以及MVCC的缺点 MVCC&#xff0c;即多版本并发控制…

Linux clock子系统及驱动实例

文章目录基本概念CLK子系统时钟API的使用clock驱动实例1、时钟树2、设备树3、驱动实现fixed_clk固定时钟实现factor_clk分频时钟实现gate_clk门控时钟实现基本概念 晶振&#xff1a;晶源振荡器 PLL&#xff1a;Phase lock loop&#xff0c;锁相环。用于提升频率 OSC&#xff1a…

数据结构与算法这么难,为什么我们还要学习?

文章目录前言1. 数据结构与算法是什么&#xff1f;2. 为什么数据结构与算法很难&#xff1f;3. 如何系统学习数据结构与算法&#xff1f;&#x1f351; 复杂度&#x1f351; 线性表&#x1f351; 树形结构&#x1f351; 图&#x1f351; 排序&#x1f351; 字符串&#x1f351;…

Django框架之站点管理

站点管理 站点: 分为内容发布和公共访问两部分内容发布的部分由网站的管理员负责查看、添加、修改、删除数据Django能够根据定义的模型类自动地生成管理模块使用Django的管理模块, 需要按照如下步骤操作 : 1.管理界面本地化2.创建管理员3.注册模型类4.发布内容到数据库 1.管理…

备战蓝桥杯第一天【二分查找无bug版】

&#x1f339;作者:云小逸 &#x1f4dd;个人主页:云小逸的主页 &#x1f4dd;Github:云小逸的Github &#x1f91f;motto:要敢于一个人默默的面对自己&#xff0c;强大自己才是核心。不要等到什么都没有了&#xff0c;才下定决心去做。种一颗树&#xff0c;最好的时间是十年前…

基于 python获取教育领域新闻进行分词关键词词共现分析 Gephi 软件绘制主题知识图谱

本文着眼于对疫情期间教育领域新闻的分析&#xff0c;基于 python 语言&#xff0c;利用爬虫获取教育领域的最新新闻&#xff0c;并将其内容进行分词&#xff0c;抓取关键词。在此基础上&#xff0c;根据关键词进行共现分析&#xff0c;并利用 Gephi 软件绘制主题知识图谱&…

超详细讲解长度受限制的字符串函数(保姆级教程!!!)

超详细讲解长度受限制的字符串函数&#xff08;保姆级教程&#xff01;&#xff01;&#xff01;&#xff09;长度受限制的字符串函数strncpy函数strncpy函数的使用strncpy函数的模拟实现strncat函数strncat函数的使用strncat函数的模拟实现strncmp函数strncmp函数的使用strncm…

【golang分布式Job调度服务】

需求背景 目前各服务里经常会有定时任务相关需求&#xff0c;而定时任务通常要求同时只有一个任务执行&#xff0c;为了保证定时任务高可以通常也需要主备部署&#xff0c;导致开发定义任务时需要考虑锁竞争关系&#xff0c;以及考虑任务执行状态&#xff08;成功、失败、重试…

Lambda表达式了解到使用(清晰明朗)

这里写目录标题lambda简述以前的实现初次使用lambda表达式Lambda表达式的语法测试方法lambda表达式简写示范测试方法运行结果方法引用测试方法运行结果构造方法引用定义一个Boy类编写测试方法运行结果实际应用测试代码运行结果内置函数式接口FunctionalInterface注解Functional…

自从学会了Python,我实现了壁纸自由(6)

小朋友们好&#xff0c;大朋友们好&#xff01;我是猫妹&#xff01;哈哈哈&#xff0c;又到周末啦&#xff01;这周过得怎么样&#xff1f;马上就要开学了&#xff0c;寒假作业早已写好了吧&#xff1f;开学让人兴奋&#xff0c;上了很久网课都要吐啦&#xff01;开学也让人有…