Java 安装 Java:第一个程序 Hello World Java:建议使用 UTF-8 编写 Java 代码 Java:package 包命名规范 使用 Intellij IDEA 创建 Java 项目 Java 布尔类型 Java 处理日期和时间 Java 正则表达式 Java finalize 方法 Java:空值 null Java 如何触发垃圾回收 Java ThreadLocal Java InheritableThreadLocal Java Integer之间的比较 Java 动态代理 Java 匿名类 Java 枚举 Java 如何静态导入 import static println Java 引用级别:强引用、软引用、弱引用、幽灵引用 Java try finally return 解惑 Java WeakHashMap Java ReferenceQueue 怎么写 Java 示例代码? Java 匿名类双大括号初始化 什么是 Java Bean Java 多行字符串 Java 快速生成 List Java 快速生成 Map Java 将异常堆栈转换为 String JDK SPI 的使用和源码分析 Java Map 中的 key 和 value 能否为 null ? Java List 和 数组的互相转换 Java 获取环境变量 Java 获取和设置系统属性 Java:如何获取当前进程的 PID ? Java 字符串左侧 右侧补充空格或者其他字符 Java 线程 Java:如何获取文本文件内容 Java:读取资源文件内容 Java:使用 JavaFx 构建 GUI Java:Class 类 Java:使用 instanceof 判断对象类型 一个自定义的 Java 工具类 Java:获取当前函数所属类的类名 Java:获取当前执行的函数名 Java:使用 String 的 split 函数拆分字符串 Java:获取字符的 Unicode 编号(代码点) Java:获取当前工作目录 Java:使用 Class 对象的 isArray 方法判断对象是否为数组 使用 Java 生成 CSV 文件 Java Mockito 测试框架快速入门 JUnit 入门 JUnit 单测隔离 Java JOOR 反射库 Java alibaba transmittable-thread-local 库:让 ThreadLocal 跨线程传播 Java 日志组件 slf4j 的使用和源码分析 Java Lombok 库:为你减少样板代码 Java:使用 cglib 实现动态代理 Java Hibernate validator 校验框架 Java 使用 Hessian2 序列化和反序列化 H2 数据库快速入门 Java:使用 Gson 库处理 JSON 数据 Java 集成 groovy 构建规则引擎 Java 13:安装 Java 13 新特性:文本块(多行字符串) 卸载 MacOS 上安装的 Java Java:执行 sql 文件 Java JDK 有哪些发行版 ? java拾遗:String和数组 java拾遗:由反转数组想到System.out的实现机制 java拾遗:如何读取properties文件内容 Java并发概念汇总 java拾遗:System.out.println()是什么? java拾遗:通过示例理解位运算 使用“庖丁解牛”进行中文分词 DBUtils简明教程 试用velocity模板引擎 Java:将字符串哈希为数字 kafka SnappyError no native library is found 问题

java拾遗:String和数组


#Java


2015-01-22

环境:jdk 1.7。

本文可以用一句话总结:String不是基本类型,不是数组,是类,而数组也是类。

由于脚本语言使用较多,我在java中也写过这样的代码:

String str = "hi";
System.out.println(str[1]); // 错误的

可惜是错的。str是java.lang.String类的一个对象,无法使用[]运算符。真正可用的代码是这样的:

String str = "hi";
System.out.println(str.charAt(1));

数组是类

请参考 java拾遗:由反转数组想到System.out的实现机制

其实,从下面的语句就可以看出来:

int[] ia = new int[9];

也可以从下面的代码片段中看出来:

char[] arr = {'a', 'b', 'c'};
char[] arr2 = arr;         // 引用   
System.out.println(arr);   // abc
System.out.println(arr2);  // abc
arr[1] = '6';
System.out.println(arr);   // a6c
System.out.println(arr2);  // a6c

String不是基本类型

java提供了8个基本类型,分别是:byte、short、int、long、float、double、boolean和char,并没有String。基本类型可以以下面的形式定义变量并赋值:

int a = 5;

上面的变量a是一个基本类型变量,不是对象,所以变量a没有任何属性和方法,只有

当然,这8个基本类型也有对应的类,分别是java.lang包下的Byte、Short、Integer、Long、Float、Double、Boolean、Character。这8个类都有一个私有的使用基本类型的变量value来保存值。

Java中有“自动装箱”、“自动拆箱”的概念,这是针对基本类型和它们对应的类而言的。

自动装箱:把基本类型用它们对应的引用类型包装起来,使它们具有对象的特质,可以调用toString()、hashCode()、getClass()、equals()等方法。

拆箱:跟自动装箱的方向相反,将Integer及Double这样的引用类型的对象重新简化为基本类型的数据。

注意:自动装箱和拆箱是由编译器来完成的,编译器会在编译期根据语法决定是否进行装箱和拆箱动作。

代码示例:

Integer a = 5;          // 自动装箱
int b = new Integer(6); // 自动拆箱
System.out.println(a);  // 5
System.out.println(b);  // 6

String不是数组

因为无法使用数组运算符[]

String内部实现

String是一个类,具体是java.lang.String,查看其源码,可以看到

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];

char数组value用来存储字符串。注意char类型的下面的性质:

char类型是一个单一的16位Unicode字符;
最小值是’\u0000’(即为0);
最大值是’\uffff’(即为65,535);
char数据类型可以储存任何字符;
例子:char letter = 'A'。

String类并没有暴露可以修改value的方法,所以可以认为String是无法修改的。

下面是一个示例:

char[] arr = {'a', 'b', 'c'};
String str = new String(arr);
System.out.println(arr);  // abc
System.out.println(str);  // abc
arr[1] = '6';
System.out.println(arr);  // a6c
System.out.println(str);  // abc

其中:new String(arr);使用了下面的构造方法:

public String(char value[]) {
    this.value = Arrays.copyOf(value, value.length);
}

这相当于把arr的内容拷贝一份,让后让this.value指向它新生成的数组。所以,在arr[1] = '6';后,字符串对象str的value并没改变。

那么"123"意味着?

看一下下面的代码:

package hellojava;
    
public class HelloJava {
    
    void hi(char[] arr) {
        System.out.println("hi, char[]");
    }
    
    void hi(String str) {
        System.out.println("hi, String");
    }
    
    public static void main(String[] args) {

        HelloJava hj = new HelloJava();
        char[] arr = {'a', 'b'};
        hj.hi("123");
        // hj.hi({'a', 'b'}); // 错误
        hj.hi(arr);
        
    }
}

运行结果是:

hi, String
hi, char[]

所以"123"是代表着String。

如果上面的代码中,没有void hi(String str)方法,那么hj.hi("123");也会报错。

补充

String类下还有这个构造方法:

String(char[] value, boolean share) {
    // assert share : "unshared not supported";
    this.value = value;
}

如果该构造方法使用public修饰了,那么,下面的代码是成立的:

// 这段代码在实际中是不成立的
char[] arr = {'a', 'b', 'c'};
String str = new String(arr, true);
System.out.println(arr);  // abc
System.out.println(str);  // abc
arr[1] = '6';
System.out.println(arr);  // a6c
System.out.println(str);  //-> 结果会是: a6c

也就是String能被修改了,然而该构造方法没有使用public修饰。只有同在java.lang包中的类才能使用。



( 本文完 )