上一章介绍了有关WPF应用程序中使用Application对象的方式,接下来看一下如何使用Application对象来处理一些更普通的情况,接下俩介绍如何初始化界面、如何处理命名行参数、如何处理支付窗口之间的交互、如何添加跟踪文档以及如何创建单示例应用程序。

一、显示初始化界面

  WPF应用程序的运行速度快,但并不能在瞬间启动。当第一次启动应用程序时,会有一些延迟,因为公共语言运行时(Common Language Runtime,CLR)首先需要初始化.NET环境,然后启动应用程序。

  这一延迟未必成为问题。通常,只需要经历很短的时间,就会出现第一个窗口,但如果具有更耗时的初始化步骤,或者如果只是希望通过显示打开的图像应用程序显得更加专业,这时可使用WPF提供的简单初始界面特性。

  下面是添加初始界面的方法:

  (1)为项目添加图像文件(通常是.bmp、.png或.jpg文件)。

  (2)在Solution Explorer中选择图像文件。

  (3)将Build Action修改为SplashScreen。

  下次运行应用程序时,图像会立即在屏幕中央显示出来。一旦准备好运行时环境。而且Application_Startup方法执行完毕,应用程序的第一个窗口就将显示出来,这时初始化界面图像会很快消失(约需300毫秒).

  该特性听起来很简单,事实上也确实如此。只需要记住显示的初始化界面没有任何装饰,在它周围没有窗口边框,所有由你决定是否为初始界面图像添加边框,也无法通过显示一系列的多幅图像或动画让初始化图像显得更富有想象力的效果。如果希望得到这种效果,需要采用传统方法:创建在运行初始化代码的同事显示你所希望的图像界面的启动窗口。

  顺便提一下,当添加初始界面时,WPF编译器会自动生成的App.g.cs文件添加与下面的类似的代码:

SplashScreen splashScreen = new SplashScreen("images/blue.jpg");
//Show the splash screen
//The true parameter sets the splashScreen to fade away automatically
//after the first window appears
splashScreen.Show(true); //Start the application
AssemblyResources.App app = new AssemblyResources.App();
app.InitializeComponent();
app.Run();
//The splash screen begins ites automatic fade-out now.

  也可自行编写这一剪短逻辑,而不死使用SplashScreen 生成操作。但有一点需要指出,可以改变的唯一细节是初始界面褪去的速度。为此,需要项SplashScreen.Show()方法传递false(从而使WPF不会自动淡入初始化界面)。然后由你负责通过调用SplashScreen.Close()方法在恰当的时机隐藏初始界面,并提供TimeSpan值来指示经过多长时间淡出初始化界面。

二、处理命令行参数

  为处理命令行参数,需要相应Application.Startup事件。命令行参数是通过StartupEventArgs.Args属性作为字符串数组提供的。

  例如,假定希望加载文档,文档名作为命令行参数传递。在这种情况下,有必要读取命令行参数并执行所需的一些额外初始化操作。在下面的示例中,通过响应Application.Startup事件实现了这一模式。在该例中,没有在任何地方设置Application.StartupUri属性——而是使用代码实例化窗口。

public partial class App : Application
{
// The command-line argument is set through the Visual Studio
// project properties (the Debug tab).
private void App_Startup(object sender, StartupEventArgs e)
{
// At this point, the main window has been created but not shown.
FileViewer win = new FileViewer(); if (e.Args.Length > )
{
string file = e.Args[];
if (File.Exists(file))
{
// Configure the main window.
win.LoadFile(file);
}
} // This window will automatically be set as the Application.MainWindow.
win.Show();
}
}
<Application x:Class="LoadFromCommandLine.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Startup="App_Startup">
<Application.Resources> </Application.Resources>
</Application>

App.xaml

  上面的方法初始化主窗口,然后当App_Startup()方法结束是显示主窗口。上面的代码假定FileViewer类有名为LoadFile()的共有方法。这只是一个示例,它只读取并显示指定文件的文本。

三、访问当前Application对象

  通过静态的Application.Current属性,可在应用程序的任何位置获取当前应用程序实例,从而在窗口之间进行基本交互,因为任何窗口都可以访问当前Application对象,并通过Application对象,并通过Application对象获取主窗口的作用:

Window main=Application.Current.MainWindow;
MessageBox.Show("The main window is "+main.Title);

  当然,如果希望访问在自定义主窗口类中添加的任意方法、属性或事件,需要将窗口对象转换为正确类型。如果主窗口的自定义类MainWindow的实例,可使用与下面类似的代码:

