Wednesday, April 29, 2009

An severe performance impact by subselect in sql "from" clause (DB2)

Started working on the auto-pricing process performance test, but an very strange thing blocks me: a very simple query runs in very very bad performance, only 1 record inserted per second! The database is DB2 8.5 on zLinux server which runs in normal CPU utilization (35% only).

Originally, I thought there must be some problem in the database, so bad performance should not happen even the SQL hasn't been optimized. But the feedback from DBA is that he checked all things in DB, but can not find any abnormal indication. I really felt overwhelmed!

But today, I finally acknowledge that the bad performance is really caused by my query:

// less 1 record inserted per second, 30000 records insertion need 10 hours to complete (timeout...)
INSERT INTO WWPRT.CABLEPRODUCT_PRICETYPE_JOIN_CN (CABLEPRODUCTID, PRICETYPE, SLAVE) (
     SELECT cableproduct.ID, pricetypes.PRICETYPE, 'N'
     FROM WWPRT.CABLE_PRODUCT_JOIN_CN AS cableproduct,
     (
          SELECT DISTINCT usprice.PRICETYPE FROM WWPRT.PRICE_CN usprice
          JOIN WWPRT.CONF_PRICETYPE pt ON pt.ID = usprice.PRICETYPE
          WHERE usprice.CABLEID = 'USCABF3FVT01'
          AND pt.DOMAIN IN ( SELECT DISTINCT DOMAIN FROM WWPRT.CONF_GAP )

     ) AS pricetypes
     WHERE CABLEID = 'GEOANNF3FVT1'
     AND EXISTS (
         ......
     )
)


The arch-criminal is the subselect statement in "from" clause, the subselect query is performed for each record in WWPRT.CABLE_PRODUCT_JOIN_CN table. (awful!)

I changed it to a new way, using temporary table which query is executed only once.

// 30000 records insertion only costs 9 seconds to complete!
WITH US_PRICETYPES (PRICETYPE) AS (
    SELECT DISTINCT usprice.PRICETYPE FROM WWPRT.PRICE_CN usprice
    JOIN WWPRT.CONF_PRICETYPE pt ON pt.ID = usprice.PRICETYPE
    LEFT OUTER JOIN WWPRT.CONF_GAP gap ON pt.DOMAIN = gap.DOMAIN
    WHERE usprice.CABLEID = 'USCABF3FVT01'
    AND gap.DOMAIN IS NOT NULL
)
SELECT COUNT(*) FROM NEW TABLE
(INSERT INTO WWPRT.CABLEPRODUCT_PRICETYPE_JOIN_CN (CABLEPRODUCTID, PRICETYPE, SLAVE) (
     SELECT cableproduct.ID, pricetypes.PRICETYPE, 'N'
     FROM WWPRT.CABLE_PRODUCT_JOIN_CN AS cableproduct,
     US_PRICETYPES AS pricetypes
     WHERE CABLEID = 'GEOANNF3FVT1'
     AND EXISTS (
         ......
     )
))


The improvement is awesome! From 10 hours to 10 seconds! Although it happens in DB2, but I think this case should also exist in other types of database, but not verified.

--------------------------------------------------------------------------------------
Feedback from Daniel, thank you very much:

If you don't care the result of how many records inserted, I think you can use 'FETCH FIRST 1 ROWS ONLY' instead of count(*).

SELECT CABLEPRODUCTID FROM NEW TABLE
(INSERT INTO WWPRT.CABLEPRODUCT_PRICETYPE_JOIN_CN (CABLEPRODUCTID, PRICETYPE, SLAVE) (
     SELECT cableproduct.ID, pricetypes.PRICETYPE, 'N'
     FROM WWPRT.CABLE_PRODUCT_JOIN_CN AS cableproduct,
     US_PRICETYPES AS pricetypes
     WHERE CABLEID = 'GEOANNF3FVT1'
     AND EXISTS (
         ......
     )
)) FETCH FIRST 1 ROWS ONLY


Monday, April 20, 2009

Rock's first outing

Today, I play in the park first time with my daddy and mama.











Wednesday, April 15, 2009

An Derby Properties and Connection URL Example

