溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

netty中常用的xml編碼解碼器怎么用

發(fā)布時間:2022-05-16 09:16:16 來源:億速云 閱讀:149 作者:iii 欄目:開發(fā)技術

這篇文章主要講解了“netty中常用的xml編碼解碼器怎么用”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“netty中常用的xml編碼解碼器怎么用”吧!

簡介

在json之前,xml是最常用的數據傳輸格式,雖然xml的冗余數據有點多,但是xml的結構簡單清晰,至今仍然運用在程序中的不同地方,對于netty來說自然也提供了對于xml數據的支持。

netty對xml的支持表現在兩個方面,第一個方面是將編碼過后的多個xml數據進行frame拆分,每個frame包含一個完整的xml。另一方面是將分割好的frame進行xml的語義解析。

進行frame拆分可以使用XmlFrameDecoder,進行xml文件內容的解析則可以使用XmlDecoder,接下來我們會詳細講解兩個decoder實現和使用。

XmlFrameDecoder

因為我們收到的是數據流,所以不確定收到的數據到底是什么樣的,一個正常的xml數據可能會被拆分成多個數據frame。

如下所示:

+-------+-----+--------------+
   | <this | IsA | XMLElement/> |
   +-------+-----+--------------+

這是一個正常的xml數據,但是被拆分成為了三個frame,所以我們需要將其合并成為一個frame如下:

 +-----------------+
   | <thisIsAXMLElement/> |
   +-----------------+

還有可能不同的xml數據被分拆在多個frame中的情況,如下所示:

+-----+-----+-----------+-----+----------------------------------+
   | <an | Xml | Element/> | <ro | ot><child>content</child></root> |
   +-----+-----+-----------+-----+----------------------------------+

上面的數據需要拆分成為兩個frame:

+-----------------+-------------------------------------+
   | <anXmlElement/> | <root><child>content</child></root> |
   +-----------------+-------------------------------------+

拆分的邏輯很簡單,主要是通過判斷xml的分隔符的位置來判斷xml是否開始或者結束。xml中的分隔符有三個,分別是&rsquo;<', &lsquo;>&rsquo; 和 &lsquo;/&rsquo;。

在decode方法中只需要判斷這三個分隔符即可。

另外還有一些額外的判斷邏輯,比如是否是有效的xml開始字符:

 private static boolean isValidStartCharForXmlElement(final byte b) {
        return b >= 'a' && b <= 'z' || b >= 'A' && b <= 'Z' || b == ':' || b == '_';
    }

是否是注釋:

  private static boolean isCommentBlockStart(final ByteBuf in, final int i) {
        return i < in.writerIndex() - 3
                && in.getByte(i + 2) == '-'
                && in.getByte(i + 3) == '-';
    }

是否是CDATA數據:

 private static boolean isCDATABlockStart(final ByteBuf in, final int i) {
        return i < in.writerIndex() - 8
                && in.getByte(i + 2) == '['
                && in.getByte(i + 3) == 'C'
                && in.getByte(i + 4) == 'D'
                && in.getByte(i + 5) == 'A'
                && in.getByte(i + 6) == 'T'
                && in.getByte(i + 7) == 'A'
                && in.getByte(i + 8) == '[';

通過使用這些方法判斷好xml數據的起始位置之后,就可以調用extractFrame方法將要使用的ByteBuf從原始數據中拷貝出來,最后放到out中去:

final ByteBuf frame =
                    extractFrame(in, readerIndex + leadingWhiteSpaceCount, xmlElementLength - leadingWhiteSpaceCount);
            in.skipBytes(xmlElementLength);
            out.add(frame);

XmlDecoder

將xml數據拆分成為一個個frame之后,接下來就是對xml中具體數據的解析了。

netty提供了一個xml數據解析的方法叫做XmlDecoder,主要用來對已經是一個單獨的xml數據的frame進行實質內容的解析,它的定義如下:

public class XmlDecoder extends ByteToMessageDecoder

XmlDecoder根據讀取到的xml內容,將xml的部分拆分為XmlElementStart,XmlAttribute,XmlNamespace,XmlElementEnd,XmlProcessingInstruction,XmlCharacters,XmlComment,XmlSpace,XmlDocumentStart,XmlEntityReference,XmlDTD和XmlCdata。

這些數據基本上覆蓋了xml中所有可能出現的元素。

所有的這些元素都是定義在io.netty.handler.codec.xml包中的。

但是XmlDecoder對xml的讀取解析則是借用了第三方xml工具包:fasterxml。

XmlDecoder使用了fasterxml中的AsyncXMLStreamReader和AsyncByteArrayFeeder用來進行xml數據的解析。

這兩個屬性的定義如下:

 private static final AsyncXMLInputFactory XML_INPUT_FACTORY = new InputFactoryImpl();
    private final AsyncXMLStreamReader<AsyncByteArrayFeeder> streamReader;
    private final AsyncByteArrayFeeder streamFeeder;
            this.streamReader = XML_INPUT_FACTORY.createAsyncForByteArray();
        this.streamFeeder = (AsyncByteArrayFeeder)this.streamReader.getInputFeeder();

decode的邏輯是通過判斷xml element的類型來分別進行不同數據的讀取,最后將讀取到的數據封裝成上面我們提到的各種xml對象,最后將xml對象添加到out list中返回。

感謝各位的閱讀,以上就是“netty中常用的xml編碼解碼器怎么用”的內容了,經過本文的學習后,相信大家對netty中常用的xml編碼解碼器怎么用這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI