String是一个final类,代表不可变的字符序列
其内部是使用private final char[] value实现的
String:字符串,使用一对""引起来表示
1.String声明为final的,不可被继承
2.String实现了Serializable接口:表示字符串是支持序列化的
实现了Comparable接口:表示String可以比较大小
3.String内部定义了final char[] value用于存储字符串数据
4.String:代表不可变的字符序列。简称:不可变性
体现:1.当对字符串重新赋值时,需要重写指定内存区域赋值,不能使用原有的value进行赋值
2.当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值
3.当调用String的replace()方法修改指定字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值
4.通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中
5.字符串常量池中是不会存储相同内容的字符串的
方式一:通过字面量定义的方式 String s1 = "abc";
方式二:通过new + 构造器的方式 String s1 = new String();
面试题:String s = new String("abc");方式创建对象,在内存中创建了几个对象?
两个:一个是堆空间中new结构,另一个是char[]对应的常量池中的数据:"abc"
1.常量与常量的拼接结果在常量池。且常量池中不会存在相同内容的常量。
2.只要其中有一个是变量,结果就在堆中。
3.如果拼接的结果调用intern()方法,返回值就在常量池中
例如:
String s1 = "javaEE";
String s2 = "hadoop";
String s3 = "javaEEhadoop";
String s4 = "javaEE" + "hadoop";
String s5 = s1 + "hadoop";
String s6 = "javaEE" + s2;
String s7 = s1 + s2;
System.out.println(s3 == s4);//true
System.out.println(s3 == s5);//false
System.out.println(s3 == s6);//false
System.out.println(s3 == s7);//false
System.out.println(s5 == s6);//false
System.out.println(s5 == s7);//false
System.out.println(s6 == s7);//false
String s8 = s6.intern();//返回值得到的s8使用的常量值中已经存在的“javaEEhadoop”
System.out.println(s3 == s8);//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(不包含)的一个子字符串。
boolean endsWith(String suffix):测试此字符串是否以指定的后缀结束
boolean startsWith(String prefix):测试此字符串是否以指定的前缀开始
boolean startsWith(String prefix, int toffset):测试此字符串从指定索引开始的子字符串是否以指定前缀开始
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):返回指定子字符串在此字符串中最后一次出现处的索引,从指定的索引开始反向搜索
注:indexOf和lastIndexOf方法如果未找到都是返回-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 替换此字符串匹配给定的正则表达式的第一个子字符串。
匹配:
boolean matches(String regex):告知此字符串是否匹配给定的正则表达式。
切片:
String[] split(String regex):根据给定正则表达式的匹配拆分此字符串。
String[] split(String regex, int limit):根据匹配给定的正则表达式来拆分此字符串,最多不超过limit个,如果超过了,剩下的全部都放到最后一个元素中。
String与基本数据类型、包装类之间的转换。
String --> 基本数据类型、包装类:调用包装类的静态方法:parseXxx(str)
基本数据类型、包装类 --> String:调用String重载的valueOf(xxx)
String与char[]之间的转换
String --> char[]:调用String的toCharArray()
char[] --> String:调用String的构造器
String与byte[]之间的转换
String --> byte[]:调用String的getBytes()
byte[] --> String:调用String的构造器
编码:字符串 -->字节 (看得懂 --->看不懂的二进制数据)
解码:编码的逆过程,字节 --> 字符串 (看不懂的二进制数据 ---> 看得懂)
说明:解码时,要求解码使用的字符集必须与编码时使用的字符集一致,否则会出现乱码
String:不可变的字符序列;底层使用final 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
//问题2. 扩容问题:如果要添加的数据底层数组盛不下了,那就需要扩容底层的数组。
默认情况下,扩容为原来容量的2倍 + 2,同时将原有数组中的元素复制到新的数组中。
指导意义:开发中建议大家使用:StringBuffer(int capacity) 或 StringBuilder(int capacity)
因为StringBuffer的底层数组一般设置为16,当超出该数时就要扩容,要把数组中所有的内容copy一份放到另一个更长的数组中,这种形式太费内存了
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 > StringBuffer > String
System类中的currentTimeMillis()
返回当前时间与1970年1月1日0时0分0秒之间以毫秒为单位的时间差,称为时间戳
java.util.Date类
|---java.sql.Date类
java.util.Date类是java.sql.Date的父类
1.java.util.Date两个构造器的使用
>构造器一:Date():创建一个对应当前时间的Date对象
>构造器二:创建指定毫秒数的Date对象 Date(long time)
2.java.util.Date两个方法的使用
>toString():显示当前的年、月、日、时、分、秒
>getTime():获取当前Date对象对应的毫秒数。(时间戳)
3. java.sql.Date对应着数据库中的日期类型的变量
>如何实例化 java.sql.Date date = new java.sql.Date(long time); -->只有父类中的毫秒数创建方法
>如何将java.util.Date对象转换为java.sql.Date对象,即javaSE层面的Date对象要向数据库进行操作,要进行转换
情况一:多态性的体现,new的是子类,变量接收的是父类,将父类变量强转为子类
Date date = new java.sql.Date(452065342165L);
java.sql.Date date2 = (java.sql.Date)date; //之前多态的使用,用instanceof关键字判断是否是,然后强转
情况二:从util层面的Date直接转换成sql层面的Date,new的也是util下的Date
Date date = new Date();
java.sql.Date date2 = new java.sql.Date(date.getTime()); //调用构造器转化
通过调用sql下的构造器转化,因为sql的构造器中的参数就是long型的,而util中Date的getTime()方法返回的就是时间戳,他们的时间戳是一致的
因为Date类型的输出形式不太好看,所以用SimpleDateFormat类对Date进行修饰
SimpleDateFormat的使用:SimpleDateFormat对日期Date类的格式化和解析
1.两个操作:
1.1格式化:日期 --->字符串
String format = sdf.format(date); //调用format方法将Date改为我们想看到的形式
1.2解析:格式化的逆过程,字符串 ---> 日期
Date date1 = sdf.parse(str); //调用parse方法将输入的字符串改为Date的形式,字符串要符合要求,否则就会有异常,我们可以在SimpleDateFormat的初始化中设定某种形式
2.SimpleDateFormat的实例化
//实例化SimpleDateFormat:使用默认的构造器
SimpleDateFormat sdf = new SimpleDateFormat();
//按照指定的方式格式化和解析:调用带参的构造器
使用这种构造器的原因是在通过字符串改为Date类型时,由于输入的字符串类型要匹配否则抛出异常,在这里我们可以自己规定就方便了
SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
Calendar日历类(抽象类)的使用
1.实例化
方式一:创建其子类(GregorianCalendar)的对象
方式二:调用其静态方法getInstance() //返回的对象是GregorianCalendar的对象
2.常用方法
get()
set()
add()
getTime():日历类---> Date
setTime():Date ---> 日历类
第一代时间API是Date,第二代是Calendar,但是Calendar可变,有偏移量等等问题,不适合使用,在JDK8中又增加LocalDate、LocalTime、LocalDateTime
LocalDate、LocalTime、LocalDateTime的使用
说明:
1.LocalDateTime相较于LocalDate、LocalTime,使用频率要高
2.类似于Calendar
1.初始化
now():获取当前的日期、时间、日期+时间
of():设置指定的年、月、日、时、分、秒。没有偏移量
2.方法
getXxx():获取相关的属性
withXxx():设置相关的属性
plusXxx():增加相关属性
minusXxx():减少相关属性
类似于java.util.Date类
now():获取本初子午线对应的标准时间
Instant instant = Instant.now();
System.out.println(instant);//2019-02-18T07:29:41.719Z
添加时间的偏移量
OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours(8));
System.out.println(offsetDateTime);//2019-02-18T15:32:50.611+08:00
toEpochMilli():获取自1970年1月1日0时0分0秒(UTC)开始的毫秒数 ---> Date类的getTime()
long milli = instant.toEpochMilli();
System.out.println(milli);
ofEpochMilli():通过给定的毫秒数,获取Instant实例 -->Date(long millis)
Instant instant1 = Instant.ofEpochMilli(1550475314878L);
System.out.println(instant1);
类似于SimpleDateFormat
1.实例化
方式一:预定义的标准格式。如:ISO_LOCAL_DATE_TIME;ISO_LOCAL_DATE;ISO_LOCAL_TIME
DateTimeFormatter formatter = DateTimeFormatter.ISO_LOCAL_DATE_TIME;
方式二:
本地化相关的格式。如:ofLocalizedDateTime()
FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT :适用于LocalDateTime
DateTimeFormatter formatter1 = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.LONG);
本地化相关的格式。如:ofLocalizedDate()
FormatStyle.FULL / FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT : 适用于LocalDate
DateTimeFormatter formatter2 = DateTimeFormatter.ofLocalizedDate(FormatStyle.MEDIUM);
重点:方式三:自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)
DateTimeFormatter formatter3 = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss");
说明:Java中的对象,正常情况下,只能进行比较:== 或 != 。不能使用 > 或 < 的
但是在开发场景中,我们需要对多个对象进行排序,言外之意,就需要比较对象的大小。
如何实现?使用两个接口中的任何一个:Comparable 或 Comparator
Comparable属于自然比较,Comparator属于定制比较
Comparable接口的方式一旦一定,保证Comparable接口实现类的对象在任何位置都可以比较大小。
Comparator接口属于临时性的比较。
我们可以将要比较的类(implements)实现Comparable接口,实现CompareTo()方法,而Comparator接口我们在使用时做了个临时的比较,采用匿名实现类的匿名对象方式比较,再次比较时还要再创建,不过对于已经在Java源码中确定的比较方式例如String重写了CompareTo()方法只能从小到大排,我们在调用Arrays.sort(String[])对String进行排序时,想要从大到小排,可以实现Comparator接口,例如:
Arrays.sort(String[],new Comparator(){
compare() //重写compare方法,可以从大到小排
})
System类是系统类
native long currentTimeMillis();获取1970年到现在的毫秒数,时间戳
void exit(int status);status为0表示正常退出,非0表示异常退出
void gc();请求垃圾回收,具体什么时候回收看计算机自己的处理
String getProperty(String key);获得系统中属性名为key的属性对应的值
Math类是进行数字操作的类
BigInteger和BigDecimal可以存储任意精度,任意位数的数,int和float还有double可能不够用,可以用他们替换
原文:https://www.cnblogs.com/tianzhenaichirou/p/14586411.html