No.5 表达式中的陷阱

1. 关于字符串的陷阱

  • JVM对字符串的处理
    • String java = new String("Java"); 创建了几个对象?
      • 2个。“Java”直接量对应的字符串对象;new String()构造器返回的字符串对象
      • java.intern() 可以字符串池中的直接量对象。
    • 字符串及基本类型的包装类,Java允许通过直接量的方式来创建对应的Java对象;除此之外,简单的算术表达式、连接计算(编译时可以确定具体值)也可以直接创建对应的Java对象
    • 字符串直接量,JVM会使用一个字符串池保存,第一次使用该直接量时,会将其放在字符串池进行缓存(缓存的意思就是以后再用的话,就直接从池中取,而不会新建啦),一般情况下不会对缓存池中的字符串进行垃圾回收
    • 只要是编译时就可以确定其具体值的字符串直接量,JVM会在编译时确定其具体值,并让它指向字符串池中对应的字符串,否则,就不能利用JVM的字符串池了
      • 何为编译时确定具体值?
        • 没有方法调用;没有变量引用
        • 当然,“宏替换”的变量除外
    • String str = "hello" + “world”; 创建了几个字符串对象
      • 1个。因为在编译时就可以确定其计算后的值,因此就直接把计算后的值放到池中就好了,其它的也没有必要啊
    • 在使用字符串、基本类型包装类的对象实例时,尽量通过  直接量 来创建实例。可以少创建一个对象,提升性能。
  • String是典型的不可变类。可变字符串推荐使用StringBuilder
  • 字符串比较
    • 底层是字符串数组
    • 两个字符串左对齐,依次比较二者的每个字符

2. 表达式类型的陷阱

  表达式也是具有制定的数据类型的

  • 表达式类型的自动提升
    • 当算事表达式中含有多个基本数据类型时,整个表达式的数据类型会自动提升,提升到与表达式中最高等级操作数同样的类型。
    • 操作数的等级排列如图示

No.5 表达式中的陷阱

  • 复合赋值运算符的陷阱(+=、-=、/=、...)
    • E1 op= E2 等价于  E1  = (E1的类型)(E1 op E2)
    • eg:a += 5  等价于 a = (a的类型)(a + 5)
    • 陷阱:它潜在的隐式转换  可能 导致计算结果的   高位“截断”(结果强制转换发生溢出时)

3. 输入法导致的陷阱

  • 代码中 含有   全角空格。。。提示:“非法字符:\12288”
  • java程序中不能包含  全角字符  ,但  字符串中和注释中可以

4注释的字符必须合法

  • 大部分情况,编译器会直接忽略掉注释部分,例外:要求注释部分为合法字符,Java允许直接使用\uXXXX形式来表示字符,其中X必须为0~F间的字符
  • // \unit5\5.4\Hello.javaInvalid unicode

5. 转义字符: Java对待转义字符的处理:直接替换成对应的字符

  • 慎用Unicode转义形式
    • System.out.println("abc\u000a".length());String literal is not properly closed by a double-quote  因为\u000a表示一个换行符
  • 中止行注释的
    • // \u000a代表一个换行符 ,会造成错误:代表一个换行符 cannot be resolved to a type

6. 泛型可能引起的错误

  • 如果使用带泛型声明的类时没有传入类型实参,那么这个类型参数默认是声明该参数时指定的第一个上限类型,这个类型参数也被称为raw type(原始类型)
  • 原始类型带来的擦除
    • 当把一个具有泛型信息的对象赋给另一个没有泛型信息的变量时,所有尖括号之间的类型信息都将被丢弃。例:讲一个List<String>类型的对象转为List,则该List对集合元素的类型检查变成了类型变量的上限(即 Object)
    •  package test1;
      
       import java.util.ArrayList;
      import java.util.List; public class Test<T extends Number> {
      T size;
      public Test(){}
      public Test(T size){
      this.size = size;
      }
      public T getSize() {
      return size;
      }
      public void setSize(T size) {
      this.size = size;
      }
      public List<String> getStrings(){
      List<String> list = new ArrayList<>();
      for (int i = 0; i < 3; i++) {
      list.add(new Test<Integer>(10 * i).toString());
      }
      return list;
      } public static void main(String[] args) {
      Test<Integer> test1 = new Test<>(6);
      for (String string : test1.getStrings()) {
      System.out.println(string);
      } Test test2 = test1;
      //test2.getStrings()方法中List<String>的String也会丢失;因此是所有尖括号中的泛型信息都会丢失
      for (String string : test2.getStrings()) {//Type mismatch: cannot convert from element type Object to String
      System.out.println(string);
      } }
      }

7. 正则

  • split()方法中传入的是正则,正则中“.”匹配任何字符,故需要对其转义 str.split("\\.");
  • ps:"\"  <mine>
    • 字符串中、正则中的“\”都表示各自的转义,要用真正的斜线“\”时,需要用"\\"表示。两者均是如此
    • Java中又是用字符串来表示正则,故表示正则中的斜线时,需要用“\\\\”前两个斜线表示正则中的第一个斜线,后两个代表第二个
    • (\\)(\\)即是(\)(\),因为字符串中表示斜线,需要用\\。另,各自的转义符\是各自的,正则不能利用字符串的转义符,正则的转义符在字符串看来就是一个“\”,两者互不干扰,只是可以互相表示
    • ——> 方法:先写出来普通的正则表达式,在Java的字符串中用“\\”将正则表达式的斜杠一一表示出来就可以了,有几个就替换几次
      • 例:正则:\d  Java中字符串“\\d”,即“(\\)d”的意思;正则:\\  字符串“\\\\”,即“(\\)(\\)”;正则:\\\d,字符串“\\\\\\d”,即“(\\)(\\)(\\)d”

8. 多线程

  • start()方法启动线程,而不是run()方法
  • 静态同步方法的同步监视器为  当前类本身(类对象)
  • 分析一个程序不能仅仅停留在静态的代码上,而是应该从程序执行过程来把握程序的运行细节
上一篇:vue.js 精学组件记录


下一篇:JS简单表单验证