2011年10月6日 星期四

使用EMF來Serialize XML

EMF提供了將定義好的metamodel serialize成XMI(XML Interchange)格式的功能。使用這個功能,可以將Object 序列化為XMI文檔。程式如下:(此範例程式使用了由My.ecore這個EMF meta model產生出來的model code。此處可下載emftest.zip(內含My.ecore),下載後需再用EMF的工具,產生EMFTest這些package的Java code)

import EMFTest.EMFTestFactory;
  import EMFTest.EMFTestPackage;
  import EMFTest.vincent;
  import EMFTest.impl.EMFTestPackageImpl;

public class TestData {

    /**
     * @param args
     */
    public static void main(String[] args) {
        //要先Initial PackageImpl,才能使用EMF的功能
        EMFTestPackageImpl.init();
        //Serialize時,所需的參數儲存於 saveOptions
        Map<String, Object> saveOptions = new HashMap<String, Object>();
        //啟用OPTION_EXTENDED_META_DATA後,才能指定要讓某個attribute Serialize為Element,而不是Attributes
        saveOptions.put(XMLResource.OPTION_EXTENDED_META_DATA, Boolean.TRUE);
        //設定vincent.company及vincent.location這2個attribute要serialize成Element
        EMFTestPackage testPackage = EMFTestPackageImpl.eINSTANCE;
        EAttribute companyAttr = testPackage.getvincent_Company();
        EAttribute locationAttr = testPackage.getvincent_Location();
        XMLMapImpl map = new XMLMapImpl();
        XMLInfoImpl x = new XMLInfoImpl();
        x.setXMLRepresentation(XMLInfoImpl.ELEMENT);
        map.add(companyAttr, x);
        map.add(locationAttr, x);
        saveOptions.put(XMLResource.OPTION_XML_MAP, map); 
        saveOptions.put(XMLResource.OPTION_ENCODING, "UTF-8");      

     EMFTestFactory testFactory = EMFTestFactory.eINSTANCE;
        vincent testElement = testFactory.createvincent();
        testElement.setAge(32);
        testElement.setCompany("IBM");
        testElement.setLocation("Taipei");
        ResourceSet resourceSet = new ResourceSetImpl();
        // Register XML resource factory
        resourceSet.getResourceFactoryRegistry().getExtensionToFactoryMap().put("xml", new XMLResourceFactoryImpl());
        Resource resource = resourceSet.createResource(URI.createFileURI("Vincent.xml"));
        // add the root object to the resource
        resource.getContents().add(testElement);
        // serialize resource – you can specify also serialization
        // options which defined on org.eclipse.emf.ecore.xmi.XMIResource
        try {
            resource.save(saveOptions);
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

}

上面的程式中,透過啟用OPTION_EXTENDED_META_DATA,使得可以設定那些EMF model的attribute要產生成XML的element。另一種做法為,直接在Ecore model的diagram中,增加Extended Meta Data,把要設定的Attribute設為element。如下圖

注意:這個方法要work,一樣要在serialize XML的程式中,加入

saveOptions.put(XMLResource.OPTION_EXTENDED_META_DATA, Boolean.TRUE);

image

參考資料:

2011年10月3日 星期一

XSLT及Eclipse Tutorial

  1. 更新Eclipse,安裝 Helios下的Eclipse XSL Developer Tools
    image
  2. 安裝完後,Eclipse會要求重啟。重新啟動完Eclipse後,建立一個新的Java Project
  3. 在新建的Java Project下新增一個files的目錄
  4. 在files目錄下, 建立source.xml,內容如下
    <?xml version="1.0"?>
    <!-- This is a comment -->
    <people>
    <address type="personal">
    <name>Lars </name>
    <street> Test </street>
    <telephon number="0123" />
    </address>
    <address type="personal">
    <name>Joe </name>
    <street> Test2 </street>
    <telephon number="1234" />
    </address>
    <address type="business">
    <name>Jim</name>
    <street> Test3 </street>
    <telephon number="2345" />
    </address>
    </people>
  5. 在同一目錄下,建立一個transform2.xsl檔案,內容如下
  6. <?xml version="1.0"?>
    <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
        version="1.0">

        <xsl:output method="xml" />

        <!-- Copy everything -->
        <xsl:template match="*">
            <xsl:copy>
                <xsl:copy-of select="@*" />
                <xsl:apply-templates />
            </xsl:copy>
        </xsl:template>

        <!-- Do some adjustments for the address -->
        <xsl:template match="address">
            <xsl:element name="place-where-person-live">
                <xsl:apply-templates />
            </xsl:element>
        </xsl:template>

        <!-- Put the name in a <hubba> tag -->
        <xsl:template match="name">
            <xsl:element name="name">
                <hubba>
                    <xsl:apply-templates />
                </hubba>
            </xsl:element>
        </xsl:template>
    </xsl:stylesheet>

  7. 於Project explore中,對新建立的transform2.xsl按右鍵—>Run As—>XSL Transformation
    image
  8. Eclipse會跳出一個視窗,選取要轉換的XML檔案
    image 
  9. 轉換的結果會存在同目錄下的 source.out.xml檔案內

翻譯自 http://www.vogella.de/articles/XSLT/article.html

2011年10月2日 星期日

用SQL產生具有Referential Integrity的測試資料

題目:

假設有兩個表格parent_table及child_table,其結構如下

parent_table:

parent_id child_count
101 2
102 1
103 3

child_table:

parent_id child_id
101 11
101 12
102 13
103 14
103 15
103 16

其中,parent_id為parent_table的主鍵, child_id為child_table的主鍵 而child_count的數目表示該parent_id可以map到多少個child_id。試圖自動產生child_table的內容,使其能夠符合parent_table的child_count數目。也就是說,產生出來的child_table需要出現2次parent_id:101,1次parent_id:102以及3次parent_id:103,如同上圖所示child_table的內容。

解答:

  1. 遇到這類的情況,必需把parent_table中的rows展開,產生一個中介的表格。方法為把parent_table與一個sequence table做cartesian join。所需的SQL及產生的結果如下
    select parent_table.parent_id,parent_table.child_count, sequence_tab.id
    from parent_table join (select row_number() over()  as id from syscat.tables) as sequence_tab on sequence_tab.id between 1 and parent_table.child_count

    image
  2. 接著,Select 上述結果的parent_id,進行primay key編號,即為chile_table的內容。所需的SQL如下
    select temp_tab.parent_id, row_number() over()+10 as child_id from (select parent_table.parent_id,parent_table.child_count, sequence_tab.id
    from parent_table join (select row_number() over()  as id from syscat.tables) as sequence_tab on sequence_tab.id between 1 and parent_table.child_count)
    as temp_tab

    image
  3. 最後將上述SQL放在INSET INTO Child_table 之後,即告完成
    INSERT INTO CHILD_TABLE
    select temp_tab.parent_id, row_number() over()+10 as child_id from
    (select parent_table.parent_id,parent_table.child_count, sequence_tab.id
    from parent_table join (select row_number() over()  as id from syscat.tables) as sequence_tab on sequence_tab.id between 1 and parent_table.child_count)
    as temp_tab

參考來源:http://www.ibm.com/developerworks/data/library/techarticle/dm-0405kuznetsov/index.html
 

2011年9月29日 星期四

取消Java mixed components warning

在Java 1.6以後,Sun加入了新的安全機制,會檢視一個有signed過的程式是否摻雜unsigned的component。遇到這樣的情況,browser預設會跳出如下視窗,讓使用者確認。
這個情況下,若是按Yes,這個應用程式將會無法正常執行。所以如相信這個應用程式,必需選擇No。

 

若不想出現這個視窗,可以點選控制台-->Java,開啟Java主控台。然後選取進階-->安全-->混合程式碼。下面有四個選項,預設是選第一個 啟用 - 必要程顯示警告。若想直接執行類似的元件而不要跳出上面的視窗,可以如下圖所示,選取啟用-隱藏警告並在防護下執行 

image

 


資料來源:http://download.oracle.com/javase/6/docs/technotes/guides/jweb/mixed_code.html#manifest

2011年9月17日 星期六

如何安裝EMF

依照http://www.vogella.de/articles/EclipseEMF/article.html#emfinstallation 連結的介紹,只要裝Ecore Tools SDK及EMF-Eclipse Modeling Framework SDK。然而在安裝時卻遇到缺乏core-platform的訊息。經Google了一下,發現要先裝http://www.eclipse.org/eclipse/platform-core/downloads.php#updates這個東西後,就可以安裝所要的EMF元件了

2011年7月13日 星期三

Web Security考量

Robust Defenses for Corss-Site Request Forgery這篇paper主要是介紹Cross-Site-Request-Forgery這個攻擊的手法以及防治方法。不過同時也提到了其它Web應用程式應該注意的security issue。如下列

  • XSS(Cross-Site-Scripting)
  • DNS Rebinding
  • Certificate Errors
  • Phishing
  • User Tracking

2011年6月9日 星期四

客製化Sun的JAXB implemtation--XJC

使用Sun的XJC建立Java code時,預設遇xml schema中有fixed attribute時,不會自動把它轉成Java的constant,要達到這個目的,必需客製化XJC的 binding方式。作法如下(XJC提供inline及external的customization,其中inline的客製化是直接把客製化設定寫在要進行轉換的xml schema中;而exetrnal的customization則是另外寫在一個外部檔案,這裡使用external customization為例子)

  1. 撰寫customization檔案(xjc根目錄下的 samples/external-customize 子目錄有更詳細的例子)
       <jxb:bindings version="1.0"
                xmlns:jxb="http://java.sun.com/xml/ns/jaxb"
                xmlns:xsd="http://www.w3.org/2001/XMLSchema">
      <jxb:bindings schemaLocation="ExercisePrescription - reference - 20110530 hugang.xsd" node="/*">
      <jxb:globalBindings
        fixedAttributeAsConstantProperty="true"/>
      </jxb:bindings>
    </jxb:bindings>
  2. 執行轉換指令
    xjc   xml_schema_to_convert     -b   customization_file_name
  3. 如此一來,xjc在轉換java code時,會把fixed value轉成java constant

2011年1月12日 星期三

Database in Depth筆記-2(Predicate)

關係與關係變數

關係變數(用R表示)跟程式語言的變數一樣,是可以放置某種型別(這裡是關係)的容器。而關係(用r表示),則是指在某個時點,實際存在的n個Tuples(n>=0)。以C.J. Date的觀點,所謂Relation Model的Insert、Delete、Update動作,都是透過關係變數R,取出所要的關係後,對其Tuple進行一些處理後,再重新指向回關係變數。

Predicate

Predicate的觀念是由集合論而來的。在指涉某個集合時,常常無法透過列出集合中包含的所有元素的方式,來表示該集合。因此改以使用用該集合中每個元素的共同因子作為參數來描述該集合。對應至Relational Model,組成關係變數Heading中的屬性集合,即可組成描述該關係變數的Predicate。例如某個關係變數 R的Heading為 {Customer_ID,Customer_Name,Customer_Address},其意含可以解釋為這樣的Predicate:  編號為Customer_ID的客戶,其名稱為 Customer_Name,且其住址為Customer_Address

將Assign到該關係變數的關係r中的每一Tuple的值,分別套上這個Predicate的結果後,就形成一個Proposition(命題)。此命題應該都為True,因為若結果為False,則此關係根本不存在,也就不會被Assign到關係變數中。所以C.J. Date提到一個重點:在EF Codd原始的Relational Model中,資料庫不是資料的集合,而是Proposition(命題),則事實的集合。系統執行關聯式運算的過程,就是由既有事實推論出新的事實

Database in Depth筆記-1 (Relation及Tuple定義)

Database in Depth: Relational Theory for Practitioners

這本書買了好幾年了,因為讀起來有點艱澀,一直沒有把它翻完(作者太龜毛了,一直在辨證Relational Model不能有NULL、SQL不是Relational Model之類對我來說沒有意義的論點),不過看在它是和EF Codd一起work on Relational Model的大師,還是應該把它翻過一遍,趁這次再翻它的機會,做一些筆記。

Tuple定義:

相對應於Table的Row。在原始的Relational Model中,是用Tuple這個概念來代表一組資料,而不是大家所熟知的Row。以下是它的定義:

假設T1,T2…,Tn(n>=0)是型別的名稱,而A1,A2…,An是代表n個不同的屬性名稱,若將每個屬性名稱Ai對應到相對的Ti型別,則如此組合成的每一個屬性名稱:型別 (Ai:Ti)就是一個屬性。若將每個屬性都賦予符合型別Ti的值Vi,則形成的屬性:值,就稱為一個Component。如此一來,n個屬性:值組成的n個Component的集合就稱為一個Tuple。而這n個Component中的屬性集合就形成讓Tuple的Heading,以集合{H}表示

Relation定義:

相當於RMDBS的Table。初學Relational Model的人可能會把關聯式資料庫的關聯想成是不同Table之間可以建立Join的關係。然而,在原始的Relational Model,Relation是較為抽象的定義。它是基於集合論而來的概念。假設Order Table有CustomerID及ProductID這兩個屬性,把CustomerID和ProductID這兩個屬性放到同一個實體來看,就是Relation的意義了。以下是CJ Date在此書中所下的定義

假設{H}是Tuple的Heading,而且t1,t2…,tm (m>=0)是帶Heading {H}的不同Tuple,則該Heading {H}與該不同Tuple的集合(t1,t2…,tm)的組合叫做Relation。此Relation就是帶有屬性A1,A2…An的關係,換句話說,該Relation的Body就是Tuple(t1,t2…,tm)的集合

1NF(一階正規化)

有了Relation及Tuple的觀念後,一階正規化就可以簡單用一句話來描述:每個Tuple的每一個屬性只能帶有單一值

2011年1月6日 星期四

使用JAXB Parse XML

轉載自:http://www.javaworld.com.tw/jute/post/view?bid=19&id=5482&sty=1&tpg=1&age=-1

相信正常一點的程式設計人員都覺得XML很煩
如果你是技術狂,那就算了,當我沒說 :p
通常我們只是想要讀一個能跨平台的文件
幹嘛要去了解一堆有的沒的定義及如何parse文件的東東
為什麼不讓這些煩人的事情包裝起來
我們只要關心我讀進來的資料該如何處理
處理完再把它順利的寫回就好哩
這個念頭相信在大家的心中都出現過
所以恭喜大家,前人已經幫我們種好樹了
大家趕快到樹下乘涼吧
JAXB就是我們要的東東
Java Architecture for XML Binding (JAXB)
簡單的講就是提供了一個自動在xml文件與java物件mapping的機制
所以我們不用去了解啥米弟踢弟、死雞碼該如何定義
也不用學習如何去吹薩克斯風或挖洞給自已跳
首先要下載Java Web Services Developer Pack
其中便包含有JAXB的實作套件
接下來要設定相關的環境變數
範例如下:
set JAVA_HOME=your_jdk_home_path
set JWSDP_HOME=your_jwsdp_home_path
set JAXB_HOME=%JWSDP_HOME%\jaxb-1.0
set JAXB_LIBS=%JAXB_HOME%\lib
set JAXP_LIBS=%JWSDP_HOME%\jaxp-1.2.2\lib
set JWSDP_LIBS=%JWSDP_HOME%\jwsdp-shared\lib
set PATH=%JAXB_HOME%\bin;%JWSDP_HOME%\jwsdpshared\bin;%PATH%
以下請自行連結為一行
set CLASSPATH=%JAXB_LIBS%\jaxb-api.jar;
%JAXB_LIBS%\jaxb-ri.jar;%JAXB_LIBS%\jaxb-xjc.jar;
%JAXB_LIBS%\jaxb-libs.jar;
%JAXP_LIBS%\jaxb-api.jar;
%JAXP_LIBS%\endorsed\xercesImpl.jar;
%JAXP_LIBS%\endorsed\xalan.jar;
%JAXP_LIBS%\endorsed\sax.jar;
%JAXP_LIBS%\endorsed\dom.jar;
%JWSDP_LIBS%\jax-qname.jar;
%JWSDP_LIBS%\namespace.jar;.;
接下來我們一步步來看該如何使用JAXB
首先要先綁架xml schema
透過JAXB所提供的工具xjc
可產生一組用以描述xml schema的Java介面及實作類別
xjc指令格式如下:
xjc -p <generated_package> -d <target_directory> xml_schema
例如:
xjc -p test.jaxb -d work_area books.xsd
接著要編譯所產生的介面及實作類別
javac test/jaxb/*.java test/jaxb/impl/*.java
再來就可試著透過Unmarshal的方式來取得xml文件裡的資料
可以在測試程式中撰寫類似以下程式碼
JAXBContext jc = JAXBContext.newInstance("test.jaxb");
Unmarshaller unmarshaller = jc.createUnmarshaller();
Collection collection =
(Collection)unmarshaller.unmarshal(new File( "books.xml"));
CollectionType.BooksType booksType = collection.getBooks();
List bookList = booksType.getBook();
然後就可以取出xml文件裡的資料
for( int i = 0; i < bookList.size();i++ ) {
test.jaxb.BookType book =(test.jaxb.BookType) bookList.get(i);
System.out.println("Item id: " + book.getItemId());
System.out.println("Book Name: " + book.getName().trim());
}
另外也可利用
unmarshaller.setValidating(true);
來指定是否要在Unmarshal的同時來驗證此份xml文件格式正確與否
再來就可試著透過Marshal的方式來將資料寫到所指定的xml文件裡
可以在測試程式中撰寫類似以下程式碼
JAXBContext jaxbContext = JAXBContext.newInstance("test.jaxb");
Marshaller marshaller = jaxbContext.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT ,
new Boolean(true));
ObjectFactory objFactory = new ObjectFactory();
Collection collection= (Collection) objFactory.createCollection();
Collection.BooksType booksType = objFactory.
createCollectionTypeBooksType();
List bookList = booksType.getBook();
BookType book = objFactory.createBookType();
book.setItemId("777");
book.setName("JAXB簡介");
然後就可以將資料寫入我們所指定的xml文件裡
bookList.add(book);
collection.setBooks(booksType);
marshaller.marshal(collection,
new FileOutputStream("jaxb_test.xml"));
另外也可利用Validator類別來驗證此份xml文件格式正確與否
範例如下:
Validator validator = jaxbContext.createValidator();
validator.validate(collection));
若您還想知道更多關於JAXB使用上的細節,請上官方網站喔 ^_^
http://java.sun.com/xml/jaxb/
*********************無版權宣告*********************
版權沒有,歡迎轉載
轉載時請保留作者姓名及此份宣告
本文是為了方便新手入門學習
歡迎大家給小弟指正或補充更多詳盡的資料
my email: iampoya@pchome.com.tw
本文同步刊於
http://www.javaworld.com.tw
http://www.midicat.com/poya/ezjava