利用 Win32 啟動和檢測 UWP App 的方法

来源:https://www.cnblogs.com/cjw1115/archive/2018/04/26/8953065.html
-Advertisement-
Play Games

一種啟動和檢測 UWP 應用的方法 背景 我們發佈過多款 UWP 平臺的同類型 App ,最近有一個需求:用傳統 Win32 程式啟動我們的 UWP 程式。因為我們的每一個UWP App在客戶機器上都是互斥的,也就是同時只能存在一個,並且我們的win32程式也只有一個版本,所以啟動 UWP App ...


一種啟動和檢測 UWP 應用的方法

背景

我們發佈過多款 UWP 平臺的同類型 App ,最近有一個需求:用傳統 Win32 程式啟動我們的 UWP 程式。因為我們的每一個UWP App在客戶機器上都是互斥的,也就是同時只能存在一個,並且我們的win32程式也只有一個版本,所以啟動 UWP App 時,需要先檢測,再啟動。

我們大概有4個辦法,前3個比較扯,第4個目前可行,也是我們採用的。這4個方法的主要關註點是:如何檢測客戶機器上是否有我們的 UWP App。至於調用,方法比較簡單。

Solution 1

Win32 和 UWP 交互,首先想到的就是微軟的 Desktop Bridge 相關的內容,找了一圈,倒是發現了 Win32 調用 UWP Api 的方法,不過可以調用的 Api 有限,而且文檔比較殘缺,最麻煩的就是要對 Win32 Project 配置修改,引入一堆 WinRT 的東西。嘗試了半天,終於不報錯了,但是運行時會奔潰,原因未知,有待繼續探索。而且比較存疑的是官方文檔有矛盾,我們用到的 Windows.System.Launcher Api 是否被這種調用方式支持不明確,因為報錯我們也無法驗證。

有興趣的小伙伴可以參考以下鏈接:

Desktop Bridge

Detect UWP App

Solution 2

簡單粗暴,直接檢測 UWP 的安裝目錄。一般 UWP 的預設安裝路徑就是 "C:\Program Files\WindowsApps"。這種方法真的很簡單粗暴,但是有幾個缺點:

  1. 可能有強迫症用戶修改了 UWP 的安裝路徑。這種情況下,需要自行去查註冊表,當然註冊表鍵值是什麼就需要baidu了;
  2. 如果直接枚舉 "C:\Program Files\WindowsApps"的子目錄,會有許可權問題(System),普通用戶許可權只能訪問類似 "C:\Program Files\WindowsApps\microsoft.windowscommunicationsapps_17.9126.21695.0_x64__8wekyb3d8bbwe"的特定 UWP App 目錄,這就需要我們提前確定要查找的 UWP App的 pfn (package family name,UWP App 的特定標識,全球唯一)和版本,但是版本因為經常變化,比較不好確定。

Solution 3 (Solution 1和這個差不多)

微軟為我們提供了許多啟動 UWP 的方式,比如什麼協議啟動,命令行啟動等,但是這些方法的使用前提是:我們的UWP app需要修改現有的 App Manifest,這對於已經發佈出去的UWP App,顯然是不可能的。(在我們的場景下,因為我們的 UWP App 和驅動綁定,一般隨驅動升級,比較穩定,所以此方法不可用)

Solution 4 (Best solution)

隱約記得以前使用 Fiddler 的時候,有一個 WinConfig 功能,可以列出當前電腦上所有的 UWP 程式(實際上是 沙箱類程式,從 Windows 8 開始, UWP 也包含其中),然後可以進行 web 調試。所以就想能不能借鑒 Fiddler 的做法。然後理所應當的發現 Fiddler 安裝目錄下麵有一個名為 EnableLoopback.exe 的程式,沒有為什麼,我就把它丟到了ILSpy裡面,完美的反編譯出了C#代碼,然後經過一番探索,發現了AppContainer類,無論看類名還是類的定義,都很明確,這就是我們要找的東西,然後順著這個類看下去,找到了它獲取所有 UWP 程式的方法:通過 FirewallAPI.dll 裡面的介面 NetworkIsolationEnumAppContainers 來枚舉。

AppContainer

有了瞭解,開始Coding!

BTW,如果想省事兒的話,直接把這個類相關的內容導出,是可以直接用的。不過我們的 Win32 是用C++寫的,所以要稍稍轉換一下。

C++代碼如下:

#include <Netfw.h>
#include <string>
#include <vector>
#include <algorithm>
using namespace std;
namespace Launcher
{
    typedef DWORD(*pNetworkIsolationEnumAppContainers)(
        _In_  DWORD                        Flags,
        _Out_ DWORD                        *pdwNumPublicAppCs,
        _Out_ PINET_FIREWALL_APP_CONTAINER *ppPublicAppCs
        );
    typedef DWORD(*pNetworkIsolationFreeAppContainers)(
        _In_ PINET_FIREWALL_APP_CONTAINER pPublicAppCs
        );
    void LaunchSpecifcApp(wstring *pfn)
    {
        TCHAR szCommandLine[1024];
        wsprintf(szCommandLine, L"explorer.exe shell:AppsFolder\\%ws!App", (*pfn).c_str());
        STARTUPINFO si;
        PROCESS_INFORMATION pi;
        ZeroMemory(&si, sizeof(si));
        si.cb = sizeof(si);
        ZeroMemory(&pi, sizeof(pi));

        si.dwFlags = STARTF_USESHOWWINDOW;
        si.wShowWindow = TRUE;

        BOOL bRet = ::CreateProcess(NULL, szCommandLine, NULL, NULL, FALSE, CREATE_NEW_CONSOLE, NULL, NULL, &si, &pi);

    }

