小編剛剛接觸微信開發,一直想做一個自己能查詢課表和成績的微信功能,在研究了些許時間,故在這里和大家一起分享一下。因為平常小編課程比較多,這篇文章可能寫的很倉促,代碼寫的肯定不是很規范,還希望大家多交流指點。
設計思路:先登錄頁面獲取COOKIES,然后拿著cookies找服務器要驗證碼。最后提供服務器需要的全部信息。其實通俗的講就是我們模擬瀏覽器訪問界面,我們用cookies去代替人工去登錄教務系統。
1、CURL的PHP請求跟你游覽器發出的PHP屬于兩個不同的線程,所以,我們采用CURL這個庫來進行我們功能的實現。
2、sessionID的本質
客戶端用cookie保存了sessionID
客戶端用cookie保存了sessionID,當我們請求服務器的時候,會把這個sessionID一起發給服務器,服務器會到內存中搜索對應的sessionID,如果找到了對應的 sessionID,說明我們處于登錄狀態,有相應的權限;如果沒有找到對應的sessionID,這說明:要么是我們把瀏覽器關掉了(后面會說明為什 么),要么session超時了(沒有請求服務器超過20分鐘),session被服務器清除了,則服務器會給你分配一個新的sessionID。你得重 新登錄并把這個新的sessionID保存在cookie中。
在沒有把瀏覽器關掉的時候(這個時候假如已經把sessionID保存在cookie中了)這個sessionID會一直保存在瀏覽器中,每次請求的時候都會把這個sessionID提交到服務器,所以服務器認為我們是登錄的;當然,如果太長時間沒有請求服務器,服務器會認為我們已經所以把瀏覽器關掉了,這個時候服務器會把該sessionID從內存中清除掉,這個時候如果我們再去請求服務器,sessionID已經不存在了,所以服務器并沒有在內存中找到對應的 sessionID,所以會再產生一個新的sessionID,這個時候一般我們又要再登錄一次。
客戶端沒有用cookie保存sessionID
客戶端沒有用cookie保存sessionID這個時候如果我們請求服務器,因為沒有提交sessionID上來,服務器會認為你是一個全新的請求,服務器會給你分配一個新的sessionID,這就是為什么我們每次打開一個新的瀏覽器的時候(無論之前我們有沒有登錄過)都會產生一個新的sessionID(或者是會讓我們重新登錄)。
當我們一旦把瀏覽器關掉后,再打開瀏覽器再請求該頁面,它會讓我們登錄,這是為什么?我們明明已經登錄了,而且還沒有超時,sessionID肯定還在服 務器上的,為什么現在我們又要再一次登錄呢?這是因為我們關掉瀏覽再請求的時候,我們提交的信息沒有把剛才的sessionID一起提交到服務器,所以服務器不知道我們是同一個人,所以這時服務器又為我們分配一個新的sessionID,打個比方:瀏覽器就好像一個要去銀行開戶的人,而服務器就好比銀行, 這個要去銀行開戶的人這個時候顯然沒有帳號(sessionID),所以到銀行后,銀行工作人員問有沒有帳號,他說沒有,這個時候銀行就會為他開通一個帳號。所以可以這么說,每次打開一個新的瀏覽器去請求的一個頁面的時候,服務器都會認為,這是一個新的請求,他為你分配一個新的sessionID。
get_headers() 是PHP系統級函數,他返回一個包含有服務器響應一個 HTTP 請求所發送的標頭的數組。如果失敗則返回 FALSE 并發出一條 E_WARNING 級別的錯誤信息(可用來判斷遠程文件是否存在)。
函數定義
array get_headers ( string $url [, int $format = 0 ] )
參數
url 目標 URL
format 如果將可選的 format 參數設為 1,則 get_headers() 會解析相應的信息并設定數組的鍵名。
廢話不多說上代碼:
class TestController extends Controller {
public function index() {
$url = 'http://教務處登錄地址';
$u = get_headers ( $url, 1 );
$login_url = 'http://教務處登錄地址' . dirname ( $u ['Location'] ) . '/default2.aspx';
$hiden = $this->getView ( $login_url );
$post_fields = array (
'txtUserName' => '學號',
'TextBox2' => '密碼',
'RadioButtonList1' => '學生',
'__VIEWSTATE' => $hiden,
"Button1" => " ?登錄 ?"
);
$cookie_file = tempnam ( 'Public', 'cookie' );
$ch = curl_init ( $login_url );
curl_setopt ( $ch, CURLOPT_HEADER, 0 ); // 顯示頭部
curl_setopt ( $ch, CURLOPT_POST, 1 ); // post傳遞
$header [] = 'User-Agent:Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/34.0.1847.116 Safari/537.36';
curl_setopt ( $ch, CURLOPT_HTTPHEADER, $header ); // 模擬客戶端
curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, 0 ); // 顯示頁面
curl_setopt ( $ch, CURLOPT_POSTFIELDS, $post_fields ); // post傳遞值
curl_setopt ( $ch, CURLOPT_COOKIEJAR, $cookie_file ); // 存入cookie
curl_setopt ( $ch, CURLOPT_FOLLOWLOCATION, 1 ); // 允許重定向
curl_exec ( $ch );
curl_close ( $ch );
// 獲取課表
$class_url = '教務處登錄地址' . dirname ( $u ['Location'] ) . '/xskbcx.aspx?xh=' . '學號';
$main_url = '教務處登錄地址' . dirname ( $u ['Location'] ) . '/xs_main.aspx?xh=' . '學號';
$this->getClass ( $class_url, $main_url );
}
// 登陸頁面的隱藏字段
private function getView($url) {
$result = curl_request ( $url );
$pattern = '//is';preg_match_all ( $pattern, $result, $matches );
$res [0] = $matches [1] [0];
return $res [0];
}
// 查詢課表
public function getClass($url, $main_url) {
$cookie_file = tempnam ( 'Public', 'cookies' );
$ch = curl_init ( $url );
curl_setopt ( $ch, CURLOPT_TIMEOUT, 60 );
curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, 1 );
curl_setopt ( $ch, CURLOPT_REFERER, $main_url );
curl_setopt ( $ch, CURLOPT_COOKIEJAR, $cookie_file ); // 存入cookie
$str = curl_exec ( $ch );
$this->classResult ( $str );
}
// 返回課表字符串
private function classResult($result) {
preg_match_all ( '/([\\w\\W]*?)<\\/table>/', $result, $out );
$table = $out [0] [0];
// 獲取整個課表
//print_r($table);
preg_match_all ( '/([\\w\\W]*?)<\\/td>/', $table, $out );$td = $out [1];$length = count ( $td ); //print_r($td) ;
// 獲得課程列表
for($i = 0; $i < $length; $i ++) {
$td [$i] = str_replace ( "", "", $td [$i] );
$reg = "/{(.*)}/";
if (! preg_match_all ( $reg, $td [$i], $matches )) {
unset ( $td [$i] );}
}
$td = array_values($td); //將課程列表數組重新索引$tdLength = count($td);$this->converttoTable ( $td );
}
}