您好,登錄后才能下訂單哦!
這篇文章主要講解了“netty中常用的xml編碼解碼器怎么用”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“netty中常用的xml編碼解碼器怎么用”吧!
在json之前,xml是最常用的數據傳輸格式,雖然xml的冗余數據有點多,但是xml的結構簡單清晰,至今仍然運用在程序中的不同地方,對于netty來說自然也提供了對于xml數據的支持。
netty對xml的支持表現在兩個方面,第一個方面是將編碼過后的多個xml數據進行frame拆分,每個frame包含一個完整的xml。另一方面是將分割好的frame進行xml的語義解析。
進行frame拆分可以使用XmlFrameDecoder,進行xml文件內容的解析則可以使用XmlDecoder,接下來我們會詳細講解兩個decoder實現和使用。
因為我們收到的是數據流,所以不確定收到的數據到底是什么樣的,一個正常的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中的分隔符有三個,分別是’<', ‘>’ 和 ‘/’。
在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);
將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編碼解碼器怎么用這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。