2021年3月5日 星期五

php 同時傳送超過1000個變數時的做法


php 同時傳送超過1000個變數時,比方
  1. <form id="myForm">
  2. <?php
  3. for ($i=0; $i <1200 ; $i++) {
  4. echo '<button type="submit">送出</button>';
  5. }
  6. ?>
  7. <form>
此時會報錯誤
  1. PHP Warning: Unknown: Input variables exceeded 1000. To increase the limit change max_input_vars in php.ini.
雖然可以照它說的去更改 max_input_var 的數字,但終歸無法一牢永逸(因為我的變數一直在增加)
所以上網找到這個解法:將1000個變數的值集中到一個變數中,接收後再拆開來使用


在 html 的部份,加上一變數,在送出前集中所有變數的值到此一變數
  1. <input name="storeids" type="hidden" />
  2. <button type="submit" onclick="changeidstostring();">送出</button>
  1.  
在 js 的部份這樣寫
  1. function changeidstostring() {
  2. $('#myForm').submit(function() {
  3. var storeids = $("input[name='ids[]']").map(function(){return $(this).val();}).get();
  4. $("input[name='storeids']").val(storeids);
  5. $("input[name='ids[]']").attr("disabled","disabled"); // 將原來會產生很多變數的元素disabled 掉,就不會傳送了
  6. });
  7. }
在接收的部分這樣寫
  1. $ids = explode(",", $_POST['storeids']); // 因為參數太多(超過1000個,所以改成先集中到某個參數,傳送過來再拆解)

2019年8月27日 星期二

PHP PDO IN 條件

使用PHP PDO 在組合查詢語法時若有 IN 條件,通常會用此方法
  1. $myArray = array(1,2,3);
  2. $in = trim(str_repeat('?,', count($myArray)),',');
  3. $stm = $conn->prepare("SELECT * FROM table WHERE id IN ($in)");
  4. $stm->execute($myArray);
萬一遇上 $myArray 的個數很多時,會造成語法太長,無法執行怎麼辦?
試試下面這個方法
  1. $myArray = array(1,2,3);
  2. $in = implode(',', $myArray);
  3. $stm = $conn->prepare("SELECT * FROM table WHERE FIND_IN_SET(id, :inArray) ORDER BY name");
  4. $stm->bindParam('inArray', $in);
  5. $stm->execute();
如果是 not in,就加上一驚嘆號,變成 !FIND_IN_SET 就可以了
  1. $myArray = array(1,2,3);
  2. $in = implode(',', $myArray);
  3. $stm = $conn->prepare("SELECT * FROM table WHERE !FIND_IN_SET(id, :inArray) ORDER BY name");
  4. $stm->bindParam('inArray', $in);
  5. $stm->execute();

2019年7月9日 星期二

使用 Gmail 的兩步驟認證下,透過 PHPMailer 發送郵件範例

到google 設定兩步驟認證
1.登入google後,找到 google帳戶,在"安全性"下會看到如下圖
未啟用兩步驟驗證的畫面

2.選擇上圖中的"兩步驟驗證"後,按下"開始使用"後,依照步驟操作完成,會得到一組16位數的密碼,此密碼會用於步驟三中的
  1. $mail->Password = '密碼';

程式碼配置與發信程式撰寫(在沒有 composer 的環境下)
一、程式碼下載
1.到 github 上下載程式碼:PHPMailer,並解壓縮
解壓縮後資料夾內容
2.將 PHPMailer 資料夾放在自己知道的路徑下,比方:/var/www/mySite/PHPMailer

