Natty’s blog

Stay hungry. Stay foolish. — Steve Jobs

Archive for the ‘PHP’ Category

[PHP] การเพิ่ม Performance ให้กับ PHP+MySQL ด้วย xDebug,APC,memcache

Posted by natty on June 10, 2009

เขียนโปรแกรมเล็กๆ ไม่ค่อยมีปัญหาอะไร แต่เมื่อไหร่ก็ตาม ที่งานของเรา ต้องทำงานกับข้อมูลเยอะๆ และ Application ขนาดใหญ่ มีหลาย module โดยเฉพาะ หากใช้ object-oriented ด้วยแล้ว แม้จะมีประโยชน์ในด้านของการออกแบบก็จริง แต่ด้านของ performance ก็มีผลกระทบในแง่ร้ายเช่นกัน ดังนั้น จึงควรที่จะเรียนรู้เกี่ยวกับการ config ค่าต่าง ๆ ทั้งในส่วนของ PHP และ MySQL ค่ะ

blog นี้จะไม่กล่าวถึงการ install ในขั้นต้นนะคะ โดยเป็น Apache Server บน linux ค่ะ

MySQL

เริ่มกันก่อนในส่วน MySQL

ปกติ หากเราลง MySQL แบบ default เราจะใช้ my.ini แบบที่กำหนดมาเป็น default แต่หากเราต้องการจะ modify ให้ใช้งานและรองรับข้อมูลจำนวนมาก ให้เราลองสังเกตใน folder MySQL ของเรา จะมีไฟล์ config อีกมากมายเลย เช่น my-huge.ini, my-innodb-heavy-4G.ini, my-large.ini, my-medium.ini, my-small.ini, my-template.ini ซึ่งใน file พิเศษๆ เหล่านี้ จะมีการ set ค่าเหล่านี้ ยกตัวอย่างจาก my-huge.ini

key_buffer = 384M
max_allowed_packet = 1M
table_cache = 512
sort_buffer_size = 2M
read_buffer_size = 2M
read_rnd_buffer_size = 8M
myisam_sort_buffer_size = 64M
thread_cache_size = 8
query_cache_size = 32M

ซึ่งแต่ละ File ก็จะมีการตั้งค่าแตกต่างกันไป วิธีการเอาไปใช้งาน ก็แค่เปลี่ยนชื่อไฟล์ที่เหมาะกับเราเป็นชื่อ my.ini (back up อันเดิมไว้ด้วยก็ดีค่ะ เผื่อมีอะไรผิดพลาด)

xDebug

xDebug คือ PHP profiler และช่วยจัด format PHP ให้อ่านง่ายๆ ได้ด้วย

ในส่วนของ profiler จะมีการเก็บ log ของ server เกี่ยวกับเวลาในการรันของแต่ละ class แต่ละ method กันเลยทีเดียว แต่ไฟล์ที่ได้ออกมาจะเป็น text file โง่ๆ ที่อ่านยากมากๆ ต้องลงพวก GUI เป็น WinCacheGrind สำหรับ Windows และ KCacheGrind สำหรับ Linux

ก่อนอื่น ต้องลง xDebug ไว้ที่ server ก่อน

#yum install php-pear
#pecl install xdebug
#
vim /etc/php.ini   <– เพิ่มบรรทัด extension = xdebug.so ลงไป และบรรทัดต่อไปนี้

[xdebug]
xdebug.remote_enable=On
xdebug.remote_autostart=On
xdebug.remote_handler=dbgp
xdebug.remote_host=zero.haadthip.com
xdebug.remote_port=9000
xdebug.remote_mode=req
xdebug.profiler_enable=1
xdebug.profiler_output_dir=”/tmp/” <– log เก็บที่นี่
xdebug.profiler_append=On
xdebug.profiler_enable_trigger=On
xdebug.profiler_output_name=%s

ตรวจสอบให้แน่ใจว่ามี file xdebug.so อยู่ใน /usr/lib/php/modules

และ restart mysql + Apache ด้วยค่ะ ลอง echo phpinfo หากสำเร็จแล้วจะได้ผลดังนี้