Have been using Derby for a long time, which is a very good choice as memory database or embedded database, let me backup some useful Derby DB Properties here:

Developer can use a derby.properties file in the application classpath or javadb classpath to configure the database, these are some configuration examples in the file:

---------------------------- Derby derby.properties information ----------------------------
--### Derby System Home Directory
derby.system.home=D:/workspace/Boulevard/BoulevardDB

--### Derby Authentication Configuration and Turn On
derby.connection.requireAuthentication=true
derby.authentication.provider=BUILTIN
derby.database.propertiesOnly=true

--### User/Password Definition (System Level)
derby.user.alfred=alfred

--### User Authorization Definition (System Level)
derby.database.defaultAccessMode=fullAccess
derby.fullAccessUsers=cube

--### Some Others Configuration
derby.infolog.append=true
derby.storage.pageSize=8192
derby.storage.pageReservedSpace=60
derby.database.defaultConnectionMode
derby.locks.deadlockTimeout=20
derby.locks.waitTimeout=30
---------------------------- Derby derby.properties information ----------------------------


You have 2 ways to start the database:

1. Start embedded Derby from java application. The database is active from the first time you connect to it, and become inactive when your java application stop; and any other application outside of the same JVM can not access the database instance.

To connect to embedded Derby by URL:
------------------------------- Derby connection information -------------------------------
-- ### Use ij command to connect embeded database; if not exist, create it.
CONNECT 'jdbc:derby:data;create=true;user=alfred;password=alfred' AS CUBE;
-- ### Use ij command to connect embeded database; if not exist, don't create it, throw out error.
CONNECT 'jdbc:derby:data;user=alfred;password=alfred' AS CUBE;
------------------------------- Derby connection information -------------------------------


2. Start Derby network server. The network server instance can only be started by command "...\javadb\bin\startNetworkServer.bat", and it runs as a server which will not stop when your application stops; and other applications in any JVM can also access the database locally or remotely.

To connect to Derby network server:
------------------------------- Derby connection information -------------------------------
-- ### Use ij command to connect network server database, reading default repository;
-- ### If not exist, create it.
CONNECT 'jdbc:derby://localhost:1527/data;create=true;user=alfred;password=alfred' AS BOULEVARD;
-- ### Use ij command to connect network server database; reading default repository;
-- ### If not exist, don't create it, throw out error.
CONNECT 'jdbc:derby://localhost:1527/data;user=alfred;password=alfred' AS BOULEVARD;
-- ### Use ij command to connect network server database, reading specific repository;
-- ### If not exist, create it.
CONNECT 'jdbc:derby://localhost:1527/D:/workspace/Boulevard/BoulevardDB/data;create=true;user=alfred;password=alfred' AS BOULEVARD;
-- ### Use ij command to connect network server database, reading specific repository;
-- ### If not exist, don't create it, throw out error.
CONNECT 'jdbc:derby://localhost:1527/D:/workspace/Boulevard/BoulevardDB/data;user=alfred;password=alfred' AS BOULEVARD;
------------------------------- Derby connection information -------------------------------



Let's rock without Microsoft

Today, Microsoft formally announced to stop the major technical support for Windows XP, and the extension support will stop before 2014. Are you using Window XP or Vista? or Window 7? Do you feel the Windows is your only choice as the OS on your machine?

I am imaging a world without Microsoft in a short future. At least three major trends can lead to a new world without Microsoft:

1. Cloud computing, Virtual Infrastructure Access service, all these new technologies show an approach to compute and run applications without a complex local OS. More and more applications are deployed on server side, and they can be started and run remotely easily. In a short future, we only need a thin client or just only a browser in computer, cell phone, play station or something other machine like current TV station to run most of useful applications. The OS is not important any longer, while the key things are the applications or services in the network.

2. Many office applications become more and more mature, such as Google Apps including Gmail, Document and Calendar, IBM's Lotus Notes and Symphony, Sun's OpenOffice. Actually these software are powerful enough to totally replace the Microsoft Office now, and these software are more open and flexible through the open API library.