二、發信程式碼撰寫
  1. <?php
  2. try {
  3. /* Set the mail encode. */
  4. CharSet = 'UTF-8';
  5.  
  6. /* Set the mail sender. */
  7. $mail->setFrom('寄件人@gmail.com', '寄件人名稱');
  8.  
  9. /* Add a recipient. */
  10. $mail->addAddress('收件人郵件地址', '收件人名稱');
  11.  
  12. /* Set the subject. */
  13. $mail->Subject = '主旨';
  14.  
  15. /* Set the mail message body. */
  16. $mail->Body = '信件內容';
  17.  
  18. /* SMTP parameters. */
  19. /* Tells PHPMailer to use SMTP. */
  20. $mail->isSMTP();
  21. /* SMTP server address. */
  22. $mail->Host = 'smtp.gmail.com';
  23.  
  24. /* Use SMTP authentication. */
  25. $mail->SMTPAuth = TRUE;
  26. /* Set the encryption system. */
  27. $mail->SMTPSecure = 'tls';
  28. /* SMTP authentication username. */
  29. $mail->Username = '寄件人@gmail.com';
  30. /* SMTP authentication password. google 兩步驟認證的密碼*/
  31. $mail->Password = '密碼';
  32. /* Set the SMTP port. */
  33. $mail->Port = 587;
  34.  
  35. /* Finally send the mail. */
  36. $mail->send();
  37.  
  38. echo "成功發送信件!";
  39. }
  40. catch (Exception $e)
  41. {
  42. echo "發送失敗!";
  43. /* PHPMailer exception. */
  44. echo $e->errorMessage();
  45. }
  46. catch (\Exception $e)
  47. {
  48. echo "發送失敗!";
  49. /* PHP exception (note the backslash to select the global namespace Exception class). */
  50. echo $e->getMessage();
  51. }
  52. ?>

參考資料/來源: https://alexwebdevelop.com/phpmailer-tutorial/ 阿力獅的教室

2019年3月28日 星期四

php 將接收 POST 來的資料放到 html 元素中要注意的地方

事情是這樣的,因為某些因素,要 POST 的資料包含 HTML 中的 TAG 符號,如下
  1. $str = "<option value="1">student</option><option value="2">teacher</option><option value="3">teachers' office</option>";
在接收資料時,要將它還原成一下拉選單,如下
  1. <select id="select_1">
  2. <option value="1">student</option>
  3. <option value="2">teacher</option>
  4. <option value="3">teachers' office</option>
  5. </select>
但是因為文字中有特殊符號  ' ,造成只能顯示
<select id="select_1">
<option value="1">student</option>
<option value="2">teacher</option>
<option value="3">teachers

上網 google 了一下後,修改方式:
在 POST 前用 htmlentities 將資料包起來,這樣接收時就可以正常了


參考資料/來源: [PHP] HTML特殊字元轉換

2019年1月29日 星期二

php 的日期比較

在比對之前,先說明一下如何得到目前日期 在 php 5.2(含) 前可以用 date('Y-m-d') 在 php 5.2(不含) 後可以用 new DateTime() 雖然函數有一點點不同,但都需要注意時區可能產生的問題

接下來開始比較日期
使用函數 strtotime
  1. $Date1 = date('Y-m-d');
  2. $Date2 = '9999-12-31';
  3. if(strtotime($Date1) > strtotime($Date2)) {
  4. echo 'Date1 較新';
  5. }
  6. else {
  7. echo 'Date2 較新';
  8. }
基本上這樣就可以了,但是若是使用XAMPP,上面的語法可能會出現錯誤,原因在於 Y2K38漏洞,也被稱為Unix Millennium Bug,也就是 strtotime 只認得 2038-1-19 03:14:07 之前的日期時間,在此日期時間後的就會溢位,進而產生錯誤。

所以將語法修改如下,順便用了 php 5.2 版之後的語法
  1. $Date1 = date_format(new DateTime(),'U');
  2. // $Date2 = date_format(new DateTime('9999-12-31'),'U');
  3. // 這邊需要注意因為 DateTime()會包含時間的部分,所以在比較時要同時加上時間的部分(或去除時間的部分)來比較會較正確,
  4. // 或是將 $Date2 多加一天後再比較也行,所以將 $Date2 修改一下
  5. $Date2 = date_format(date_modify(new DateTime('9999-12-31'),'+1 days'),'U');
  6. if($Date1 > $Date2) {
  7. echo 'Date1 較新';
  8. }
  9. else {
  10. echo 'Date2 較新';
  11. }
參考資料/來源: 藍色小舖 php 討論區- 日期的比較 PHP轉換超過2038年的日期出錯問題解決

