當前位置:首頁文章首頁 新聞中心

國產WindowsPhone 7 PC端實時截圖工具制作進程

作者:  來源:  發(fā)布時間:2011-6-24 0:02:56  點擊:

三、圖片傳出去

我們已經解決了截屏和后臺運行的問題,下面怎么把截屏下來的內容顯示到PC上呢?肯定使用網絡了。簡單測試以后非常開心,只要把手機連接到電腦上,并且啟動Zune,將電腦和手機設置為Connected狀態(tài),這樣手機就可以連接電腦上的網絡服務了,還可以通過電腦來訪問外網,無需Wifi。

因為Mango之前的WP7只支持通過WebClient訪問Http網絡,無法編寫普通Socket程序(除非用上面的方法調用Dll),因此服務器端我用一個Http服務器接收客戶端提交上來的截圖圖片流然后顯示出來即可,服務器端我用了一個開源的嵌入式http服務器“C# WebServer”(本地下載包),這樣避免部署IIS,直接啟動WinForm程序就能運行服務器,手機端通過WebClient來提交截屏的數(shù)據(jù)。服務器端和客戶端的實現(xiàn)都是普通的.Net操作,不再詳細講,感興趣的可以參考傳智播客的.Net免費視頻教程。

編寫手機端程序,定時截取圖片傳到服務器,服務器顯示正常。偶耶!

依照我多年的開發(fā)經驗,我感覺這么順的就實現(xiàn)了絕對不正常,果然發(fā)現(xiàn)了問題。在截圖程序中進行的操作都能截屏顯示出來,但是離開程序后截屏過程竟然停止了,畫面一直停在離開前的畫面上。我勒個去,難道"修改注冊表讓WP7支持多任務的方法"只是一個傳說?但是一調試就傻了,定時截屏的任務依然在忠實的執(zhí)行著,也就是截屏程序確實是在后臺執(zhí)行的。這是為什么呢?

熟悉Silverlight、Windows Phone的朋友知道,SL/WP7中的WebClient操作全部是異步的,沒有普通.net中DownloadData()、UploadData()等阻塞方法,例如:

WebClient wc = new WebClient();

wc.OpenWriteCompleted+=new OpenWriteCompletedEventHandler(wc_OpenWriteCompleted);

wc.OpenWriteAsync(Uri);

OpenWriteAsync執(zhí)行完畢后上傳操作并沒有立即開始,而是wc_OpenWriteCompleted這個異步方法中執(zhí)行。經過調試發(fā)現(xiàn)當離開手機程序以后,雖然后臺線程還在運行,但是wc_OpenWriteCompleted方法不會再被觸發(fā)。我猜測Silverlight中也是通過類似于Win32中的WM_TIMER事件實現(xiàn)的消息泵來進行任務的所謂“異步執(zhí)行”的,當離開程序以后UI線程就暫停,所以異步操作就執(zhí)行不了了。經過一通的反編譯、查資料,發(fā)現(xiàn)確實如此,Dispatcher是靠DispatcherTimer來實現(xiàn)的。

沒辦法了嗎?No!向灰太狼先生學習“我一定會回來的”!不要以為這樣就能攔住我!消息泵停了我能給你來一個“心臟起搏器”強迫消息泵繼續(xù)運轉,經過反編譯、反射發(fā)現(xiàn)Dispatcher有一個私有的Dispatch方法用來驅動消息泵,我不斷的調用Dispatch方法不就可以強迫消息泵運轉了嗎?有人問“私有方法不是不能調用嗎?”,一切限制都是給別人設置的,堅強的灰太狼總能想到解決方案。“通過反射調用私有成員”這已經是一個非常常見的一個高級技術了,可以解決很多常規(guī)手段解決不了的問題。當然如果你不了解如何實現(xiàn)“通過反射調用私有成員”這又想學習的朋友可以關注傳智播客的.Net培訓,其中“.Net 高級技術”的課程就講到了這些。

編寫類似于下面的代碼來調用私有方法吧!

var m = d.GetType().GetMethod("Dispatch", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);

mInvoke(d, new object[] {});

不幸的是調用過程發(fā)生異常,經過查詢資料得知一個不幸的消息,基于安全考慮,Windows Phone中不能通過反射調用私有成員。

看來通過WebClient發(fā)送截圖的思路失敗了,幸好能夠通過前面說的方法調用Native的代碼,我開啟VC++調用socket api編寫了提交Http二進制數(shù)據(jù)的一個Com組件,沒想到自認為TCP搞的還算不錯的我竟然在這個“小環(huán)節(jié)”上載了跟頭,連續(xù)熬了兩個晚上調試Http發(fā)送圖片流都沒有調試成功,總是發(fā)送前幾幀能顯示后面的數(shù)據(jù)就全部亂掉了,沒過多久還報了10061之類經典的Socket錯誤碼。而把同樣的代碼放到Windows XP下編譯運行就沒問題。在無數(shù)次“我的Socket代碼的錯?”、“Windows CE Socket有特殊的地方?”、“凌晨三點了,快點睡吧”、“差一點就好了,再熬一會”的掙扎中,我最終放棄了。改為調用高度封裝wininet中的HttpOpenRequest等方法來實現(xiàn),很快就運行通過了。至今沒有搞清楚到底是我的代碼的問題還是Windows CE的Socket有特殊的地方,如果是我代碼的原因那么可能就是沒有處理好大數(shù)據(jù)量的緩沖區(qū)、“Http 1.1/ 100 continue” 等細節(jié)問題。還好通過HttpOpenRequest這種方法實現(xiàn)了,期待網絡編程牛人的指導。

文章評論

軟件按字母排列: A B C D E F G H I J K L M N O P Q R S T U V W X Y Z