3. Even for the OS itself, the Windows is challenged by many competitors. Ubuntu is very nice and more and more powerful Linux system for personal usage. And these days, you know, most of eyes are focused on some new operation systems in mobile phone, like Google Android, Apple iPhone and Nokia Symbian. This is another new and more important war field than traditional computer desktop, as more and more applications and services are provided through network in future, but Microsoft is a loser in this field.

Less Microsoft, more wonderful world! Let's rock without Microsoft!

For now, I am also using Windows as it's required by company, but I believe when Microsoft stops all the support for Windows XP, I bet that the company will embrace Linux, for example Ubuntu, instead of paying money for next Windows version. These are some major applications for daily work in my computer:
  - Using Firefox as browser, instead of IE; It's faster, and there are many Firefox Add-ons to make the browser powerful, but IE can not;
  - Using Lotus Symphony and OpenOffice to create and edit document, spreadsheet or presentation materials, instead of using Microsoft Office. And these software can also easily create and edit Microsoft Office format document for customer, but Microsoft Office can not support OpenOffice format;
  - Using Lotus Notes (v8) to communicate with others. It integrated To Do List, Calendar, Email, Contacts, Instant Messging, Feed and many other useful functions together. Instead of using the Microsoft Outlook;
  - Using Lotus Sametime for online conference, instead of NetMeeting which function is too limitted and very slow;
  - Using Rational RAD/RSA for software design/development, and RTC for working planning, tracking, and for team communication and collaboration;
  - Using Twitter and Blue Community to build up my social network;
......

And the most important is all above applications can run well in Linux system, even better. If it was not required by company, I would like to use Ubuntu. I installed it which runs very well.

Monday, April 13, 2009

Baby's new clothes - PiYo PiYo

Today, my daddy dresses me in new clothes - my favorite brand, isn't it nice?

35 days ...











Thursday, April 9, 2009

Some Essential Twitter Tools

Looks like Twitter is very simple and easy tool, how to effectively use it? If you’re using Twitter for personal, corporate use, or to manage the brand of a client, you’ll need the right tools to find and engage the discussions.

These are some essential Twitter tools to improve the experience. Not all, but something I am using now, share with you:

1. Client: Using the web to fresh it manually is really boring. Twhirl is the most popular client, using adobe air technology. Leave it on your desktop and the client may be more accurate in seeing who’s replying to you, unlike the browser version. I highly recommand it. Another recommended client are the add-ons for Firefox if you are using it. You can find them in "Get Add-ons", the TwitterFox is a good choice.

2. Search: Use Tweetscan or Twitter Search to search tweets. You will feel the world is not too bad. If you use Twhirl as your client, you can find both of them have ben integrated into the search panel.

3. Aggregation: FriendFeed puts all of our RSS content onto one page, making it easy to see from one glance (rather than going to different web sites) and you can even reply from FriendFeed to different tools. It’s smarter to organize around people, rather than tools. I use it to feed my Google Blogger, Google Picasa Web Albums, WordPress, Facebook, and Twitter to same page, awesome!

4. Image Upload: Can we post image to Twitter? Yes, use Twitpic to store and share photos on Twitter. I didn't test the exact limit size, but too big file cause failure. I tried 3 ways to post image:
a. Upload photo in Twitpic web site, and copy the shorten link to Twitter;
b. Use the client Twhirl to upload image, the image will be stored in Twitpic, a link will be added to the Twitter status;
c. Use Firefox add-on "Power Twitter" to post photo, the result is same as Twhirl.
In the Twitter timeline home page, the image can only be displayed when Add-on "Power Twitter" is enabled, otherwise user have to click the shorten link to open picture in Twitpic site.

More to be added ...

Wednesday, April 8, 2009

My lovely baby 3 (One month old)

I am one month old today!







Some lessons learned from my TDD practice

Attended 2 days TDD course in last Dec. and started to use it in my project from Jan. These are some lessons learned from my TDD practices so far:

1. For any change into the code behavior, must think and write the test first (of course, this is the core principle of TDD);

2. In an ideal way, the TDD can totally replace low level design. But according to the result of a survey for TDD developers, most of new TDD developers found it was really hard to write effective test case directly without any low level design. As a trade off, developer can document a simplified low level design before coding test case. (The low level design should not be too detail, Simple Design is one of practices in eXtreme Programming, which is one part of Agile)