MainWindow main=(MainWindow)Application.Current.MainWindow;
main.DoSomething();

  在窗口中还可以检查Application.Windows集合的内容,该属性提供了所有当前打开窗口的引用:

foreach(Window window in Application.Current.Windows)
{
MessageBox.Show(window.Title+" is open.");
}

  实际上,大多数应用程序通常使用一种更具结构化特点的方式在窗口之间进行交互。如果有几个长时间运行的窗口同事打开,并且它们之间需要以某种方式进行通信,在自定义应用程序类中保存这些窗口的引用可能更有意义。这样,总可以找到所需的窗口。与此类似,如果有基于文档的应用程序,那么可选择创建跟踪文档窗口的集合,而不是跟踪其他内容。

四、在窗口之间进行交互

  整如在前面已经看到的,自定义应用程序类是放置响应不同应用程序事件的代码的好地方。应用程序类还可以很好地达到另一个目的:保存重要窗口的引用,使一个窗口可访问另一个窗口。

  例如,假设希望跟踪应用程序使用的所有文档窗口。为此,可在自定义应用程序类中创建专门的集合。下面是使用泛型列表集合保存一组自定义窗口对象的示例。在这个示例中,每个文档窗口由名为Document类的实例表示:

 public partial class App : Application
{
private List<Document> documents = new List<Document>(); public List<Document> Documents
{
get { return documents; }
set { documents = value; }
}
}

  现在,当创建新文档时,只需要记住将其添加到Documents集合中即可。下面是响应按钮单击事件的事件处理程序,该事件处理程序完成了所需的工作:

private void cmdCreate_Click(object sender, RoutedEventArgs e)
{
Document doc = new Document();
doc.Owner = this;
doc.Show();
((App)Application.Current).Documents.Add(doc);
}

  同样,也可在Document类中响应Window.Loaded类这些事件,以确保当创建文档对象时,总会在Documents集合中注册该文档对象。

  现在,可在代码的其他任何地方进行集合来遍历所有文档,并使用公有成员。在该例中,Document类包含用于更新显示的自定义方法SetContent();

private void cmdUpdate_Click(object sender, RoutedEventArgs e)
{
foreach (Document doc in ((App)Application.Current).Documents)
{
doc.SetContent("Refreshed at " + DateTime.Now.ToLongTimeString() + ".");
}
}
<Window x:Class="WindowTracker.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="WindowTracker" Height="300" Width="300"
>
<StackPanel>
<Button Click="cmdCreate_Click" Margin="10,10,10,5" Name="cmdCreate">Click to Create a Document</Button>
<Button Click="cmdUpdate_Click" Margin="10,5,10,10" Name="cmdUpdate">Click to Refresh the Documents</Button>
</StackPanel>
</Window>

MainWindow.xaml

<Window x:Class="WindowTracker.Document"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="WindowTracker" Height="78" Width="209"
>
A new document.
</Window>

Document.xaml

public partial class Document : Window
{
public Document()
{
InitializeComponent();
} public void SetContent(string content)
{
this.Content = content;
}
}

Document.xaml.cs

  下图显示了最终效果图。最终结果谈不上华美,但这种交互方式值得注意——演示了一种通过自定义应用程序类在窗口之间进行交互的安全规范的方式。这种方式比使用Window属性要好,因为是强类型,只包含Document窗口(而不是包含窗口应用程序中所有窗口的集合)。通过这种方式还可使用另一种更有用的方式对窗口进行分类。例如,可使用字典集合,通过键名更方便地查找文档。在基于文档的引用程序中,可通过文件名来索引集合中的窗口。

五、单实例应用程序

  通常,只要愿意就可以加载WPF应用程序的任意多个副本。某些情况下, 这种设计时非常合理的。但在另外一些情况下,这可能会成为问题,当构建基于文档的应用程序时更是如此。

  对于单实例应用程序,WPF本身并未提供自带的解决方法,但可使用几种变通方法。基本技术时当触发Application.Startup事件时,检查另一应用程序实例是否已在运行。最简单的方法是使用全局的mutex对象(mutex对象时操作系统提供的用于进程间通信的同步对象)。这种方法很简单,但功能有限。最重要的是,应用程序的新实例无法与已经存在的实例进行通信。对于基于文档的应用程序而言这确实是一个问题,因为新实例可能需要告诉已经存在的应用程序实例打开某个特定的文档(如果该文档是通过命令行参数传递的)。

