ผมอยากทราบ เหตุผลที่สำคัญในการ เขียน contents ลง buffer ด้วย php ครับ
ป้องกันการ overflow มั่ง
Date :
2010-03-27 10:55:24
By :
onedan
จะทราบได้โดยวิธีการใดครับว่า buffer overflow ได้เกิดใน ระบบที่เราพัฒนาไหม หมายถึง ข้อสังเกตนะครับ ตรวจสอบได้ไหมครับ
Date :
2010-03-27 11:35:44
By :
nilas
บัฟเฟอร์ ก็คือบัฟเฟอร์ ครับ
บัฟเฟอร์ก็คือ ข้อมูลเก็บไว้ในหน่วยความจำ ดังนั้นก่อนมี output เราสามารถเอาบัฟเฟอร์มาแก้ได้ ก่อนส่งเอาท์พุทคับ
ทำไมต้องใช้บัฟเฟอร์ มีประโยชน์อย่างไร
ช่วยให้เขียนโปรแกรมได้สะดวกขึ้นครับ เช่น
เว็บเราใช้ เทมเพลต แน่นอนว่าเทมเพลตมันเป็นรูปร่างเว็บ แต่อาจมีข้อมูลที่เป็นใดนามิคอยู่ข้างในด้วย
ดังนั้นการจะเขียนการส่งข้อมูลเข้าบัฟเฟอร์ ต้องไม่ส่งเอาท์พุดไปก่อน เช่น
Code (PHP)
$d_data1 = '<img src="xxx" />...<div>x</div> .etc';
$d_data2 = 'xxx';
//ส่งข้อมูล ไดามิค(สมมติว่า $d_data1 เป็น dynamic)
function make_page($d_data1 = '', $data2 = '', $d_datax = '')
{
$rt <<<TEMPLATE
<html>
<body>
<div id="head">{$d_data1}</div>
<div id="body">{$d_data2}</div>
<div id="foot">{$d_datax}</div>
</body>
</html>
TEMPLATE;
return $rt;
}
จะเห็นได้ว่าเราต้องเก็บข้อมูลไว้ในหน่วยความจำก่อนในที่นี้คือตัวแปร $d_datax; และค่อยเอาข้อมูลไปใส่ในส่วนต่างๆของ เทมเพลตที่เราได้สร้างไว้ ก็จะถูกส่งเอาท์พุตไปได้อย่างถูกต้องตามตำแหน่งของข้อมูล
แต่วิธีการเขียนแบบนี้ เราต้องมานั่งเก็บข้อมูลใส่ตัวแปร จึงทำให้โค้ดมีจำนวนมาก หรือ โค้ดสับซ้อน อ่านยากในการจะทำ สตริงเช่น
Code (PHP)
$arr = array('opt1' => 'หน้าแรก', 'opt2' => 'หน้าที่สอง', ...);
$d_data1 = '<select>';
foeach($arr as $key => $val)
$d_data1 .= "\n".'<option value="'.$key.'">'.$val.'</option>';
$d_data1 .= '</select>';
แบบนี้มันก็เขียนได้ แต่ถ้ามีอะไรเยอะๆอยู่ มันก็ยิ่งต้อง คอนเซนเทรด และโค้ดอ่านยากขึ้น
คราวนี้เราลองมาใช้ buffer กัน
Code (PHP)
$arr = array('opt1' => 'หน้าแรก', 'opt2' => 'หน้าที่สอง', ...);
ob_start();
echo '<select>';
foreach($arr as $key => $val)
echo "\n".'<option value="'.$key.'">'.$val.'</option>'; // echo เข้าบัฟเฟอร์
echo '</select>';
$d_data1 = ob_get_clean();
แบบนี้ จะดูเป็นธรรมชาติมากขึ้นครับ
อีกอย่างที่บอกไปแล้วนั้นคือ การเก็บไว้ในบัฟเฟอร์เราสามารถจะ ทำ adapter คล้ายตัวแปลงก่อน ส่งไปที่เอาท์พุต
ตัวอย่างเช่น การกรอง คำหยาบ หรือ บีฟไฟล์ในรูปแบบต่างๆ
Code (PHP)
$output = make_page($d_data1);
$คำหยาบ = array('xxx', 'xxxx', etc.);
// เปลี่ยน กำหยาบ
$output = str_replace($คำหยาบ , '!!!', $output);
header('Content-type: text/html');
echo $output;
หรืออาจบีฟไฟล์ก็ได้ แต่ไม่เคยเขียน ขอผ่านละกัน
หรืออาจเข้ารหัสก็ได้ ในกรณีที่ ให้ดาวโหลดแต่ต้องมีโปรแกรมถอด
เห็นประโยชน์ หรือ ความจำเป็นหรือยังคับ
http://gunner.freetzi.com
Date :
2010-03-27 12:24:43
By :
pjgunner
ผมไม่เข้าใจว่า "ช่วยให้เขียนโปรแกรมได้สะดวกขึ้นครับ" อธิบาย ให้เห็นภาพมากกว่านี้ได้ไหมครับ
เปรียบเทียบกับการไม่ใช้ buffer ก็จะขอบพระคุณมากกครับ
ผมเริ่มเห็น ประโยชน์แล้วครับ แต่อยากเห็นภาพมากกว่านี้นะครับ เพราะ ผมจะได้เข้าใจ และ เลือกใช้ได้อย่างเหมาะสมนะครับ
Date :
2010-03-27 12:59:01
By :
nilas
ผมว่าผมอธิบาย เครียร์แล้วนะ
ที่ว่า "ช่วยให้เขียนโปรแกรมได้สะดวกขึ้นครับ" ก็คือ ทำให้ออกแบบโค้ดภายในได้ สะอาด และ หลากหลายวิธีมากขึ้นคับ
Date :
2010-03-27 13:14:04
By :
pjgunner
Code (PHP)
<?php
function callback($buffer)
{
// replace all the apples with oranges
return (str_replace("apples", "oranges", $buffer));
}
ob_start("callback");
?>
<html>
<body>
<p>It's like comparing apples to oranges.</p>
</body>
</html>
<?php
ob_end_flush();
?>
กับ
Code (PHP)1
<?php
function callback($buffer)
{
// replace all the apples with oranges
return (str_replace("apples", "oranges", $buffer));
}
$data='<html>
<body>
<p>It's like comparing apples to oranges.</p>
</body>
</html>';
echo callback($data);
?>
ในการใช้ หน่วยความจำ ทั้งสองตัวอย่าง ต่างกันอย่างไรครับ ผมว่า ผลที่ได้ออกมาเหมือนกัน แต่ไม่เข้าใจส่วนที่ ต่างกันนะครับ ช่วย อธิบาย หน่อยนะครับ
Date :
2010-03-27 13:34:02
By :
nilas
จากผมไปลองนั่งคิด ดู ระหว่าง สอง ตัวอย่างนี้ ไม่รู้ว่า ผมเข้าใจถูกไหมครับ
ตัวอย่างแรก ถ้า ผมเลือก ใช้ ob_xxx หรือการนำข้อมูลไปเก็บใน buffer ผมสามารถ ใช้คำสั่ง ob_end_flush หรือ อื่น เพื่อคืนค่า ให้ buffer ได้
แต่ ตัวอย่างที่ 2 ผมใช้ ตัวแปร $data ในการเก็บข้อมูล แต่ จองใช้เลย โดย ไม่ได้สั่งคืนค่าให้หน่วยความจำ
ไม่รุ้ว่าผม เข้าใจความต่างถูกไหมครับ ถ้าไม่ถูกต้อง ประการได้ ช่วยชี้แนะด้วยนะครับ
Date :
2010-03-27 13:59:17
By :
nilas
มันไม่ต่างกันหรอกคับ แต่ถ้าหากมีข้อมูลเยอะๆล่ะ คุณ จะมานั่ง callback() ทุกตัว เหรอคับ
Date :
2010-03-27 14:03:10
By :
pjgunner
แล้ว ถ้าสั่ง พวก ob_xxx ทำครั้งเดี่ยวเหรอครับ คือผมไม่เข้าใจนะครับ
Date :
2010-03-27 14:19:04
By :
nilas
ไม่มีอะไรให้ต้องงง คับ แปลตามตัว ใช้ได้เรื่อยๆ
Date :
2010-03-27 14:43:19
By :
pjgunner
ก่อนอื่น ผมต้อง ขออภัยก่อนเลยนะครับ ที่ผมมีคำถาม มากมาย บ้างคำถาม อาจไม่สบอารมณ์ คุณ เอี่ยว(Awake) สักเท่าไร
ผมเข้าใจว่า คุณ เอี่ยว ตอบในฐานะ ผมเข้าใจ และมี ประสบการณ์ ในการนำ ob_xxx ไปใช้มากมาย บ้างครั้ง อาจจะ ไม่เข้าใจว่า ผม ทำไม ไม่เข้าใจสิ่งที่บอกไปนะครับ
แต่ จริงๆ แล้ว ผม ถาม ในฐานะ ผู้ไม่เคยใช้ พวก ob_xxx แต่ก็เขียน โปรแกรม มาเยอะอยู่ เขียนแบบไม่ได้มีหลักการ อะไรนะครับ
เลยสงสัยเรื่อง ประโยชน์ให้เยอะๆ ของ function พวก ob_xxx จากคำถามที่ยังหาคำตอบยังไม่ได้ เพื่อจะได้นำไปใช้อย่างถูกต้อง และ มีประโยชน์มากเท่าที่จะทำได้นะครับ
ต่อนะครับ No. 10 ใช้ได้เรื่อยๆ ที่ว่านี้ มี code ตัวอย่างที่เคยทำไหมครับ...
Date :
2010-03-27 15:04:42
By :
nilas
จริงๆ แล้วผมน่ะศึกษาแค่ทฤษฎี ก็จำไว้แค่นั้นแหละคับ
เพียงแต่เวลาเราคิดจะโค้ดอะไรนั้น เราต้องเจอปัญหามากมาย และหาโซลูชั่น เทคนิคที่เหมาะสม ดังนั้น เราจะใช้มันเมื่อไหร่นั้น ก็ใช้ในกรณีที่มันช่วย ให้แก้ปัญหาของเราได้
ส่วนการใช้ output buffer นั้น ส่วนมากก็ไม่ค่อยได้ใช้หรอกคับ ถ้าเขียนธรรมดา
บัฟเฟอร์มัน ก็เหมือนกับตัวแปรธรรมดานั่นแหละคับ ว่าจะ echo ออกไป หรือไม่ เพียงแต่ เราไม่ต้องไป เก็บค่าในตัวแปร เพียงแต่ส่งเอาท์พูทไปที่ buffer คือมันดักเอาท์พุทเราอัตโนมัติ
ตัวอย่าง เช่น ถ้าเราจะใช้ header() เราต้อง ห้ามมี output ออกไปก่อน เพราะ header มันต้องส่งบอกก่อนว่ามีข้อมูลอะไร ดังนั้นถ้าเราเขียน header() ไว้บนสุด จะทำให้เราเขียนโปรแกรมไว้ได้ยากขึ้น แต่ถ้าเรากั๊กไว้ในบัฟเฟอร์ก่อน มันก็สามารถเรียก ใช้ header() ไว้ได้ทุกที่ ตราบใดที่ยังไม่ส่ง output
โค้ดตัวอย่างเหรอคับ ดูว่าโค้ดมันจะยาวไป ผมเอามาแค่หน้าที่ใช้ บัฟเฟอร์ลกัน
Code (PHP)
<?php
require_once 'config.inc.php';
require_once 'include/view.inc.php';
Util::register_request_vars();
// ตรวจสอบสิทธิการเข้าถึง
if ($_GET['for_id'])
{
if ($_SESSION['user']['type'] !== 'admin') exit('Permission denied.');
$hotel_id = (int) $_GET['for_id'];
}
elseif ($_GET['hotel_id'])
{
if ($_SESSION['user']['type'] !== 'hotel' OR $_SESSION['user']['id'] !== $_GET['hotel_id']) exit('Permission denied.');
$hotel_id = (int) $_GET['hotel_id'];
}
else{
exit('Permission denied.');
}
$query = mysql_query("SELECT hotel_id FROM hotel WHERE hotel_id=$hotel_id");
if ( ! mysql_num_rows($query)) exit('Permission denied.'); // ไม่มีโรงแรม id นี้อยู่จริง
// หน้าบันทึกข้อมูล
$save_file = 'hotel_save.php?'.($_GET['for_id'] ? 'for_id='.$_GET['for_id'] : 'hotel_id='.$_SESSION['user']['id']);
$sql1 = "SELECT * FROM roomtype LEFT JOIN roomprice USING(roomtype_id) WHERE hotel_id=$hotel_id";
$query = mysql_query($sql1);
if ( ! mysql_num_rows($query)) // ยังไม่มีข้อมูลราคา ต้องเพิ่มเข้าไปใหม่
{
$query = mysql_query("SELECT * FROM roomtype");
while ($row = mysql_fetch_assoc($query))
{
mysql_query("INSERT INTO roomprice VALUES($hotel_id, {$row['roomtype_id']}, 0, NULL, 'N')");
}
$query = mysql_query($sql1);
}
?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>Eastern Travel-> จัดการข้อมูลราคาห้อง</title>
<link href="media/style/style.css" rel="stylesheet" type="text/css" />
<script type="text/javascript" src="media/javascript/jquery-1.3.2.min.js"></script>
<script type="text/javascript">
var regex_picture = /\.(jpe?g|gif|png)$/i;
var regex_number = /^\d+(\.\d{1,2})?$/;
$(document).ready(function()
{
// ตรวจสอบข้อมูลตอนซับมิตฟอร์ม
$('form').submit(function()
{
var accept = true;
var display;
var occur;
// เช็คราคา
$(':text[name^=price]').each(function(i)
{
if ($(this).val() != '')
{
if ( ! regex_number.test($(this).val()))
{
accept = false;
display = 'โปรดกรอกราคาห้องให้ถูกต้อง';
occur = this;
return false; // break each()
}
}
});
// เช็ครูปภาพ เฉพาะ jpg,png,gif
$(':file[name^=picture]').each(function(i)
{
if ($(this).val() != '')
{
if ( ! regex_picture.test($(this).val()))
{
accept = false;
display = 'โปรดกรอกชนิดรูปให้ถูกต้อง';
occur = this;
return false; // break each()
}
}
});
if ( ! accept)
{
alert(display);
$(occur).focus();
return false;
}
return true;
}); // submit()
});
</script>
</head>
<body>
<?php
// แสดงหัวเว็บ
show_header();
// เปิดใช้ บัฟเฟอร์
ob_start();
?>
<div align="center">
<table width="740" border="0" cellspacing="0" cellpadding="3">
<tr>
<td align="left"><span style="font-weight:bold; font-size:14px; color:#FF6600">จัดการข้อมูลราคาห้อง</span></td>
</tr>
<tr>
<td align="left" bgcolor="#eeeeee">รูปห้องจะกรอกหรือไม่กรอกก็ได้ แต่ถ้ากรอกจะเป้นการบันทึกซ้ำเข้าไป ถ้าไม่กรอกจะใช้รูปล่าสุดที่เคยกรอกไปคับ </td>
</tr>
</table>
<form id="form1" name="form1" method="post" action="<?=$save_file?>" enctype="multipart/form-data">
<input type="hidden" name="save_type" value="change_roomprice" />
<table width="740" border="0" cellspacing="0" cellpadding="3">
<?php
$col_size = 2; // จำนวนคอมลัมน์
$type_size = 3;
$bg_colors = array('#f8f8f8', '#eeeeee', '#dddddd', '#cccccc');
$x = 0;
while ($row = mysql_fetch_assoc($query))
{
if ($i % $col_size === 0) echo '<tr>';
if ($i % $type_size === 0) $bg_color = $bg_colors[$x++];
if ($row['roomprice_status'] === 'Y')
{
$status_y = 'checked="checked"';
$status_n = '';
}
else
{
$status_y = '';
$status_n = 'checked="checked"';
}
?>
<td width="33%"><table width="100%" bgcolor="<?=$bg_color?>" border="0" cellspacing="0" cellpadding="3">
<tr>
<td align="right">ชนิดห้อง</td>
<td align="left"><strong><?="{$row['roomtype_name']} ({$row['roomtype_pnumber']} คน)"?></strong></td>
</tr>
<tr>
<td align="right">ค่าพักต่อวัน</td>
<td align="left"><input name="price[<?=$row['roomtype_id']?>]" type="text" value="<?=$row['roomprice_price']?>" style="width:80px" maxlength="11" />
บาท</td>
</tr>
<tr>
<td align="right">รูปห้อง</td>
<td align="left"><input type="file" name="picture[<?=$row['roomtype_id']?>]" /></td>
</tr>
<tr>
<td align="right">แสดง?</td>
<td align="left"><input name="status<?=$row['roomtype_id']?>" id="status<?=$row['roomtype_id']?>_Y" type="radio" value="Y" <?=$status_y?> />
<label for="status<?=$row['roomtype_id']?>_Y">แสดง</label>
<input name="status<?=$row['roomtype_id']?>" id="status<?=$row['roomtype_id']?>_N" type="radio" value="N" <?=$status_n?> />
<label for="status<?=$row['roomtype_id']?>_N">ไม่แสดง</label></td>
</tr>
</table></td>
<?php
$i++;
} // while
?>
</table>
<div align="center">
<input type="submit" name="Submit" value=" แก้ไขข้อมูล >> บันทึก " />
</div>
</form>
</div>
<?php
// เก็บค่า บัฟเฟอร์ที่เก็บไว้ ใส่ตัวแปร แล้วล้างบัฟเฟอร์
$buffer = ob_get_clean();
// แสดง ส่วนบอดี้เว็บ
show_body($buffer, get_menu_html());
// แสดง ส่วนท้าย
show_footer();
?>
</body>
</html>
ส่วนบอดี้นี้ ทำเป็น html code ธรรมดา แล้วเก็ฐไว้ในบัฟเฟอร์ แล้วส่งให้ฟังชั่น show_body ซึ่งมีสองคอลั่ม ที่เป็นไดนามิคคอนเทนต์เหมือนกัน
โค้ดนี้ไม่ได้มีอะไรเป็นพิเศษหรอกคับ เขียนแบบลวกๆ ด้วยซ้ำ
แต่สังเกตดูว่า ส่วน html ธรรมดา ไม่ได้เขียนในส่วนของ php ด้วยซ้ำ มันก็ถูกดักเข้าบัฟเฟอร์เช่นกัน และโคดได้เร็วในดีไซน์วิวด้วย
Date :
2010-03-27 15:29:29
By :
pjgunner
ขอบคุณครับ
ไม่รู้ที่ thaicreate ตอนนี้ที่ กดปุ่ม Preview Form ซึ่งผมเชื่อว่ายังไม่มีการ บันทึก เพื่อให้ผู้ post ตรวจสอบคำผิดก่อนบันทึกนี้ ใช้ การ เขียนลง buffer ด้วยไหมครับ
Date :
2010-03-29 11:16:22
By :
nilas
ไม่จำเป็นเลยคับ กรณีนี้ก็แค่ ส่งไฟล์ไปอีกหน้านึงและคืนค่า text กลับมาแค่นั้นคับ
หมายเหตุ. ถ้าจบรีเควส หรือจบไฟล์ php ก็จะ flush ให้อัติโนมัติคับ
ปล. ไม่ได้แกะโค้ดนะคับ
Date :
2010-03-29 11:50:18
By :
pjgunner
ปกติผมเขียน code โดยการนำไปใส่ใน ตัวแปรกัน ทั้งหมด ไม่ค่อยได้ใช้ echo เลย ผมจะ นำตัวแปรที่ได้เก็บข้อมูลทั้งหมด ไปส่งให้ buffer อย่างไร ดีครับ
Code (PHP)
<?php
// แสดงหัวเว็บ
show_header();
// เปิดใช้ บัฟเฟอร์
ob_start();
$data='<div align="center">
<table width="740" border="0" cellspacing="0" cellpadding="3">
<tr>
<td align="left"><span style="font-weight:bold; font-size:14px; color:#FF6600">จัดการข้อมูลราคาห้อง</span></td>
</tr>
<tr>
<td align="left" bgcolor="#eeeeee">รูปห้องจะกรอกหรือไม่กรอกก็ได้ แต่ถ้ากรอกจะเป้นการบันทึกซ้ำเข้าไป ถ้าไม่กรอกจะใช้รูปล่าสุดที่เคยกรอกไปคับ </td>
</tr>
</table>
<form id="form1" name="form1" method="post" action="<?=$save_file?>" enctype="multipart/form-data">
<input type="hidden" name="save_type" value="change_roomprice" />
<table width="740" border="0" cellspacing="0" cellpadding="3">';
$col_size = 2; // จำนวนคอมลัมน์
$type_size = 3;
$bg_colors = array('#f8f8f8', '#eeeeee', '#dddddd', '#cccccc');
$x = 0;
while ($row = mysql_fetch_assoc($query))
{
if ($i % $col_size === 0) echo '<tr>';
if ($i % $type_size === 0) $bg_color = $bg_colors[$x++];
if ($row['roomprice_status'] === 'Y')
{
$status_y = 'checked="checked"';
$status_n = '';
}
else
{
$status_y = '';
$status_n = 'checked="checked"';
}
$data.=' <td width="33%"><table width="100%" bgcolor="'.$bg_color.'" border="0" cellspacing="0" cellpadding="3">
<tr>
<td align="right">ชนิดห้อง</td>
<td align="left"><strong>'.$row['roomtype_name'].$row['roomtype_pnumber'].' คน).'</strong></td>
</tr>
<tr>
<td align="right">ค่าพักต่อวัน</td>
<td align="left"><input name="price['.$row['roomtype_id'].']" type="text" value="'.$row['roomprice_price'].'" style="width:80px" maxlength="11" />
บาท</td>
</tr>
<tr>
<td align="right">รูปห้อง</td>
<td align="left"><input type="file" name="picture['.$row['roomtype_id'].']" /></td>
</tr>
<tr>
<td align="right">แสดง?</td>
<td align="left"><input name="status'.$row['roomtype_id'].'" id="status<?=$row['roomtype_id']?>_Y" type="radio" value="Y" '.$status_y.' />
<label for="status'.$row['roomtype_id'].'_Y">แสดง</label>
<input name="status'.$row['roomtype_id'].'" id="status'.$row['roomtype_id'].'_N" type="radio" value="N" '.$status_n.' />
<label for="status'.$row['roomtype_id'].'_N">ไม่แสดง</label></td>
</tr>
</table></td>';
$i++;
} // while
$data.='</table>
<div align="center">
<input type="submit" name="Submit" value=" แก้ไขข้อมูล >> บันทึก " />
</div>
</form>
</div>';
// เก็บค่า บัฟเฟอร์ที่เก็บไว้ ใส่ตัวแปร แล้วล้างบัฟเฟอร์
$buffer = ob_get_clean($data);// เก็บค่าใน buffer โดยระบบ ตัวแปรที่เก็บข้อมูลไว้ ไม่รู้ว่าถูกไหม
// แสดง ส่วนบอดี้เว็บ
show_body($buffer, get_menu_html());
// แสดง ส่วนท้าย
$data.=show_footer();
$data.='
</body>
</html>';
?>
Date :
2010-03-29 13:33:14
By :
nilas
Load balance : Server 03