3. If developer write low level design first before coding test case, list out needed TDD test cases in the design document. Two benefits:
     a. Enforce developer to think about how to test it when he think about the design;
     b. Make sure the developer will not forget any test when he is coding.

4. Test case code and running result should also be reviewed during code review activity.

5. Unit test cases must be frequently executed to ensure all tests can be passed at the end of every day (it's better to adopt Continuous Integration if possible, CC can automatically run all test cases periodically and generate feedback);

6. Isolate the test data within one test class as much as possible, in order to make sure changes in one test will not impact others, and all tests can be re-run easily by another one; and the developer must make sure the test case running should not be impacted by local environment configuration or data.

7. Feel comfortable only when test case is written for function or business scenario, not for java class or method. You know, actually TDD is a business driven design/development approach.

More to be added...


Friday, April 3, 2009

My lovely baby 1 (The first day)

I was born today. These are pictures just after 10 hours:







My lovely baby 2 (The 10th day)

After 10 days, so beautiful:





Automatically feed Blogger to Twitter by twitterfeed.com

What a great service! I love it!

When I have both of the Blogger and Twitter, I want to use Twitter as a window to inform my friends about my blog update. But if I have to manually send tweet to Twitter, that's time waste. As what I like to do for development, it's perfect if I can make it automated. Although Twitter has its own API for developer, but looks like the ScribeFire still can not support posting to it. How?

Before my giving up finally, I found this service: http://twitterfeed.com/. My God! It's free, and looks like it can help to automatically feed tweet to Twitter from any type of blog, and generate a shorten link to your blog entry, so nice!

Very easy:
1. Login to twitterfeed.com using your OpenID (dont say you dont know what is OpenID if you really use blog frequently)

2. Create a new feed, input all required information: (note: the RSS Feed URL of your blog is the most important input, for Blogger you can see the RSS URL format here)


3. Enjoy it! Now all the blog new entries will be automatically fed to my Twitter every 30 minutes (of course with the characters number limit) and a shorten link.

My blog entry flow chart:



Use ScribeFire to post entry to Blogger

Yesterday, I successfully configure the ScribeFire to easily post entry to Google Blogger, let me share the steps using Blogger as an example: (actually I also use it for WordPress now)

Follow me:
Step 1. You should have a blog site in Blogger first (http://www.blogger.com)


Step 2. Download and install the Firefox add-on ScribeFire


After the installation, you should be able to find an icon at the right hand of the browser status bar like this:

Step 3. Click the ScribeFire icon to open the application. You can configure it to be open in new tab, or in new window, or in split-screen.

Step 4. Click the "Launch Account Wizard" button to set up your account for Blogger:
a. Enter your blog's URL like this: http://yourname.blogspot.com/, click Next;
b. The Account Wizard should be able to automatically get the proper configuration for your Blogger in next window, click Next; (note: this is the most important step for the configuration, if you attempt to configure ScribeFire for any other Blog service, you must find a proper blog system type and the blog service API URL values)
If the wizard can not detect the proper configuration for you, please input them by yourself:
>> select Blogger as blog system type;
>> input API URL like this format: http://www.blogger.com/feeds/yourblogId/posts/default, and you can find your blogId value in the browser address bar when you enter your Blogger setting page. To find the address looks like: http://www.blogger.com/blog-options-basic.g?blogID=20398450394850394. Have you see it?
c. Input your Blogger username and password for authentication purpose, click Next;
d. If everything is OK, the wizard should be able to retrieve all the blog sites under your Blogger account, select one of them for future posting, click Next;
e. Click Finish, a new account should be listed in the ScribeFire Blogs tab.

Step 5. Go! Select the account by radio button (note: clicking on the account name trigger to open your blog), write a new entry in ScribeFire, and click Publish to "your account title" button. For new entry, publish as a new post; for existing entry update, publish as an edit of the original entry.


Step 6. Wish you luck!:) Enjoy the great tool! You can use this offline editor tool to easily post your entry to your several blogs as you want.


Thursday, April 2, 2009

Hello world!

Hello world! I started a new blog in Google Blogger.

And successfully configure the ScribeFire to post entry to my blog, highly recommend this tool!

Congratulations! Alfred.