String类型,也就是字符串类型,是Redis中最简单的存储类型。
其value是字符串,不过根据字符串的格式不同,又可以分为3类:
-
string:普通字符串
-
int:整数类型,可以做自增、自减操作
-
float:浮点类型,可以做自增、自减操作
不管是哪种格式,底层都是字节数组形式存储,只不过是编码方式不同。字符串类型的最大空间不能超过512m.
-
String类常见面试题
-
String类常用API
-
String案例:验证码、登录、隐私号码
String类型的常见命令
-
SET:添加或者修改已经存在的一个String类型的键值对
-
GET:根据key获取String类型的value
-
MSET:批量添加多个String类型的键值对
-
MGET:根据多个key获取多个String类型的value
-
INCR:让一个整型的key自增1
-
INCRBY:让一个整型的key自增并指定步长,例如:incrby num 2 让num值自增2
-
INCRBYFLOAT:让一个浮点类型的数字自增并指定步长
-
SETNX:添加一个String类型的键值对,前提是这个key不存在,否则不执行
-
SETEX:添加一个String类型的键值对,并且指定有效期
String常用使用技巧
1.根据下标得到一个字符串中对应下标的那个字符(首字符下标为0)(掌握)
方法:public char charAt(int index(下标)){}
调用:char c = "String".charAt(int index);
char c = String s.charAt(int index);
2.比较两个字符串大小并且输出相减的值(按字典顺序)(了解)
字符串之间比较大小不能直接使用>或<,需要使用compareTo方法,它是拿着字符串第一个字母和后面字符串的第一个字母比较,能分胜负就不再比较了
方法:public int compareTo(String anotherString){}
调用:int i = "String".compareTo("String anotherString");
int i = String s.compareTo(String anotherString);
3.判断前面的字符串中是否包含后面的子字符串(掌握)
方法:public boolean contains(CharSequence s(子字符串)){}
调用:boolean b = "String".contains("CharSequence s");
boolean b = String s.contains(CharSequence s);
4.判断当前字符串是否以某个子字符串结尾(掌握)
方法:public boolean endsWith(String suffix){}
调用:boolean b = "String".endsWith("String suffix");
boolean b = String s.endsWith(String suffix);
5.将此字符串与指定的对象比较。比较两个字符串必须使用equals方法,不能使用“==”(掌握)
方法:public boolean equals(Object anObject){}
调用:boolean b = "String".equals("Object anObject");
boolean b = String s.equals(Object anObject);
equals只能看出相等不相等
compareTo方法可以看出是否相等,并且同时还可以看出谁大谁小
equals方法有没有调用compareTo方法?老版本可以看一下。JDK13中并没有调用compareTo()方法
6.判断两个字符串是否相等,并且同时忽略大小写(掌握)
方法:public boolean equalsIgnoreCase(String anotherString)
调用:
boolean b = "String".equalsIgnoreCase("String anotherString");
boolean b = String s.equalsIgnoreCase(String anotherString);
7.将字符串对象转换成字节数组(掌握)
方法:public byte[] getBytes(){}
调用:byte[] bytes = "String".getBytes();
byte[] bytes = String s.getBytes();
8.判断某个子字符串在当前字符串中第一次出现处的索引即下标(掌握)
方法:public int indexOf(String str){}
调用:int i = "String".indexOf(String str);
int i = String s.indexOf(String str) ;
9.判断某个字符串是否为“空字符串”(底层源代码调用的应该是字符串的length()方法)(掌握)
方法:public boolean isEmpty(){}
调用:boolean b = "String".isEmpty();
boolean b = String s.isEmpty();
10.返回字符串的长度(掌握)
方法:public int length(){}
调用:int i = "String".length();
int i = String s.length();
问题:判断数组长度和判断字符串长度是不是一样?不一样,判断数组长度是length属性,判断字符串长度是length()方法
面试题,不要被String类的问题给问死了!
1. String & StringBuilder & StringBuffer 的区别
1.1 效率
String 是不可变的,每次操作都会创建新的变量,而另外两个是可变的,不需要创建新的变量;另外,StringBuffer 的每个操作方法都使用 synchronized 关键字保证线程安全,增加了更多加锁 & 释放锁的时间。因此,操作效率的简单排序为:StringBuilder > StringBuffer > String。
1.2 线程安全
String 不可变,所以 String 和 StringBuffer 都是线程安全的,而 StringBuilder 是非线程安全的。
类型 | 操作效率 | 线程安全 |
---|---|---|
String | 低 | 安全(final) |
StringBuffer | 中 | 安全(synchronized) |
StringBuilder | 高 | 非安全 |
2. 下面这句话在内存中创建了几个对象
String st1 = new String(“abc”);
答案是:在内存中创建两个对象,一个在堆内存,一个在常量池,堆内存对象是常量池对象的一个拷贝副本。
分析:
我们下面直接来一个内存图。
当我们看到了new这个关键字,就要想到,new出来的对象都是存储在堆内存。然后我们来解释堆中对象为什么是常量池的对象的拷贝副本。“abc”属于字符串,字符串属于常量,所以应该在常量池中创建,所以第一个创建的对象就是在常量池里的“abc”。
第二个对象在堆内存为啥是一个拷贝的副本呢,这个就需要在JDK API 1.6找到String(String original)这个构造方法的注释:初始化一个新创建的 String 对象,使其表示一个与参数相同的字符序列;换句话说,新创建的字符串是该参数字符串的副本。所以,答案就出来了,两个对象。
3、什么是String,它是什么数据类型?
String是定义在 java.lang 包下的一个类。它不是基本数据类型。
String是不可变的,JVM使用字符串池来存储所有的字符串对象。
4、创建String对象的不同方式有哪些?
-
和使用其他类一样通过new关键字来创建。 使用这种方式时,JVM创建字符串对象但不存储于字符串池。我们可以调用intern()方法将该字符串对象存储在字符串池,如果字符串池已经有了同样值的字符串,则返回引用。
-
使用双引号直接创建。
使用这种方式时,JVM去字符串池找有没有值相等字符串,如果有,则返回找到的字符串引用。否则创建一个新的字符串对象并存储在字符串池。 String str = new String("abc"); String str1 = "abc";
5 为什么 String 要设计为不可变?
-
1、不可变类 String 可以避免修改后无法定位散列表键值对: 假设 String 是可变类,当我们在 HashMap 中构建起一个以 String 为 Key 的键值对时,此时对 String 进行修改,那么通过修改后的 String 是无法匹配到刚才构建过的键值对的,因为修改后的 hashCode 可能是变化的。而不可变类可以规避这个问题。
-
2、线程安全: 不可变对象本质是线程安全,不需要同步;
提示: 反射可以破坏 String 的不可变性。