【WPF学习】第二十七章 Application类的任务的更多相关文章

  1. 《Linux命令行与shell脚本编程大全》 第二十七章 学习笔记

    第二十七章:shell脚本编程进阶 监测系统统计数据 系统快照报告 1.运行时间 uptime命令会提供以下基本信息: 当前时间 系统运行的天数,小时数,分钟数 当前登录到系统的用户数 1分钟,5分钟 ...

  2. Gradle 1.12用户指南翻译——第二十七章. Ear 插件

    其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://g ...

  3. “全栈2019”Java多线程第二十七章:Lock获取lock/释放unlock锁

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  4. “全栈2019”Java第八十七章:类中嵌套接口的应用场景(拔高题)

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  5. “全栈2019”Java第三十七章:类与字段

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  6. “全栈2019”Java第二十七章:流程控制语句中循环语句for

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  7. 风炫安全WEB安全学习第二十七节课 XSS的防御措施

    风炫安全WEB安全学习第二十七节课 XSS的防御措施 XSS防御措施 总的原则 控制好输入/输出 过滤:根据业务需求进行过滤,对email,手机号码这样的输入框进行验证. 转义:所有输出到前端的数据都 ...

  8. 【WPF学习】第二十六章 Application类——应用程序的生命周期

    在WPF中,应用程序会经历简单的生命周期.在应用程序启动后,将立即创建应用程序对象,在应用程序运行时触发各种应用程序事件,你可以选择监视其中的某些事件.最后,当释放应用程序对象时,应用程序将结束. 一 ...

  9. Java基础知识二次学习--第六章 常用类

    第六章 常用类   时间:2017年4月26日16:14:49~2017年4月26日16:56:02 章节:06章_01节~06章_06节 视频长度:20:57+1:15+8:44+1:26+11:2 ...

随机推荐

  1. 如何使用IoTSharp对接ModBus?

    提到对接ModBus 那么或许大家最熟悉的可能是 HslCommunication 和SharpSCADA 了,是GitHub 上关注最多的此类开源项目,  因此IoTSharp将通过HSL组件进行数 ...

  2. Python反射笔记

    通过字符串的形式,导入模块.使用__import__ In [3]: SYS = __import__("sys") In [4]: SYS.path 如果导入模块在多级文件夹下 ...

  3. UVW平台运动控制算法以及matlab仿真

    UVW平台运动控制算法以及matlab仿真   最近公司同事因为对某视觉对位平台的运动控制算法有疑问,所以来请教我.由于我也是第一次接触到UVW自动对位平台(也可以叫XXY自动对位平台),于是找了一些 ...

  4. JMM中的Happens-Before原则

      在java内存模型中,happens-before应该理解为:前一个操作的结果,可以被后续的操作获取,即内存可见性.   为了解决多线程的内存可见性问题,就提出了happens-before原则, ...

  5. eclipse中如何配置jdk

    1.在eclipse的上方打开Windows这个选项,选择Preferences==>Java==>Installed JREs 2.然后选择Add==>Standard VM==& ...

  6. 深入ReentrantLock的实现原理和源码分析

    ReentrantLock是Java并发包中提供的一个可重入的互斥锁.ReentrantLock和synchronized在基本用法,行为语义上都是类似的,同样都具有可重入性.只不过相比原生的Sync ...

  7. DOCKER学习_008:Docker容器的运行最佳实践

    一 容器分类 容器按用途大致可分为两类: 服务类容器,如 web server. database等 工具类容器,如cur容器, Iredis-cli容器 通常而言,服务类容器需要长期运行,所以使用 ...

  8. xshell连接不上ubuntu---could not connect to 'ip' (port 22): Connection failed.

    可能是没有开启ssh server,接下来就是开启服务就好.命令如下: sudo apt-get install openssh-server 这样就OK啦

  9. CentOS7.2 部署Ceph分布式存储

    1.1 环境准备 主机名 IP地址 ceph-admin 192.168.16.220 ceph-node1,ceph-mon 192.168.16.221 ceph-node2,ceph-mon 1 ...

  10. Django之models字段属性

    目录 常用字段 AutoField IntegerField CharField 自定义及使用char DateField DateTimeField 字段合集 字段参数 null unique db ...