【WPF学习】第二十七章 Application类的任务
上一章介绍了有关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类的任务的更多相关文章
- 《Linux命令行与shell脚本编程大全》 第二十七章 学习笔记
第二十七章:shell脚本编程进阶 监测系统统计数据 系统快照报告 1.运行时间 uptime命令会提供以下基本信息: 当前时间 系统运行的天数,小时数,分钟数 当前登录到系统的用户数 1分钟,5分钟 ...
- Gradle 1.12用户指南翻译——第二十七章. Ear 插件
其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://g ...
- “全栈2019”Java多线程第二十七章:Lock获取lock/释放unlock锁
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...
- “全栈2019”Java第八十七章:类中嵌套接口的应用场景(拔高题)
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第三十七章:类与字段
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第二十七章:流程控制语句中循环语句for
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- 风炫安全WEB安全学习第二十七节课 XSS的防御措施
风炫安全WEB安全学习第二十七节课 XSS的防御措施 XSS防御措施 总的原则 控制好输入/输出 过滤:根据业务需求进行过滤,对email,手机号码这样的输入框进行验证. 转义:所有输出到前端的数据都 ...
- 【WPF学习】第二十六章 Application类——应用程序的生命周期
在WPF中,应用程序会经历简单的生命周期.在应用程序启动后,将立即创建应用程序对象,在应用程序运行时触发各种应用程序事件,你可以选择监视其中的某些事件.最后,当释放应用程序对象时,应用程序将结束. 一 ...
- 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 ...
随机推荐
- 深入JVM(二)JVM概述
深入JVM(一)JVM指令手册 深入JVM(二)JVM概述 一.JVM的原理 Java虚拟机是Java平台的基石,解决了硬件和操作系统的相互独立性.不同平台(Windows,Linux和MacOS)的 ...
- HashMap面试题解答
一.HashMap的实现原理? 此题可以组成如下连环炮来问 你看过HashMap源码嘛,知道原理嘛?为什么用数组+链表?hash冲突你还知道哪些解决办法?我用LinkedList代替数组结构可以么?既 ...
- ConcurrentHashMap 原理解析
为什么要用ConcurrentHashMap HashMap线程不安全,而Hashtable是线程安全,但是它使用了synchronized进行方法同步,插入.读取数据都使用了synchronized ...
- 微信支付与支付宝支付java开发注意事项
说明:这里只涉及到微信支付和淘宝支付 以官网的接口为准,主要关注[网关].[接口].[参数][加密方式][签名][回调] 第一步,了解自己的项目要集成的支付方式 常见的有扫码支付.网页支付.APP支付 ...
- 博帝飚速盘 16G
设备制造商: Patriot Memory当前协议 : USB2.0输入电流 : 300mA 芯片制造商: 群联(Phison)芯片型号 : PS2251-38闪存颗粒 : 美光( ...
- FacadePattern(外观模式)-----Java/.Net
外观模式(Facade Pattern)隐藏系统的复杂性,并向客户端提供了一个客户端可以访问系统的接口.这种类型的设计模式属于结构型模式,它向现有的系统添加一个接口,来隐藏系统的复杂性
- Linux常用命令大全(一)
Linux常用命令大全(一) 第一章 cal命令 $ cal 12 2017 :列出2017年12月的日历 $ cal 10 :列出公元10年的日历 $ cal 12 17 :列出公元17年12月的日 ...
- vue项目准备工作
1.写文档: 产品说明.工作日志.接口说明文档.数据库说明文档.项目架构说明文档等···· 例如:后台管理系统:商品的管理.店铺的管理.店铺类别管理.管理员的管理.用户管理等····· 前端渲染 ...
- 10_时间戳timeStamp 和 时间 time 转换, 根据时间节点倒计时
1: 时间戳 timeStamp 获取的几种方法及其优劣, 第一种只能精确到秒, 故不推荐使用, 最最常用的也是最官方的是第三种, 通过原型方法进行调用获取精确到毫秒数 : var timestamp ...
- Ubuntu管理软件源
在Ubuntu环境下,我们经常会使用apt-get(apt)命令下载各种软件,当所需软件在官方软件库中找不到时,我们需要添加第三方的软件源,或者由于位于海外的官方软件源下载速度过于感人时,需要添加国内 ...