mysql_query函數(shù)查詢的方式是查詢出全部結(jié)果后緩存到內(nèi)存中,這樣就會出現(xiàn)超內(nèi)存的現(xiàn)象,使用另外一個函數(shù)mysql_unbuffered_query可以解決這個問題,mysql_unbuffered_query不會緩存結(jié)果集,而是查詢出來數(shù)據(jù)后立馬對結(jié)果集進行操作,也就是便查詢邊返回,這樣就不會出現(xiàn)超出內(nèi)存的現(xiàn)象,但是使用mysql_unbuffered_query的是時候不能使用 mysql_num_rows() 和 mysql_data_seek()。并且向 MySQL 發(fā)送一條新的 SQL 查詢之前,必須提取掉所有未緩存的 SQL 查詢所產(chǎn)生的結(jié)果行。例如:
10年積累的網(wǎng)站設(shè)計制作、做網(wǎng)站經(jīng)驗,可以快速應(yīng)對客戶對網(wǎng)站的新想法和需求。提供各種問題對應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識你,你也不認(rèn)識我。但先網(wǎng)站制作后付款的網(wǎng)站建設(shè)流程,更有永城免費網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
使用緩存結(jié)果集的代碼:
function selecttest()
{
try {
$pdo = new PDO("mysql:host=localhost;dbname=test", 'root', '123456');
// 不使用緩存結(jié)果集方式
// $pdo-setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, false);
$sth = $pdo-prepare('select * from test');
$sth-execute();
echo '最初占用內(nèi)存大?。? . memory_get_usage() . "\n";
$i = 0;
while ($result = $sth-fetch(PDO::FETCH_ASSOC)) {
$i += 1;
if ($i 10) {
break;
}
sleep(1);
print_r($result);
echo '占用內(nèi)存大?。? . memory_get_usage() . "\n";
}
} catch (Exception $e) {
echo $e-getMessage();
}
}
執(zhí)行時將會報超出內(nèi)存的錯誤:
Fatal error: Allowed memory size of 134217728 bytes exhausted (tried to allocate 204800000 bytes) in E:\ProgramDevelopment\RuntimeEnvironment\xampp\htdocs\test\test.php on line 56
Call Stack:
0.0005 135392 1. {main}() E:\ProgramDevelopment\RuntimeEnvironment\xampp\htdocs\test\test.php:0
0.0005 135568 2. test-selecttest() E:\ProgramDevelopment\RuntimeEnvironment\xampp\htdocs\test\test.php:85
0.0050 142528 3. PDOStatement-execute() E:\ProgramDevelopment\RuntimeEnvironment\xampp\htdocs\test\test.php:56
將上面代碼中的$pdo-setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, false);一行的注釋去掉后將不在緩存結(jié)果集,這時運行該函數(shù)的結(jié)果如下:
最初占用內(nèi)存大?。?44808
Array
(
[id] = 1
[a] = v
[b] = w
[c] = i
)
占用內(nèi)存大小:145544
Array
(
[id] = 2
[a] = b
[b] = l
[c] = q
)
占用內(nèi)存大?。?45544
Array
(
[id] = 3
[a] = m
[b] = p
[c] = h
)
占用內(nèi)存大小:145536
Array
(
[id] = 4
[a] = j
[b] = i
[c] = b
)
占用內(nèi)存大?。?45536
可以看到,這時返回一條數(shù)據(jù)內(nèi)存占用非常的小,也就700多字節(jié),這樣就不會出現(xiàn)超出內(nèi)存的錯誤了。
?php
class MySQL{
private $host; //服務(wù)器地址
private $name; //登錄賬號
private $pwd; //登錄密碼
private $dBase; //數(shù)據(jù)庫名稱
private $conn; //數(shù)據(jù)庫鏈接資源
private $result; //結(jié)果集
private $msg; //返回結(jié)果
private $fields; //返回字段
private $fieldsNum; //返回字段數(shù)
private $rowsNum; //返回結(jié)果數(shù)
private $rowsRst; //返回單條記錄的字段數(shù)組
private $filesArray = array(); //返回字段數(shù)組
private $rowsArray = array(); //返回結(jié)果數(shù)組
private $charset='utf8'; //設(shè)置操作的字符集
private $query_count=0; //查詢結(jié)果次數(shù)
static private $_instance; //存儲對象
//初始化類
private function __construct($host='',$name='',$pwd='',$dBase=''){
if($host != '') $this-host = $host;
if($name != '') $this-name = $name;
if($pwd != '') $this-pwd = $pwd;
if($dBase != '') $this-dBase = $dBase;
$this-init_conn();
}
//防止被克隆
private function __clone(){}
public static function getInstance($host='',$name='',$pwd='',$dBase=''){
if(FALSE == (self::$_instance instanceof self)){
self::$_instance = new self($host,$name,$pwd,$dBase);
}
return self::$_instance;
}
public function __set($name,$value){
$this-$name=$value;
}
public function __get($name){
return $this-$name;
}
//鏈接數(shù)據(jù)庫
function init_conn(){
$this-conn=@mysql_connect($this-host,$this-name,$this-pwd) or die('connect db fail !');
@mysql_select_db($this-dBase,$this-conn) or die('select db fail !');
mysql_query("set names ".$this-charset);
}
//查詢結(jié)果
function mysql_query_rst($sql){
if($this-conn == '') $this-init_conn();
$this-result = @mysql_query($sql,$this-conn);
$this-query_count++;
}
//取得字段數(shù)
function getFieldsNum($sql){
$this-mysql_query_rst($sql);
$this-fieldsNum = @mysql_num_fields($this-result);
}
//取得查詢結(jié)果數(shù)
function getRowsNum($sql){
$this-mysql_query_rst($sql);
if(mysql_errno() == 0){
return @mysql_num_rows($this-result);
}else{
return '';
}
}
//取得記錄數(shù)組(單條記錄)
function getRowsRst($sql,$type=MYSQL_BOTH){
$this-mysql_query_rst($sql);
if(empty($this-result)) return '';
if(mysql_error() == 0){
$this-rowsRst = mysql_fetch_array($this-result,$type);
return $this-rowsRst;
}else{
return '';
}
}
//取得記錄數(shù)組(多條記錄)
function getRowsArray($sql,$type=MYSQL_BOTH){
!empty($this-rowsArray) ? $this-rowsArray=array() : '';
$this-mysql_query_rst($sql);
if(mysql_errno() == 0){
while($row = mysql_fetch_array($this-result,$type)) {
$this-rowsArray[] = $row;
}
return $this-rowsArray;
}else{
return '';
}
}
//更新、刪除、添加記錄數(shù)
function uidRst($sql){
if($this-conn == ''){
$this-init_conn();
}
@mysql_query($sql);
$this-rowsNum = @mysql_affected_rows();
if(mysql_errno() == 0){
return $this-rowsNum;
}else{
return '';
}
}
//返回最近插入的一條數(shù)據(jù)庫的id值
function returnRstId($sql){
if($this-conn == ''){
$this-init_conn();
}
@mysql_query($sql);
if(mysql_errno() == 0){
return mysql_insert_id();
}else{
return '';
}
}
//獲取對應(yīng)的字段值
function getFields($sql,$fields){
$this-mysql_query_rst($sql);
if(mysql_errno() == 0){
if(mysql_num_rows($this-result) 0){
$tmpfld = @mysql_fetch_row($this-result);
$this-fields = $tmpfld[$fields];
}
return $this-fields;
}else{
return '';
}
}
//錯誤信息
function msg_error(){
if(mysql_errno() != 0) {
$this-msg = mysql_error();
}
return $this-msg;
}
//釋放結(jié)果集
function close_rst(){
mysql_free_result($this-result);
$this-msg = '';
$this-fieldsNum = 0;
$this-rowsNum = 0;
$this-filesArray = '';
$this-rowsArray = '';
}
//關(guān)閉數(shù)據(jù)庫
function close_conn(){
$this-close_rst();
mysql_close($this-conn);
$this-conn = '';
}
//取得數(shù)據(jù)庫版本
function db_version() {
return mysql_get_server_info();
}
}
在php中,表單POST提交的數(shù)據(jù)是存放在$_POST變量中.$_POST變量是一個數(shù)組,它是一個以表單字段名作索引的數(shù)組.比如有以下表單:
form?method="post"
p姓名:input?type="text"?name="name"?value=""/p
p年齡:input?type="text"?name="age"?value=""/p
pinput?type="submit"?value="提交"/p
/form
輸入值后提交,按你的要求,在php層處理輸出提交的內(nèi)容,那么可以這樣:
?php
echo?'你的姓名是:'.$_POST['name'];//其中$_POST['name']中存放的是上面表單名為name的值
echo?'你今年'.$_POST['age'].'歲';//其中$_POST['age']中存放的是上面表單名為age的值
?
如果字段內(nèi)容很多,有時就可能需要循環(huán)處理.
foreach($_POST?as?$val){
echo?$val;
}
或直接:
print_r($_POST);
當(dāng)然處理或輸出的格式很多,以上只是一個示例.
首先 將json 解碼成數(shù)組, 用json_decode 函數(shù) 注意 一定要加上第二個參數(shù) 否則他會返回一個對象。接下來就是地遞歸了。這是一個最簡單的遞歸只要逐個遍歷即可。
下面是完整的代碼:
$data=?json_decode($str,true);
$options?=?getChildren($data);
function?getChildren($parent,$deep=0)?{
foreach($parent?as?$row)?{
$data[]?=?array("id"=$row['id'],?"name"=$row['name'],"pid"=$row['parentid'],'deep'=$deep);
if?($row['childs'])?{
$data?=?array_merge($data,?getChildren($row['childs'],?$deep+1));
}
}
return?$data;
}
?
select?name=""?id=""
?php??foreach?($options?as?$row)?{??
option?value="?php?echo?$row['id']??"?php?echo?str_pad("",$row['deep']*3,?"-",STR_PAD_RIGHT);???php?echo?$row['name'];??/option
?php?}??
/select
以上代碼已測試通過 效果圖如下
當(dāng)異常被觸發(fā)時,通常會發(fā)生:在PHP5中添加了類似于其它語言的錯誤異常處理模塊。在 PHP代碼中所產(chǎn)生的異??杀?throw語句拋出并被 catch 語句捕獲。需要進行異常處理的代碼都必須放入 try 代碼塊內(nèi),以便捕獲可能存在的異常。每一個 try 至少要有一個與之對應(yīng)的 catch。
使用多個 catch 可以捕獲不同的類所產(chǎn)生的異常,當(dāng) try 代碼塊不再拋出異?;蛘哒也坏?catch 能匹配所拋出的異常時,PHP 代碼就會在跳轉(zhuǎn)到最后一個 catch 的后面繼續(xù)執(zhí)行。當(dāng)然,PHP 允許在 catch 代碼塊內(nèi)再次拋出(throw)異常,當(dāng)一個異常被拋出時,其后(譯者注:指拋出異常時所在的代碼塊)的代碼將不會繼續(xù)執(zhí)行,而 PHP 就會嘗試查找第一個能與之匹配的 catch,如果一個異常沒有被捕獲,而且又沒用使用 set_exception_handler() 作相應(yīng)的處理的話,那么 PHP 將會產(chǎn)生一個嚴(yán)重的錯誤,并且輸出 Uncaught Exception ... (未捕獲異常)的提示信息.
1、異常類的層級關(guān)系,代碼如下:
復(fù)制代碼 代碼如下:
class NotFoundException extends Exception{}
class InputException extends Exception{}
class DBException extends Exception{}
2、配置未捕捉異常的處理器,代碼如下:
復(fù)制代碼 代碼如下:
function exception_uncaught_handler(Exception $e) {
header('Content-type:text/html; charset=utf-8');
if ($e instanceof NotFoundException)
exit($e-getMessage());
elseif ($e instanceof DBException)
exit($e-getMessage());
else
exit($e-getMessage());
}
set_exception_handler('exception_uncaught_handler');
3、在數(shù)據(jù)庫連接代碼,手動拋出DBException異常但未使用try…catch進行捕獲處理,該異常將被PHP自定義異常處理器,exception_uncaught_handler()函數(shù)處理:
復(fù)制代碼 代碼如下:
$this-resConn = mysql_connect ($CONFIGS['db_host'], $CONFIGS['db_user'], $CONFIGS['db_pwd']);
if (false == is_resource($this-resConn))
throw new DBException('數(shù)據(jù)庫連接失敗。'.mysql_error($this-resConn));
4、業(yè)務(wù)邏輯一瞥:
復(fù)制代碼 代碼如下:
if (0 != strcmp($curAlbum-interest_id, $it))
throw new NotFoundException('很抱歉,你所訪問的相冊不存在');
以上就是PHP自定義異常處理器的具體使用方法.
php實例代碼如下:
復(fù)制代碼 代碼如下:
?php
class customException extends Exception
{
public function errorMessage()
{
//error message
$errorMsg = 'Error on line '.$this-getLine().' in '.$this-getFile()
.': b'.$this-getMessage().'/b is not a valid E-Mail address';
return $errorMsg;
}
}
$email = "someone@example.com";
try
{
//check if
if(filter_var($email, FILTER_VALIDATE_EMAIL) === FALSE)
{
//throw exception if email is not valid
throw new customException($email);
}
//check for "example" in mail address
if(strpos($email, "example") !== FALSE)
{
throw new Exception("$email is an example e-mail");
}
}
catch (customException $e)
{
echo $e-errorMessage();
}
catch(Exception $e)
{
echo $e-getMessage();
}
?
例子解釋:上面的代碼測試了兩種條件,如何任何條件不成立,則拋出一個異常.
1.customException() 類是作為舊的 exception 類的一個擴展來創(chuàng)建的,這樣它就繼承了舊類的所有屬性和方法.
2.創(chuàng)建 errorMessage() 函數(shù),如果 e-mail 地址不合法,則該函數(shù)返回一個錯誤消息.
3.執(zhí)行 "try" 代碼塊,在第一個條件下,不會拋出異常.
4.由于 e-mail 含有字符串 "example",第二個條件會觸發(fā)異常.
5."catch" 代碼塊會捕獲異常,并顯示恰當(dāng)?shù)腻e誤消息.
如果沒有捕獲 customException,緊緊捕獲了 base exception,則在那里處理異常,重新拋出異常,有時,當(dāng)異常被拋出時,您也許希望以不同于標(biāo)準(zhǔn)的方式對它進行處理,可以在一個 "catch" 代碼塊中再次拋出異常,代碼如下:
總結(jié):PHP異常的使用方法分三步:
第一步:定義異常類,如果不定義就用系統(tǒng)默認(rèn)的異常類;
第二步:當(dāng)出現(xiàn)異常時用 throw 拋出異常,例如 ex1($num2);異常的參數(shù)是$num2用該異常的getMessage()獲取;
第三步:觸發(fā)異常,用try子句,當(dāng)滿足條件時 throw new ex1($num);
第四步:catch捕獲異常 catch (ex2 $e),相當(dāng)于實例化一個定義好的異常類ex2為$e;
注意,異??梢远x多個,但是只能觸發(fā)一個,也就是說只能用catch捕獲一個異常.
基本異常類,創(chuàng)建可拋出一個異常的函數(shù):
復(fù)制代碼 代碼如下:
function num($num){
if ($num1){//異常拋出條件
$msg="數(shù)值不能大于1″;//異常提示信息
throw new Exception($msg);//拋出異常
}
echo "數(shù)值小于1″;
}
//在 "try" 代碼塊中觸發(fā)異常
try {
num(3);
echo "執(zhí)行正常";
}
//捕獲異常
catch (Exception $e){
echo "錯誤信息:".$e-getMessage();//Exception()的系統(tǒng)方法獲取異常信息
echo "錯誤文件:".$e-getFile();//Exception()的系統(tǒng)方法獲取異常文件名
echo "行數(shù):".$e-getLine();//Exception()的系統(tǒng)方法獲取異常行數(shù)
}
//======================================================================
echo "br========================================================br";
//擴展基本異常類
function checkEmail($email){//定義一個可以拋出異常的判斷EMAIL合法性的函數(shù)
if (filter_var($email,FILTER_VALIDATE_EMAIL)==false){
throw new checkEmailException($email);//拋出異常用EMAIL做參數(shù)
}
echo "郵件合法";
}
class checkEmailException extends Exception{//定義擴展異常類
public function errormsg(){
$msg="錯誤原因:".$this-getMessage()."不是一個合法的EMAIL地址!";
$msg.="錯誤文件名:".$this-getFile();
$msg.="錯誤行數(shù):".$this-getLine();
echo $msg;
}
}
$email="email…..@chhua.com";
try {//觸發(fā)異常
checkEmail($email);
}
//捕獲異常
catch (checkEmailException $e){
$e-errormsg();
}
//==================================多個異常的捕獲
echo "br===================================================br";
class ex1 extends Exception{//定義一個異常類
public function msg(){
$msg="錯誤原因:".$this-getMessage()."大于100br";
$msg.="錯誤文件:".$this-getFile()."Br";
$msg.="錯誤代碼:".$this-getCode()."br";
$msg.="行數(shù):".$this-getLine()."br";
echo $msg;
}
}
class ex2 extends Exception{//定義一個異常類
public function msg(){
$msg="錯誤原因:".$this-getMessage()."等于100br";
$msg.="錯誤文件:".$this-getFile()."Br";
$msg.="行數(shù):".$this-getLine()."br";
echo $msg;
}
}
$num2=100;
try {
if ($num2100){//當(dāng)條件滿足時觸發(fā)
throw new ex1($num2);
}
if ($num2==100){//當(dāng)條件滿足時觸發(fā)
throw new ex2($num2);
}
}
catch (ex2 $e){//捕獲觸發(fā)的異常
$e-msg();
}
catch (ex1 $e){//捕獲觸發(fā)的異常
$e-msg();
}
使用for循環(huán)
定義變量$i,配合select * from table where ………… limit $i,100
這樣就可以了
我之前就做過,但因為換了工作,代碼在之前的公司,否則就可以讓你參考下了
另外,我覺得100太少了,最起碼得改成1000才行