C#WinForms崩溃,没有异常

时间:2019-07-16 17:48:08

标签: c# winforms crash httpclient

我有一个C#WinForms应用程序。

我正在使用Unity Container nuget包来注入我的服务类。如果我在Program.cs中获得了服务类,并调用了我的Web api以使用odata验证用户名和密码,则它可以成功运行。如果我调用Application.Run(myForm)(其中myForm是Telerik RadForm)并异步运行相同的调用,则应用程序将关闭,不会引发任何异常。

我正在使用Application.ThreadException并注册Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException),但这没什么用。

导致应用程序崩溃的行是对System.Net.Http.HttpClient.PostAsync()的调用。该调用是从位于单独的.Net Standard 2.0 Class Library中的服务类内部进行的。主应用程序是.NET Framework 4.7.2 Windows Application,UI控件位于.NET Framework 4.7.2 Class Library中。同样,如果在Telerik RadForm之外调用,则此调用成功。

在Telerik RadForm中,我尝试使用以下命令执行呼叫:

  • private async void btnLogin_Click(object sender, System.EventArgs e)
  • var t = new Thread(() => Runner.DoWork(new Credentials(u, p, CLIENT_ID)))
  • 不使用await的对静态方法的异步调用

当我在服务类上调用LoginAsync(...).Result时,确实得到了不同的结果。这导致应用程序进入线程锁定状态,但没有使应用程序崩溃。

更新:

有两个RadForms。 DashboardFormLoginFormProgram.cs启动仪表板,该仪表板检查现有的Bearer令牌。如果令牌不存在,则仪表板将使用.ShowDialog()显示登录名,以防止在经过身份验证之前使用仪表板。然后,登录名将使用上述服务类。

如果我不是使用Program.csApplication.Run()中启动仪表板,而是启动了Login,则用于身份验证的服务调用成功。

然后我尝试在新线程中从仪表板启动登录,但这导致了上述相同的问题。

如果System.Net.Http.HttpClient.PostAsync()从另一个表单显示的表单中调用,为什么会使应用程序崩溃(无例外)?

程序

static class Program
{
    /// <summary>
    /// The main entry point for the application.
    /// </summary>
    [STAThread]
    static void Main()
    {
        Application.EnableVisualStyles();
        Application.SetCompatibleTextRenderingDefault(false);

        WinInjectionHelper.Register(UnityConfig.RegisterComponents);

        Application.ThreadException += Application_ThreadException;
        Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);

        Application.Run(new DashboardForm());
    }

    private static void Application_ThreadException(object sender, System.Threading.ThreadExceptionEventArgs e)
    {
        MessageBox.Show("Big error...");
    }

仪表板

public partial class DashboardForm : Telerik.WinControls.UI.RadForm, IDashboard
{
    private readonly IProductService _service;

    public DashboardForm()
    {
        this._service = WinInjectionHelper.Generate2<IProductService>();
        this.InitializeComponent();
    }

    private void DashboardForm_Load(object sender, EventArgs e)
    {
        if (this.NotAuthenticated())
        {
            var frm = new LoginForm();

            if (frm.ShowDialog() != DialogResult.OK)
            {
                this.Close();
            }
        }
        else
        {
            //TODO: display error to user
            this.Close();
        }
    }
}

登录

public partial class LoginForm : Telerik.WinControls.UI.RadForm
{
    private const string CLIENT_ID = "...";
    private readonly IAuthenticationService _service;

    public LoginForm()
    {
        this._service = WinInjectionHelper.Generate2<IAuthenticationService>();
        this.InitializeComponent();
    }

    private void btnCancel_Click(object sender, System.EventArgs e)
    {
        this.DialogResult = DialogResult.Cancel;
        this.Close();
    }

    private async void btnLogin_Click(object sender, System.EventArgs e)
    {
        var u = this.txtUsername.Text.Trim();
        var p = this.txtPassword.Text.Trim();

        if (string.IsNullOrWhiteSpace(u))
        {
            MessageBox.Show("Username is required!", "NOTICE", MessageBoxButtons.OK, MessageBoxIcon.Exclamation);
            return;
        }

        if (string.IsNullOrWhiteSpace(p))
        {
            MessageBox.Show("Password is required!", "NOTICE", MessageBoxButtons.OK, MessageBoxIcon.Exclamation);
            return;
        }

        try
        {
            var jwt = await this._service.LoginAsync(new Credentials(u, p, CLIENT_ID));

            var identity = new ClaimsIdentity(new[]
            {
                new Claim(ClaimTypes.Name, u),
                new Claim(ClaimTypes.Email, u),
                new Claim(ClaimTypes2.AccessToken, jwt.AccessToken),
            }, "JWT");

            ((GenericPrincipal)Thread.CurrentPrincipal).AddIdentity(identity);

            this.DialogResult = DialogResult.OK;

            this.Close();
        }
        catch
        {
            MessageBox.Show("An error occurred while processing your request.", "ERROR", MessageBoxButtons.OK, MessageBoxIcon.Error);
        }
    }
}

2 个答案:

答案 0 :(得分:0)

您可以检查Windows事件查看器并在应用程序下查看以查找崩溃日志。我正在使用的应用程序也存在同样的问题,这是唯一包含任何信息的地方。它实际上帮助我解决了与定位特定.NET Framework版本有关的问题。单击左下方的“窗口”按钮,然后在其中键入“事件查看器”,然后单击显示的图标。

最简单的方法是运行该应用程序并使它崩溃,然后立即转到事件查看器,使其位于列表的顶部。

答案 1 :(得分:0)

我解决了该问题,方法是先调用“登录”表单,然后在关闭登录名之前显示仪表板,并更改了program.cs,以便关闭登录名不会退出应用程序。确实需要在应用程序确实需要关闭时调用Application.Exit()

程序

internal static class Program
{
    [STAThread]
    private static void Main()
    {
        ...

        new LoginForm().Show();

        Application.Run();
    }
}