Каков самый быстрый / эффективный способ чтения этого XML в словарь (Linq или что-то еще?)
Я очень новичок в разборе XML, и я начал изучать linq, который, я думаю, может быть лучшим решением здесь. Меня в основном интересует производительность, так как приложение, которое я создаю, будет читать биржевые цены, которые иногда могут меняться очень быстро. Я получаю следующее сообщение от сервера:
<?xml version="1.0" encoding="utf-16"?>
<events>
<header>
<seq>0</seq>
</header>
<body>
<orderBookStatus>
<id>100093</id>
<status>Opened</status>
</orderBookStatus>
<orderBook>
<instrumentId>100093</instrumentId>
<bids>
<pricePoint>
<price>1357.1</price>
<quantity>20</quantity>
</pricePoint>
<pricePoint>
<price>1357.0</price>
<quantity>20</quantity>
</pricePoint>
<pricePoint>
<price>1356.9</price>
<quantity>71</quantity>
</pricePoint>
<pricePoint>
<price>1356.8</price>
<quantity>20</quantity>
</pricePoint>
</bids>
<offers>
<pricePoint>
<price>1357.7</price>
<quantity>51</quantity>
</pricePoint>
<pricePoint>
<price>1357.9</price>
<quantity>20</quantity>
</pricePoint>
<pricePoint>
<price>1358.0</price>
<quantity>20</quantity>
</pricePoint>
<pricePoint>
<price>1358.1</price>
<quantity>20</quantity>
</pricePoint>
<pricePoint>
<price>1358.2</price>
<quantity>20</quantity>
</pricePoint>
</offers>
<lastMarketClosePrice>
<price>1356.8</price>
<timestamp>2011-05-03T20:00:00</timestamp>
</lastMarketClosePrice>
<dailyHighestTradedPrice />
<dailyLowestTradedPrice />
<valuationBidPrice>1357.1</valuationBidPrice>
<valuationAskPrice>1357.7</valuationAskPrice>
<lastTradedPrice>1328.1</lastTradedPrice>
<exchangeTimestamp>1304501070802</exchangeTimestamp>
</orderBook>
</body>
</events>
моя цель-проанализировать элементы ценовых точек
<pricePoint>
<price>1358.2</price>
<quantity>20</quantity>
</pricePoint>
в словарь следующей структуры:
Dictionary<double, PriceLevel>
где цена должна быть двойным и PriceLevel класс
class PriceLevel
{
int bid;
int offer;
public PriceLevel(int b, int o)
{
bid = b;
offer = o;
}
}
в зависимости от элемента, в котором существует каждая ценовая точка (либо ставки, либо предложения), количество должно быть назначено соответственно, т. е. если ценовая точка существует в заявках, то количество должно быть назначено ставке и 0 предложению. Напротив, если точка цены существует в предложениях,то количество должно быть назначено предложению и 0-предложению.
Я надеюсь, что мое объяснение ясно, однако, если у вас есть какие-либо проблемы с его пониманием, пожалуйста, сделайте не стесняйтесь просить разъяснений в комментариях. Буду весьма признателен за помощь в решении этой проблемы.
+++++++++++++++++++++++++++++++++++++++++ Update:
StreamReader sr = новый StreamReader("....видео.xml");
XmlReader xmlReader = XmlReader.Create(sr);
while (xmlReader.Read())
{
if (xmlReader.HasValue)
{
OnXmlValue(this, new MessageEventArgs(true, xmlReader.Value));//saxContentHandler.Content(xmlReader.Value);
}
else
{
if (xmlReader.IsEmptyElement)
{
OnStartElement(this, new MessageEventArgs(false, xmlReader.Name));
OnEndElement(this, new MessageEventArgs(false, xmlReader.Name));
}
else if (xmlReader.IsStartElement())
{
OnStartElement(this, new MessageEventArgs(false, xmlReader.Name));
}
else
{
OnEndElement(this, new MessageEventArgs(false, xmlReader.Name));
}
}
}
но я изо всех сил пытаюсь связать имя элемента с его значением ... т. е. как я могу узнать, какую цену я сейчас читаю, и если это существует в предложениях или предложениях? Спасибо за помощь
4 ответов
когда вы используете интерфейс на основе событий, подобный тому, который представлен в вашем обновлении, вам нужно будет запомнить имя предыдущего события start element. Часто стоит держать стопку, чтобы следить за событиями. Я, вероятно, сделал бы что-то подобное следующему:
public class PriceLevel
{
private decimal? bid = null;
private decimal? offer = null;
public decimal? Bid {
get { return bid; }
set { bid = value; }
}
public decimal? Offer {
get { return offer; }
set { offer = value; }
}
}
public delegate void OnPriceChange(long instrumentId, Dictionary<decimal, PriceLevel> prices);
public class MainClass
{
private Stack<String> xmlStack = new Stack<String>();
private Dictionary<decimal, PriceLevel> prices = new Dictionary<decimal, PriceLevel>();
private bool isBids = false;
private decimal? currentPrice = null;
private long instrumentId;
private OnPriceChange _priceChangeCallback;
public void MainClass(OnPriceChange priceChangeCallback) {
this._priceChangeCallback = priceChangeCallback;
}
public void XmlStart(object source, MessageEventArgs args) {
xmlStack.Push(args.Value);
if (!isBids && "bids" == args.Value) {
isBids = true;
}
}
public void XmlEnd(object source, MessageEventArgs args) {
xmlStack.Pop();
if (isBids && "bids" == args.Value) {
isBids = false;
}
// Finished parsing the orderBookEvent
if ("orderBook" == args.Value) {
_priceChangeCallback(instrumentId, prices);
}
}
public void XmlContent(object source, MessageEventArgs args) {
switch (xmlStack.Peek()) {
case "instrumentId":
instrumentId = long.Parse(args.Value);
break;
case "price":
currentPrice = decimal.Parse(args.Value);
break;
case "quantity":
if (currentPrice != null) {
decimal quantity = decimal.Parse(args.Value);
if (prices.ContainsKey(currentPrice)) {
prices[currentPrice] = new PriceLevel();
}
PriceLevel priceLevel = prices[currentPrice];
if (isBids) {
priceLevel.Bid = quantity;
} else {
priceLevel.Offer = quantity;
}
}
break;
}
}
}
сначала нужно получить все предложения, и все предложения
XDocument xmlDoc = XDocument.Load("TestFile.xml");
var bids = (from b in xmlDoc.Descendants("bids")
select b).ToList();
var offers = (from o in xmlDoc.Descendants("offers")
select o).ToList();
затем вы просто повторяете предложения и предложения throgh и добавляете их в словарь... но как кто-то сказал раньше... возможно, у вас будет проблема, что уровень цен будет иметь как ставки, так и предложения, если они имеют одинаковую цену
чтобы повторить throgugh список, вы просто делаете это
foreach (XElement e in bids)
{
price = e.Element("price").Value;
quantity = e.Element("quantity").Value;
dictionary.add(price, new PriceLevel(quantity,null);
}
то же самое вы делаете предложение... но опять же.. .вероятно, вам нужно проверить, существует ли этот ключ...
во-первых, я считаю, что ваш метод ввода в словарь приведет к ошибке. Если не ошибаюсь, словарь не может иметь тот же ключ, поэтому, поскольку вы используете цену в качестве ключа, будет очень высока вероятность попадания в эту проблему.
Я не могу сказать о скорости, вы должны проверить. Но пока XDocument используется работает нормально для меня.
Используя XDocument, загрузите все xml-сообщение в эту переменную, например
XDocument doc = XDocument.Load(message);
С doc вы можете использовать Linq для их группировки в bid и ask.
Как только вы достигнете этого, не должно быть никаких проблем в представлении ваших данных, поскольку вы уже получили цену и разделили их на bid и ask
мне удалось сделать что-то вроде этого:
public void messageParser()
{
int i = 0;
bool readingBids = false;
bool readingOffers = false;
decimal price=0;
int qty = 0;
StreamReader sr = new StreamReader("..\..\sampleResponse.xml");
XmlReader xmlReader = XmlReader.Create(sr);
DateTime startTime = DateTime.Now;
while (xmlReader.Read())
{
#region reading bids
if (xmlReader.IsStartElement("bids"))
{
readingBids = true;
readingOffers = false;
}
if (xmlReader.NodeType == XmlNodeType.EndElement && xmlReader.Name == "bids")
{
readingBids = false;
readingOffers = false;
}
if (readingBids == true)
{
if (xmlReader.IsStartElement("price"))
price = xmlReader.ReadElementContentAsDecimal();
if (xmlReader.IsStartElement("quantity"))
{
qty = xmlReader.ReadElementContentAsInt();
OnPricePointReceived(this, new MessageEventArgs(price, qty, "bid"));
}
}
#endregion
#region reading offers
if (xmlReader.IsStartElement("offers"))
{
readingBids = false;
readingOffers = true;
}
if (xmlReader.NodeType == XmlNodeType.EndElement && xmlReader.Name == "offers")
{
readingBids = false;
readingOffers = false;
}
if (readingOffers == true)
{
if (xmlReader.IsStartElement("price"))
price = xmlReader.ReadElementContentAsDecimal();
if (xmlReader.IsStartElement("quantity"))
{
qty = xmlReader.ReadElementContentAsInt();
OnPricePointReceived(this, new MessageEventArgs(price, qty, "offer"));
}
}
#endregion
}
DateTime stopTime = DateTime.Now;
Console.WriteLine("time: {0}",stopTime - startTime);
Console.ReadKey();
}
}
это решение проблемы? У меня есть некоторые сомнения относительно этого фрагмента кода:
if (readingBids == true)
{
if (xmlReader.IsStartElement("price"))
price = xmlReader.ReadElementContentAsDecimal();
if (xmlReader.IsStartElement("quantity"))
{
qty = xmlReader.ReadElementContentAsInt();
OnPricePointReceived(this, new MessageEventArgs(price, qty, "bid"));
}
}
Я только запускаю onpricepointreceived событие, когда мне удалось прочитать цену и кол-во. Однако, есть вероятность, что не будет количество по данной цене (или нет). Как реализовать валиадацию, чтобы избежать ошибок на основе неполных сообщений?