1

タグの内容を無視して、XML 構造と比較するにはどうすればよいですか?

たとえば、次の応答があります。

<note>
   <to>Adam</to>
   <from>Eve</from>
</note>

と:

<note>
   <to>John</to>
   <from>Joan</from>
</note>

XML 構造は同じですが、次の点に注意してください。

<note>
   <from>Joan</from>
   <to>John</to>
</note>

タグの順序が異なるため、最初の例と同じではありません。

Java で XPath 構造を比較するにはどうすればよいですか?

4

2 に答える 2

1

XML 構造のハッシュ値を計算し、2 つのハッシュ値を比較できます。以下のように StAX を使用した簡単な例です。この例では、開始要素と終了要素のみが考慮されます。他のタイプを追加することもできます。比較するのではなく、ハッシュ値をコンソールに出力します。

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.security.MessageDigest;

import javax.xml.stream.XMLEventReader;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.events.XMLEvent;

public class ComputeXMLStructureHash {

    public static void main(String[] args)
    {
        try {
            FileInputStream in1 = new FileInputStream(new File("file1.xml"));
            FileInputStream in2 = new FileInputStream(new File("file2.xml"));
            FileInputStream in3 = new FileInputStream(new File("file3.xml"));

            System.out.println(digest(in1));
            System.out.println(digest(in2));
            System.out.println(digest(in3));                        

        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
    }

    public static String digest(InputStream in) {
        MessageDigest messageDigest = null;

        // StAX for XML parsing
        XMLInputFactory inputFactory = XMLInputFactory.newFactory();

        try {
            messageDigest = MessageDigest.getInstance("MD5");
            XMLEventReader eventReader = inputFactory.createXMLEventReader(in);

            // Iterate over the XML elements and update hash
            while (eventReader.hasNext()) {
                XMLEvent event = eventReader.nextEvent();

                if (event.isStartElement()) {
                    messageDigest.update(event.asStartElement().getName().toString().getBytes());                   
                } else if (event.isEndElement()) {
                    messageDigest.update(event.asEndElement().getName().toString().getBytes());
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

        StringBuffer result = new StringBuffer();
        byte[] digest = messageDigest.digest();
        for (byte b : digest) 
            result.append(Integer.toString((b & 0xff) + 0x100, 16).substring(1));

        return result.toString(); 
    }
}
于 2013-04-28T10:34:20.940 に答える