关于InputStreamReader中的StreamDecoder中的leftoverChar

在阅读InputStreamReader源代码的时候,发现他是一个装饰器模式,所有read的方法都依托到了他的成员变量StreamDecoder sd中,在点开StreamDecoder源代码发现了其读取逻辑中有个leftoverChat的字段,查阅资料发现并没有解释这个字段的文章,回想起以前学习的关于java编码的知识,想到这个应该就是java的增补编码

首先java的Unicode编码一共有1,112,064字符,但是Unicode是16位表示,最大表示65536个字符,于是只能用两个字符来标识一个字符,把uD800到uDBFF共1024个字符空出来表示high-surrogates range,把uDC00到uDFFF共1024个字符空出来表示low-surrogates range,两辆组合就可以表示1048576,加上65536-1024-1024个就可以表示所有的字符

接下来看看StreamDecoder 读取字符的关键代码

// 读取方法,返回单个字符,实际调用下面的read0()方法
public int read() throws IOException
{
    return read0();
}
     
// 返回单个字符
private int read0() throws IOException
{
    synchronized (lock)
    {
        // haveLeftoverChar默认fasle,实例域有设置,因此最开始不会进入
        if (haveLeftoverChar)
        {
            haveLeftoverChar = false;
            return leftoverChar;
        }
        // 创建了长度为2的字符数组
        char cb[] = new char[2];
        // 调用下面read(char cbuf[], int offset, int length)方法,返回读取的字符个数
        int n = read(cb, 0, 2); 
        switch (n)
        {
                // 若返回-1.代表读到末尾
            case -1: 
                return -1;
                // 读取到2个字符,但是方法要求返回一个字符,因此第二个字符就赋予变量暂时缓存
            case 2: 
                // 第二个字符赋予变量leftoverChar 
                leftoverChar = cb[1];
                //修改条件为真  
                haveLeftoverChar = true; 
                // 代表读取到1个字符,直接返回数组的第一个元素即可
            case 1: 
                return cb[0];
            default:
                assert false : n;
                return -1;
        }
    }
}

很简单,当读取到一个字符是便直接返回,读取到两个字符是就是增补编码,先返回high-surrogates range,下次再返回low-surrogates range,而StreamDecoder 中的leftoverChat就是指low-surrogates range

你可能感兴趣的:(java)