จากนั้น ก็ลง WinCacheGrind ซึ่งเอา log file ได้จาก path ที่เสร็จไว้ใน php.ini (xdebug.profiler_output_dir) โหลดไฟล์เข้าไป ก็จะเอาไปแสดงในโปรแกรม ดังรูป

คราวนี้เราก็วัด performance ก่อน และ หลัง ได้แล้วค่ะ

APC

คือ Alternative PHP cache เอาไว้ cache code โดยจะ cache มันไว้ ณ compile state ทำให้ overhead ลดลง เอาเป็นว่าไม่เกี่ยวข้องกับ database นะคะ

ดูกระบวนการทำงานของ APC กันสักหน่อย

ขั้นตอนการ install

#yum install httpd-devel

(php-devel ลงเพราะมันต้องการ apxs หากเราไม่มี apxs ก่อนนจะลง APC มันจะ error ว่า

Sorry, I was not able to successfully run APXS.  Possible reasons:

1.  Perl is not installed;
2.  Apache was not compiled with DSO support (–enable-module=so);
3.  ‘apxs’ is not in your path.  Try to use –with-apxs=/path/to/apxs
The output of apxs follows
/tmp/tmpe1LwOh/APC-3.0.19/configure: line 3232: apxs: command not found
configure: error: Aborting
ERROR: `/tmp/tmpe1LwOh/APC-3.0.19/configure –with-apxs’ failed

)
#yum install apc
#pecl install apc
vim /etc/php.ini  เพิ่ม extension=apc.so และเพิ่มบรรทัดต่อไปนี้

[APC]
apc.enabled = 1
apc.shm_segments = 1
apc.shm_size = 64
apc.max_file_size = 10M
apc.stat=1

ทำการ restart Apache และลองดู phpinfo.php

และควรจะ download package ของ APC มาด้วย เพราะจะได้ดูทางหน้าเว็บได้ โดยแตกตัว folder เอาไปวางใน root directory นะคะ จะได้หน้าตาเว็บแบบนี้เอาไว้ดู resource เกี่ยวกับ APC ของ server

Memcache

หลังจากที่มีการ cache หน้า page กันแล้ว ตรงส่วน database ก็มีความสำคัญเช่นกัน เพราะ connection database เป็นอะไรอีกอย่างที่ทำให้เกิดความช้า หากเรา query บ่อย ๆ ประกอบกับมี query ที่ซับซ้อนและมีจำนวนข้อมูลที่เป็นผลลัพธ์มาก ใช้ memcache นี่แหละค่ะ เป็นตัวช่วย

วิธีการติดตั้ง

#wget http://monkey.org/~provos/libevent-1.3e.tar.gz
#tar zxpfv libevent*
#cd libevent*
#./configure
#make install

#wget http://www.danga.com/memcached/dist/memcached-1.2.4.tar.gz
#tar zxpfv memcached*
#cd memcached*
#./configure
#make install

#adduser memcached

หากเกิดปัญหาเกี่ยวกับการหา file libevent ไม่เจอ แก้ไขดังนี้

ค่อยมาเขียน

จากนั้น ต้องลง php-memcache ด้วย

#yum install php-pecl-memcache.i386
#vi /etc/php.ini เพิ่มบรรทัดดังต่อไปนี้

[memcache]
memcache.allow_failover=1
memcache.max_failover_attempts=20
memcache.chunk_size=8192
memcache.default_port=11211

ทำการ restart server แล้วตรวจสอบ phpinfo

การใช้งาน memcache ต้อง run ให้มันทำงานอยู่ด้วย เหมือนกับ run server นั่นแหละ โดยใช้คำสั่ง
memcached -vv -u memcached

ถ้าได้ผลแบบนี้แสดงว่า memcache server ได้ run ขึ้นมาแล้ว พร้อมใช้งานค่ะ

ลองทดสอบว่าสามารถ cache ได้แล้วจริงหรือไม่ โดย copy code ต่อไปนี้ รอไปรันดู

<?php$memcache

= new Memcache;
$memcache->connect('localhost', 11211) or die ("Could not connect"); $version = $memcache->getVersion();
echo
"Server's version: ".$version."<br/>\n"; $tmp_object = new stdClass;
$tmp_object->str_attr = 'test';
$tmp_object->int_attr = 123; $memcache->set('key', $tmp_object, false, 10) or die ("Failed to save data at the server");
echo
"Store data in the cache (data will expire in 10 seconds)<br/>\n"; $get_result = $memcache->get('key');
echo
"Data from the cache:<br/>\n"; var_dump($get_result); ?>

นั่นแสดงว่า วิธีการใช้ memcache คือ เราต้อง new memcache ให้กับ query ที่เราต้องการจะ cache หากคุณมี code อยู่แล้ว และต้องการจะ cache มีทางเดียวที่ทำได้ คือ แก้ code ค่ะ ^O^ งานเข้าล่ะสิ

และจริงๆ memcache เอง ก็สามารถดูประสิทธิภาพทางหน้าเว็บได้เหมือน APC เลย ลอง download code จาก ลิงค์นี้ค่ะ http://livebookmark.net/journal/2008/05/21/memcachephp-stats-like-apcphp/

เรื่องการ config แต่ละตัว อาจจะไม่ละเอียดมาก เรียกได้ว่าพอใช้งานได้แล้วกันค่ะ หากใครอยากจะ modify กันมากกว่านี้ ก็ตามไปอ่าน official website กันได้ค่ะ

หากมีข้อสงสัย ติชม เพิ่มเติม ก็ comment เอาไว้ได้ หากเอาไปอ้างอิง อย่าลืม credit ให้ด้วยนะคะ

ลิงค์น่าอ่านเพิ่มเติม

http://www.thaicyberpoint.com/ford/blog/id/170/
http://www.pantz.org/software/php/phptuningtips.html
http://dev.mysql.com/doc/refman/5.1/en/query-cache.html

http://www.pantz.org/software/php/phptuningtips.html

Posted in PHP | 11 Comments »

[PHP] สิ่งหนึ่งที่ทำให้เกิด Call to a member function on a non-object

Posted by natty on November 19, 2008

วันนี้ แก้ error โง่ๆ ไปพักหนึ่ง ด้วยความมึน ตอน run PHP แล้วเกิด คือ พยายามจะ new object แต่ว่าไม่สามารถจะ new ได้ และหลังจาก new ก็ไปใช้ method ของมัน ทำให้มันบอกว่า ใช้ function จากตัวที่ไม่เป็น object ไม่ได้ อย่างเช่น


$test = new AR(new Ds());

$test->getDS();

มันจะเรียก getDS(); ไม่ได้ เพราะ $test ยังเป็น null

มาเล่น PhotoHunt กัน ให้ดูว่า class ของ Ds ทำอะไร


var $dsStrategy;
public function _construct (iDSStrategy $dsStrategy){
$this->dsStrategy = $dsStrategy;
}
public  function getValForCreditFromCore(){
$this->dsStrategy->getValForCreditFromCore();
} 

นี่แหละ ทำให้เกิด Call to a member function on a non-object งงเลยใช่ไหมว่าทำไม

เหตุผลสำหรับ code นี้ เพราะที่ใส่ underscore หน้า constructor ไปอันเดียว – -”

ที่ถูกต้องเป็น

public function __construct (iDSStrategy $dsStrategy)

Posted in PHP | Leave a Comment »

[PHP] Simple Test

Posted by natty on October 22, 2008

เนื่องจากมีคนเรียกร้องอยากใช้งาน SimpleTest ใน PHP ขึ้นมาอย่างกระทันหัน ไม่ได้ตั้งตัว ก็เลยเอามาวางให้ที่นี่นะคะ

ก่อนอื่น ไป ดาวน์โหลดมาก่อนนะคะ

เมื่อได้มาแล้ว ก็แตกไฟล์ แล้วเอา folder ชื่อ simpletest ก็เอาไปวางใน apache ของตัวเอง สำหรับเก๋ เก๋จะวางไว้นอกสุด ก็คือใน htdocs เลย ใครจะมาเรียกใช้เมื่อไหร่ก็ได้

เก๋จะให้ตัวอย่างสำหรับการทำ SimpleTest ของ blog ตอนหนึ่ง คือ [PHP] Class สำหรับ convert DBF to MySQL

โดยเก๋จะสร้าง folder ชื่อ unittest และมีไฟล์ชื่อ TestDBFMySQL.php อยู่ การทำหน้า test เราต้องใส่ require_once(‘../../simpletest/autorun.php’); เพื่อ include ไปยัง lib ของ simpletest และก็ include ไฟล์ที่เราจะ test ด้วย ปกติจะ new เป็น test case และ test suite ใน class อื่นได้ แต่อันนี้ไม่ได้ new ไว้ไฟล์อื่น แบบว่าเรียกไฟล์นี้โดยตรงเลย

การ test ก็จะ assert ในหลายกรณี อันนี้ก็แล้วแต่เราว่าจะทำยังงัย อ่านได้ที่นี่

โดยหน้าตาของไฟล์ตัวอย่าง เป็นดังนี้


<?php
//DBFMySQL/unittest/TestDBFMySQL.php

require_once('../../simpletest/autorun.php');
require_once('../DBFMySQL.php');

class TestDBFMySQL extends UnitTestCase {
        //public $fromDBF='D:\PO_McDonald\INVORHD0.DBF';
        public static $fromDBF='D:\PO_McDonald\INVORHD0.DBF';
        public static $toMySQL='INVORHD0';
        public static $myDB='db_mcdonald';
        public static $myUser='root';
        public static $myPass='root';
        public static $myHost='localhost';

    function setUp() {

    }

    function tearDown() {
        //@unlink('/temp/test.log');
    }

    function testConvertDBFToMySQL(){
        $x = DBFMySQL::convertDBFToMySQL(self::$fromDBF, self::$toMySQL, self::$myDB, self::$myUser, self::$myPass, self::$myHost);
        $this->assertTrue($x);
    }
    function testReadDBFToArray(){
        $x = DBFMySQL::readDBFToArray(self::$fromDBF);
        $this->assertNotNull($x);
    }
    function testPersistToMySQL(){
        $arrTable = DBFMySQL::readDBFToArray(self::$fromDBF);
        $x = DBFMySQL::persistToMySQL($arrTable, self::$toMySQL, self::$myDB, self::$myUser, self::$myPass, self::$myHost);
        $this->assertTrue($x);
    }
}
?>

ลองเข้าไปที่ http://localhost/DBFMySQL/unittest/TestDBFMySQL.php นะคะ (เครื่องเก๋นะคะ ยกตัวอย่างให้ดู)

หากว่า Test ผ่าน จะได้ผลดังนี้ค่ะ

Image hosting by UpPicth.com

ถ้า fail

Image hosting by UpPicth.com

ถ้า connect db ไม่ผ่าน
Image hosting by UpPicth.com

สงสัยก็ comment ถามมาได้ค่ะ จะตอบให้

ดาวน์โหลดตัวอย่าง code ได้ที่นี่ค่ะ ก่อนอื่นควรไปอ่าน [PHP] Class สำหรับ convert DBF to MySQL ด้วยนะคะ เผื่อรันไม่ผ่าน

Posted in PHP | 1 Comment »

[Resin] PHP in Java: Natty & Juacompe discussion

Posted by natty on September 24, 2008

เนื่องมาจาก natty ที่พอจะมีความรู้เรื่อง resin เท่าหางอึ่ง + PHP อยู่บ้าง บวกกับ juacompe ที่อัดแน่นไปด้วยความรู้เกี่ยวกับ server ที่ใช้ run java ทำให้วันนี้ก็เกิดความเข้าใจมากขึ้นกับ PHP in Java ใน resin

หากเราต้องการทำให้ java ของเราถูกเรียกโดย PHP ได้ ต้องทำให้ class นั้น extends AbstractQuercusModule และมี method ที่รับ parameter env ซึ่งเป็น quercus environment resources หรือไม่ก็ต้องสร้าง class ไว้แล้วให้มาสร้าง instant อยู่ใน class นี้

code ด้านล่างเป็นเป็น code ที่สร้าง method ขึ้นมาเพื่อให้ถูกเรียกจาก PHP ได้ ชื่อ method ว่า hello_test โดยจะเห็นว่ามันจะรับตัวแปร env ด้วย โดย class นี้อยู่ใน package example

 package example;

import com.caucho.quercus.env.Env;
import com.caucho.quercus.module.AbstractQuercusModule;

public class HelloModule extends AbstractQuercusModule {
   /*
   ** Notice the careful use of the naming
   ** convention hello_test.  This is done
   ** in order to prevent name collisions
   ** among different libraries.
   **
   ** @param env provides access to Quercus environment resources
   ** @param name
   */
   public String hello_test(Env env, String name)
   {
     env.println("inside HelloModule  hello_test()");
     return "Hello, " + name;
   }
} 

สิ่งที่สำคัญคือ ต้องสร้างไฟล์ชื่อ com.caucho.quercus.QuercusModule ไว้ที่ classes/META-INF/services โดยใส่ชื่อ package.class เอาไว้

 example.HelloModule 

ตอนหน้าค่อยต่ออีก….นี่เป็นเพียง study ยังไม่ได้ทดสอบเลยว่าทำต่างจากนี้แล้วจะพังป่าว

Posted in PHP, configuration, java | Leave a Comment »

[PHP] Class สำหรับ convert DBF to MySQL

Posted by natty on September 22, 2008

เนื่องจากมีโปรเจคหนึ่งที่ต้องทำส่วนหนึ่ง คือส่วนของการ convert DBF to MySQL เลยเอามาฝากกัน ที่ทำเพราะว่า ระบบนี้จะเป็นต้องอ่าน DBF file ใหม่ๆ ทุกวันเพื่อนำมาใช้ในระบบ แต่ developer อยากใช้กับ MySQL database ดังนั้นเก๋จึงทำ class ที่อ่าน dbf file มาลง mysql ให้เค้า โดยทำเป็นตารางที่หน้าตาเหมือนกันใน MySQL ให้

ระบบนี้มี table ชื่อเดิม แต่ต้องอัพเดทข้อมูลตาม dbf file ทุกวัน ดังนั้นเก๋จึงไม่ได้ทำ function ที่ให้สร้าง MySQL table หน้าตาเหมือนๆ กัน ทำแค่ให้ insert ข้อมูลลงไปใน MySQL ได้ นั่นหมายความว่า หากต้องการนำ class นี้ไปใช้ ต้องสร้าง table ที่หน้าตาเหมือนกับ dbf file ไว้ก่อน แล้วค่อยเอา class นี้ไปช่วยในการ insert ข้อมูลให้

ใจจริง อยากทำ function ที่ให้ create table เหมือนกัน แต่ติดที่ว่า วิธีการอ่าน dbf file คือเก๋ใช้ dbase function ซึ่งเป็นการ read wirte dbf file แต่ไม่ใช่การ query ด้วยคำสั่ง SQL ดังนั้นจึงคิดวิธีการ get property ของ field ไม่ออก ว่าต้องทำยังงัย ก็เลยคิดว่า หากวันนึงอยากใช้ create table function ขึ้นมาจริงๆ คงต้องใช้ ODBC เข้าช่วย เพราะคำสั่ง SQL สามารถ get property ได้

มีไฟล์ที่เกี่ยวข้อง 3 ไฟล์ คือ DBFMySQL.php, iDBFMySQL.php และ test.php

เก๋กำหนดให้มี interface เอาไว้ จะเห็นว่า createTable โดน comment ไว้ แสดงว่ามันจะ coming soon ฮ่าๆ


<?php

/*created by Nattanicha Rittammanart
* Created date: Sep 22, 2008 */

interface iDBFMySQL{
    /*
        fromDBF: dbf with the right path
        toMySQL: define table name in MySQL
        myDB: MySQL database to put MySQL table
        myUser: MySQL username
        myPass: MySQL password
        myHost: MySQL Host

    */

    /**/
    public static function convertDBFToMySQL($fromDBF, $toMySQL, $myDB, $myUser, $myPass, $myHost);
    public static function readDBFToArray($fromDBF);
    public static function persistToMySQL($arrTable, $toMySQL, $myDB, $myUser, $myPass, $myHost);
    //public static function createMySQLTable($arrTable, $toMySQL, $myDB, $myUser, $myPass, $myHost);
}

function convertDBFToMySQL เป็นฟังก์ชั่นที่ให้ผู้ใช้เรียกใช้เพื่อ convert dbf to mysql โดยตรง โดยฟังก์ชั่นอื่น เป็นฟังก์ชั่นที่ใช้งานร่วมกับมันเท่านั้น ก็คือใส่ parameters ให้ครบ มันก็จะให้สิ่งที่ต้องการออกมา ฟังก์ชั่นนี้ เรียก readDBFToArray และ persistToMySQL มาใช้งาน


    public static function convertDBFToMySQL($fromDBF, $toMySQL, $myDB, $myUser, $myPass, $myHost){
        $arrTable = array();
        $arrTable = DBFMySQL::readDBFToArray($fromDBF);
        DBFMySQL::persistToMySQL($arrTable, $toMySQL, $myDB, $myUser, $myPass, $myHost);
    }

function readDBFToArray รับ path ของ dbf file เพื่อเอามาใส่ลง array โดยจะ return เป็น array 2 ชุดใหญ่ คือ ชุด column name กับชุดของ data โดยชุดของ data ก็จะเก็บเป็น array ย่อยลงไปอีก โดย array นี้จะ prepare เอาไว้ให้ function persistToMySQL เป็นคนเรียก


public static function readDBFToArray($fromDBF){
		// open in read-only mode
		$db = dbase_open($fromDBF, 0);
		$records = array();

		if ($db) {
		// read some data ..
			$record_numbers = dbase_numrecords($db);
			$field_numbers = dbase_numfields($db);
			for ($i = 1; $i <= $record_numbers; $i++) {
				$row = dbase_get_record_with_names($db, $i);
				array_push($records,array_values($row));
			}
			$columnHeader = array_keys($row);
			$arrTable = array($columnHeader,$records);
			dbase_close($db);
		}
		return $arrTable;
	}

function persistToMySQL รับ array มาจาก readDBFToArray เพื่อนำมาลง MySQL table ใน field ที่ถูกต้อง

public static function persistToMySQL($arrTable, $toMySQL, $myDB, $myUser, $myPass, $myHost){

		$connect = mysql_connect($myHost, $myUser, $myPass) or die("cannot connect to database");
		mysql_select_db($myDB, $connect);

		$columnHeader = $arrTable[0];
		$records = $arrTable[1];

		for($i=0;$i<sizeof($columnHeader)-1;$i++){
				$temp = $columnHeader[$i].",";
				$prepareFields=$prepareFields.$temp;
		}

		$prepareFields = substr($prepareFields, 0, -1);

		for($j=0;$j<sizeof($records);$j++){
			$prepareRecords='';
			for($i=0;$i<sizeof($records[$j])-1;$i++){
				$temp = "'".trim($records[$j][$i])."'".",";
				$prepareRecords=$prepareRecords.$temp;
			}
			$prepareRecords = substr($prepareRecords, 0, -1);
			$sql = "INSERT INTO $toMySQL ($prepareFields) VALUES ($prepareRecords)";
			$test = mysql_query($sql);
		}
	}

วิธีการใช้งานก็ง่ายมากๆ เลย ทำแบบนี้

<?php

/*created by Nattanicha Rittammanart
* Created date: Sep 22, 2008 */

include 'DBFMySQL.php';

$fromDBF='D:\PO_McDonald\INVORDD0.DBF';
$toMySQL='INVORDD0';
$myDB='db_mcdonald';
$myUser='root';
$myPass='root';
$myHost='localhost';

DBFMySQL::convertDBFToMySQL($fromDBF, $toMySQL, $myDB, $myUser, $myPass, $myHost);

?> 

ในอนาคตอาจจะมีการเพิ่มเติมฟังก์ชั่นเข้าไปอีกค่ะ หากเป็นประโยชน์ก็ช่วยบอกด้วยนะคะ

ดาวน์โหลด sourcecode ได้ที่นี่ ค่ะ

Posted in Foxpro, PHP | 1 Comment »

[PHP] Enable DBase function

Posted by natty on September 22, 2008

พอดีกำลังจะทำ class สำหรับ DBF MySQL ประมาณว่า convert กันไปกันมา เลยจะอ่าน DBase file แต่พอดีตะกี้ใช้แล้ว มัน error แบบนี้

Fatal error: Call to undefined function dbase_open() in C:\Program Files\Apache Software Foundation\Apache2.2\htdocs\DBFMySQL\DBFMySQL.php on line 27

พอไป search หาวิธีแก้ ตามลิงค์นี้ เค้าก็บอกว่า ต้องไปแก้ใน php.ini ให้มัน enable dbase dll ก่อน

extension=php_dbase.dll

เท่านี้แหละที่อยากบอก ;)

Posted in PHP | Leave a Comment »

PHP+Foxpro: การ query date

Posted by natty on September 17, 2008

จากตอนที่แล้ว เรื่องของ Apache + PHP + GD lib + Foxpro + JPGraph ก็มาดูกันต่อในเรื่องการ query date ของ foxpro

จะยกตัวอย่างฟิลด์ date ที่เก็บเป็น yyyy/mm/dd

สมมติว่าเรา query date ขึ้นมา


$sql = "SELECT data_date FROM tablename;
$rs=odbc_exec($conn,$sql);
        if (!$rs)
            {exit("Error in SQL");}
        while (odbc_fetch_row($rs))
        {
            $data_date = odbc_result($rs,"data_date");
        }

หากเราต้องการนำข้อมูล date ดังกล่าวไป query ต่อ สิ่งที่ต้องทำคือ ต้องเปลี่ยน format ของ date นั้นเป็น mm/dd/yyyy เนื่องจากเรากำลังจะใช้คำสั่งในการ query date ของ foxpro คือคำสั่ง ctod (character to date) ซึ่งโดยปกติหากเราต้องการใช้ format ของวันที่ยังคงเป็น yyyy/mm/dd เราจะมีการใช้คำสั่ง set century on ก่อน แต่เนื่องจากเราไม่ได้ใช้คำสั่งนี้ใน PHP เราเลยต้อง parse date ดังกล่าวออกมาเป็น character เพื่อประกอบกับไปให้ได้ตาม mm/dd/yyyy และใช้คำสั่ง ctod ได้

(*หมายเหตุ ใน foxpro มีคำสั่งที่คล้าย ctod คือ dtoc (date to character))

สิ่งที่เราจะได้จาก $data_date คือข้อมูล date โดยที่เราสามารถ parse ออกมาได้ดังนี้


$dateparse = date_parse($data_date);
$month = $dateparse ["month"];
$day = $dateparse ["day"];
$year = $dateparse ["year"];

หากเราต้องการนำข้อมูล date ดังกล่าวไป query ต่อแบบไม่ใช้ set century on ทำได้ดังนี้


$sql = "SELECT * FROM tablename WHERE data_date >= ctod('" .$month. "/" .$day. "/" .$year. "')";

เพียงเท่านี้เราก็จะหมดปัญหาการ query date ของ foxpro

Posted in Foxpro, PHP | Leave a Comment »

Apache + PHP + GD lib + Foxpro + JPGraph

Posted by natty on September 12, 2008

เริ่ม blog แรกด้วยเรื่องง่ายๆ ที่เกือบจะลืมไปแล้ว แอบทำช้านิดนึงด้วยเพราะว่าลืม และไม่เคยจะจดไว้เลย จดไว้นี่ก่อนแล้วกัน

วันนี้ลง Apache + PHP + GD Lib + Foxpro บน Windows เพราะว่าจะทำกราฟที่ดึงจาก Foxpro โดยใช้ PHP ซึ่งต้องทำงานผ่าน ODBC เลยต้องลงบน Windows

การติดตั้ง Apache + PHP
1. download and install Apache
2. download PHP และแตกไฟล์ลงใน path ที่ต้องการ ตัวนี้เป็น package เพราะเราต้องการตัว php_gd2.dll ด้วย
3. set system path to PHP (ถ้าไม่ set มันจะขึ้นหน้า page เวลาเข้า PHP ว่า You don’t have permission to access /php/php-cgi.exe)
3. เปิดไฟล์ httpd.conf เพื่อตั้งค่าให้ Apache สามารถ compile PHP ได้ โดยเพิ่มบรรทัดต่อไปนี้ลงไป

# ตำแหน่งที่เก็บ PHP Interpreter

ScriptAlias /php/ "c:/Program Files/PHP/"
AddType application/x-httpd-php .php 

#สำหรับ PHP version 5 ใช้ บรรทัดล่างนี้

Action application/x-httpd-php "/php/php-cgi.exe" 

4. ลองสร้างไฟล์ php ขึ้นมา แล้วปรินท์ phpinfo(); restart apache แล้วลองเข้า http://localhost/ชื่อไฟล์.php ถ้ามันแสดง phpinfo ให้ก็แสดงว่าเรา config สำเร็จแล้ว

5. เนื่องจากเราต้องใช้งาน php_gd2.dll ด้วย หากเราดูใน PHP/ext เราจะเห็น php_gd2.dll ซึ่งเราจะสามารถใช้งานได้โดยต้องเข้าไปที่ php.ini (ซึ่งอยู่ใน PHP package แต่เป็นชื่อ php-dist.ini เราต้องเปลี่ยนชื่อให้เป็น php.ini) แล้วเอา ; ออกที่บรรทัดชื่อว่า ;extension=php_gd2.dll

และอย่าลืมแก้ในไฟล์ php.ini ตรง extension_dir ให้ชี้ไปที่ ext เช่น extension_dir = “C:\Program Files\PHP\ext”

แล้วทำการ restart apache ใหม่

ใช้ code ต่อไปนี้เพื่อ test gd สิ่งทีจะได้คือ ข้อความสีชมพูเขียนว่า A simple Text String


$im = @ ImageCreate ( 150, 100)
or die ( “Cannot create a new GD image.”);
$background_color = ImageColorAllocate ($im, 255, 255, 255);
$text_color = ImageColorAllocate ($im, 233, 14, 91);
ImageString ($im, 1, 5, 5, “A Simple Text String”, $text_color );
header (“Content-type: image/png”);
ImagePng ($im); 

และ code ต่อไปนี้เพื่อ test gd2 สิ่งที่จะได้คือ กรอบสีเหลี่ยมสีดำกับขาว

$im =  imagecreatetruecolor ( 300, 200);
$black = imagecolorallocate ($im, 0, 0, 0 );
$white = imagecolorallocate ($im, 255, 255, 255 );

imagefilledrectangle ($im,0, 0,399,99 ,$white);
imagerectangle ($im,20, 20,250,190 ,$black);

header (“Content-type: image/png” );
imagepng ($im);

หากรันไม่ได้ มี error เหมือนกับว่ายังไม่มี jpgraph ให้เข้าไปดูที่ คำสั่ง extension_dir ใน php.ini ว่าชี้ไปยัง path ที่เป็น PHP/ext แล้วหรือยัง

การติดตั้ง jpgraph

jpgraph เป็น graph library ที่ใช้ได้กับ PHP โดยสามารถดาวน์โหลดได้จาก ที่นี่ โดยหากใช้กับ PHP5 ควรดาวน์โหลด version 2.xx แตกไฟล์ลง htdoc

การใช้งาน PHP ร่วมกับ Foxpro

1. set system ODBC (ย้ำ system นะ) โดยเลือก driver เป็น Microsoft Visual Foxpro Driver แล้วใส่ชื่อ datasource name (dsn) ที่ต้องการ เลือก database type เป็น free table และ browse ไปยัง path ที่ DBF ของเราวางอยู่

2. ลอง test ว่า php ของเราสามารถติดต่อกับ dbf ได้หรือยัง โดยใช้ code ต่อไปนี้ ซึ่งจะทำให้แสดงข้อมูลพร้อมหัวคอลัมน์ออกมา

<?php
$id = odbc_connect( “yourdsnname”,”",”");
if ( $id != 0 )   {
echo “DSN qc2004 connected
”;
echo “ID = “, $id, “
”;
$sql = “SELECT * from yourtable”;
$rs = odbc_exec( $id, $sql );
$odbc_result_all($rs);
} else  {
echo “Error connecting to qc2004 webdata”;
}
odbc_close_all();
?>

Posted in PHP | 1 Comment »