    void LaunchUWPApp()
    {
        vector<wstring> uwpApps;
        uwpApps.push_back(L"microsoft.windowscommunicationsapps_8wekyb3d8bbwe");

        HMODULE FirewallAPIModule;
        FirewallAPIModule = (LoadLibrary(L"FirewallAPI.dll"));

        auto EnumAppContainersProc = pNetworkIsolationEnumAppContainers(GetProcAddress(FirewallAPIModule, "NetworkIsolationEnumAppContainers"));
        auto FreeAppContainersProc = pNetworkIsolationFreeAppContainers(GetProcAddress(FirewallAPIModule, "NetworkIsolationFreeAppContainers"));

        DWORD pdwNumPublicAppCs = 0;
        PINET_FIREWALL_APP_CONTAINER ppPublicAppCs = NULL;
        HRESULT re = EnumAppContainersProc(0, &pdwNumPublicAppCs, &ppPublicAppCs);

        for (int i = 0; i < pdwNumPublicAppCs; i++)
        {
            auto appContainer = ppPublicAppCs[i];
            for (int j = 0; j < uwpApps.size(); j++)
            {
                auto app = uwpApps.at(j);
                transform(app.begin(), app.end(), app.begin(), tolower);
                if (app == appContainer.appContainerName)
                {
                    //launch it;
                    auto temp = uwpApps.at(j);
                    LaunchSpecifcApp(&temp);
                }
            }
        }
        FreeAppContainersProc(ppPublicAppCs);
        FreeLibrary(FirewallAPIModule);
        vector<wstring>().swap(uwpApps);
    }
}

代碼很直白,裡裡面就兩個函數,一個用來查找,一個用來啟動,額外用到的就是 Win32 Dll 調用相關的內容了。

最後

可以看到,我們查找 UWP 比較麻煩,但是調用卻很簡單,核心就是:

"explorer.exe shell:AppsFolder\\{pfn}!App"

很直白,赤裸裸的一個快捷方式呀!但是有坑,如果傳遞的參數有任何問題(要麼拼錯了,要麼不存在),explorer 會直接忽略參數,把自己啟動。這種行為,對於不明真相的用戶,會很莫名其妙,垃圾軟體。所以我們在啟動我們的 UWP App 時,要確保這個我們的 App 一定存在於用戶的電腦上面,所以才有了上面檢測 UWP App 的邏輯。如果參數錯誤,explorer 啥也不敢的話,我們就不這麼麻煩了,可以直接把我們所有的 UWP app 挨個啟動一遍,簡單粗暴!

最後的最後

我們用到了 Fillder 裡面所使用的方法,但對於 Fiddler 版權的各種問題,個人不瞭解。好在我們直接用 C++ 實現,沒有任何影響。 權當學習學習!

之前網上有 Fiddler 2.x版本的源碼,但不清楚這軟體是不是開源。

致敬 Fiddler !


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • [可迭代對象]保存的是已經生成好的數據,占用大量的空間有__iter__方法 就是可迭代對象(Iterable) [迭代器]保存的是生成數據的方法,占用極小的空間,需要時才返回數據既有__iter__,也有__next__ 就是迭代器(Iterator) [生成器]是一個[特殊]的迭代器,保存生成數 ...
  • 最近在學習redis,雖然現在還不是很熟練。不過可以進行簡單的框架整合開發。那麼我就把我的springmvc整合redis的過程分享給大家。 IDE:我使用的是IDEA。 首先看一下我的工程結構: 第一步:在pox.xml中引入jar 第二步:配置web.xml 第三步:配置springmvc.xm ...
  • 之前跟大家講的是一些python的數據基礎,從這篇文章開始,我們開始正式學習python的模塊化編程 下麵我們解釋一下什麼叫做模塊 之前已經講過怎麼去定義一個方法,如果你是用python交互器(自帶交互器,或者是ipython)來學習定義方法的話,你定義完方法後,然後退出交互器,然後你再來使用這個方 ...
  • 該文章是圖說Java系列文章中的一篇 substring(int beginIndex, int endIndex)方法在jdk 6和jdk 7中的實現是不同的。瞭解他們的區別可以幫助你更好的使用他。為簡單起見,後文中用substring()代表substring(int beginIndex, i ...
  • 1、標示符由字母、下劃線和數字組成,且數字不能開頭 12mytest 錯 mytest 對 MyText (大駝峰式命名法) myText (小駝峰式命名法) my_test 個人喜歡這種命名方法 不可以使用關鍵字命名: 在這個推薦一個強大的python工具 ipython,安裝python環境後 ...
  • 題目鏈接:http://uoj.ac/problem/239 題目大意: 這是一道交互題,交互庫維護了一個數據結構,可以存儲n為二進位串,一開始你可以向空的數據結構中插入若幹二進位串, 接下來這個數據結構會將其中存儲的二進位串進行改變。 改變的方法是生成一個0到n-1的排列pi,將原來的二進位串a0 ...
  • 一、Hibernate框架 Hibernate是一個開放源代碼的對象關係映射框架,它對 JDBC進行了非常輕量級的對象封裝,它將POJO類與資料庫表建立映射關係,是一個 全自動的ORM框架,hibernate可以自動生成SQL語句,自動執行。Hibernate可以應用在任何使用 DBC的場合,既可以 ...
  • 一 引入背景 在軟體開發周期中,無論是開發中,或是測試中,或是上線後,選擇合適的工具監控程式的運行狀態至關重要,只有如此,才能更好地排查程式問題和檢測程式性能問題等。本篇文章主要與大家分享,如何利用Log4net在項目中記錄日誌文件。在應用程式出現問題時,啟用日誌記錄有助於解決問題。使用log4ne ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...