用Temp Table取代Cursor

cursor通常用來逐筆處理資料使用,下面是一個簡單的範例(MS SQL Server 2008)
  1. declare @myId int
  2. declare @myName nvarchar(20)
  3. declare @myCursor CURSOR
  4.  
  5. set @myCursor = CURSOR FAST_FORWARD
  6. FOR
  7. SELECT ID, NAME FROM Employee
  8. open @myCursor
  9. INTO @myId, @myName
  10. WHILE @@FETCH_STATUS = 0
  11. BEGIN
  12.  
  13. --DO SOMETHING
  14.  
  15. FETCH NEXT FROM @myCursor
  16. INTO @myId, @myName
  17.  
  18. END
  19.  
  20. CLOSE @myCursor
  21. DEALLOCATE @myCursor
用temp table來模擬cursor的操作模式,簡單的說就是把SELECT ID, NAME FROM Employee的結果存到temp table中,在逐步讀取,所以必須先建立一個temp table如下
  1. create table #tempEmployee
  2. (
  3. ID int,
  4. NAME nvarchar(20)
  5. )
然後把資料insert into select 到 #tempEmployee
  1. insert into #tempEmployee (ID,NAME)
  2. select ID,NAME
  3. from Employee
接下來,就是逐步讀取這個temp table #tempEmployee
在MS SQL Server中使用SET ROWCOUNT 1來控制select時一次只撈出一筆資料,而且每做完一筆就刪掉,直到#tempEmployee沒有資料為止
  1. declare @countTemp int --用來計算#tempEmployee還剩幾筆資料
  2.  
  3. --計算#tempEmployee資料數
  4. select @countTemp = count(*) from #tempEmployee
  5.  
  6. while(@countTemp > 0)
  7. begin
  8. set rowcount 1
  9. select @myId = ID, @myName = NAME from #tempEmployee
  10. --To Something
  11. --因為set rowcount 1的關係,所以一次只會刪一筆
  12. delete from #tempEmployee
  13.  
  14. --計算還剩幾筆,@countTemp > 0繼續
  15. select @countTemp = count(*) from #tempEmployee
  16. end
  17.  
  18. --#刪除 tempEmployee
  19. drop table #tempEmployee
  20.  
  21. --把select的預設比數恢復正常
  22. set rowcount 0
如此一般,完整的程式碼如下
  1. declare @myId int
  2. declare @myName nvarchar(20)
  3.  
  4. create table #tempEmployee
  5. (
  6. ID int,
  7. NAME nvarchar(20)
  8. )
  9.  
  10. declare @countTemp int --用來計算#tempEmployee還剩幾筆資料
  11.  
  12. --計算#tempEmployee資料數
  13. select @countTemp = count(*) from #tempEmployee
  14.  
  15. while(@countTemp > 0)
  16. begin
  17. set rowcount 1
  18. select @myId = ID, @myName = NAME from #tempEmployee
  19. --To Something
  20. --因為set rowcount 1的關係,所以一次只會刪一筆
  21. delete from #tempEmployee
  22.  
  23. --計算還剩幾筆,@countTemp > 0繼續
  24. select @countTemp = count(*) from #tempEmployee
  25. end
  26.  
  27. --#刪除 tempEmployee
  28. drop table #tempEmployee
參考資料/來源: 遜砲賴的爆肝筆記-stored procedure中不使用cursor逐步讀取資料列的方法

2018年4月29日 星期日

不爽 0002

總: 董事長剛吃完飯,等等讓他吃藥
我: 好(看了一下桌面),只有一個杯子?
總: 對,你到底有沒有在顧(用心)?飯後都只有一杯藥
我: 有啊(心想早上明明看到2個杯子的藥,還是我看錯了?)
(之後)
總: 你...
我: 嗯(不想回答有意義的文字了)


(中午,問了一下護理師)
我: 請問早上餐後的藥是一杯還兩杯?
護: 兩杯呵,一杯裡面三顆藥,一杯喝的
我: 謝謝。(心想,掯!早上總經理是在唸不爽的嗎?到底是誰沒在顧?)