TaskHosting - 开发桌面工具原来还可以这么简单
由来
对于喜欢开发的我经常会写一些小工具,这些小工具多以功能为主,不要求漂亮、个性化的UI。但起码要保证使用方便,因此最基本的功能要有:
- GUI(图片用户界面)
- 程序配置的保存与读取(让用户在GUI上操作要方便)
- 用户使用习惯的自动记录(例如:上次关闭时窗口位置及大小等)
- 程序崩溃捕获及上报
- 实时显示运行日志(当前执行到哪步了,输出结果是什么)
- 多线程管理及调度框架
如果每个工具都要COPY一遍以上功能的代码以后维护起来是个大坑,封装成库调用呢?那每个工具都要写一遍组装UI组件的代码。
仔细想一下,其实我要写的就是一款小工具,它以实现功能为主。
那么我就需要有一个开发框架,它可以让我只专注于功能(业务)的实现,简化UI相关的编码,最好是能用一行代码就实现一个UI功能,这个开发框架现在写好了我给它取名为TaskHosting
TaskHosting初识

先来张截图看看它长什么样~_~
TaskHosting界面中大部分UI都是可以自定义的,大部分情况下只需要2-3行C#代码就可以定制一部分UI功能,如果你不会WPF也可以用WinForms定制里面的UI(通过WindowsFormsHost)。
当然我们的目标是不关心UI,专注于功能实现,让我们来看下如何使用它。
首先从Hello Word开始:
- 首先要创建一个类库项目,然后引用TaskHosting框架相关类库
- 项目属性 > 调试 > 启动操作 > 选择【启动外部程序】,路径填写TaskHosting.exe的路径
- 创建一个【任务】类 HelloWordTask.cs,代码如下(现在先不用明白什么意思,后面会有介绍):
[Task("Hello Word", IsMutilThread = true, ConfigType = typeof(Config))]
class HelloWordTask : Mondol.TaskHosting.Task
{
private readonly Config _cfg;
private readonly IImmediateLogger _immediateLogger; public HelloWordTask(Config cfg, IImmediateLogger immediateLogger)
{
_cfg = cfg;
_immediateLogger = immediateLogger;
} public override void OnRun()
{
_immediateLogger.Info($"设置1的值为: { _cfg.Setting1}");
_immediateLogger.Info($"设置2的值为: { _cfg.Setting2}");
}
}
} - 创建一个【任务】的配置类 Config.cs,代码如下:
class Config : Mondol.Configuration, ITaskConfig
{
private readonly IAppEnvironment _appEnv; public Config(IAppEnvironment appEnv)
{
this._appEnv = appEnv;
} protected override string GetConfigurationFilePath()
{
return Path.Combine(_appEnv.GetPluginDataBasePath(GetType().Assembly), "HelloWord.Config.json");
} [Category("基本设置"), DisplayName("设置1"), Description("程序设置1,存储 string 类型的设置")]
public string Setting1
{
get
{
return GetProperty(nameof(Setting1), "我是设置1的默认值");
}
set
{
SetProperty(nameof(Setting1), value);
}
} [Category("基本设置"), DisplayName("设置2"), Description("程序设置2,存储 int 类型的设置")]
public int Setting2
{
get
{
return GetProperty(nameof(Setting2), );
}
set
{
SetProperty(nameof(Setting2), value);
}
}
}
运行程序,效果如下:

怎么样,是不是很简单?
有什么功能
目前TaskHosting有如下功能:
- 强大的日志功能
- 输出到UI中的即时日志也可以存到文件、数据库、甚至远程服务器
- 程序崩溃自动将异常记录日志
- 即时日志可按日志级别(跟踪、调试、信息、警告、错误、致命)过滤显示
- 可用专用的文件日志分析工具
- 方便的配置存取
- 每个【任务】都可以有自己的配置文件
- 配置文件自动读取、自动保存
- 配置项可限制数据类型、支持拖拽
- 配置项可加详细描述
- 自动记录使用习惯
- 重新打开软件会恢复上次选择的【任务】、窗口大小、位置等
- 崩溃捕获及上报
- 自动捕获未处理异常,并显示友好错误处理窗口
- 未处理异常可自动上报服务器、发邮件或由用户处理
- 多线程调度框架
- 经过优化的线程调度框架、最大化利用系统资源
- 【任务】可决定是否启用多线程
- 更多特性还在完善中。。。
以上功能均可灵活的自定义,TaskHosting的目标 - 简洁而不简单
设计概念
TaskHosting是一个【任务】的管理器(继承自Mondol.TaskHosting.Task的类就是一个任务),每一个【任务】对应一个功能,例如:网址采集任务、数据同步任务等。
【任务】的类来自于你编写的类库项目(下称插件),每一个插件可以包含多个任务。
参考上面的HelloWordTask类,它继承了Mondol.TaskHosting.Task所以它是一个任务,将来会显示在【任务:】列表中。
你可以用Task属性来指定任务的名称、是否支持多线程、配置类类型等
无处不在的依赖注入
依赖注入是一个很简单的概念,例如:ClassA依赖于ClassB,ClassB又依赖于ClassC,传统的硬编码方式是先分别new出ClassB、ClassC才能new ClassA。
依赖注入框架会管理这种依赖关系,使用时无需关心谁依赖谁,只需要告诉框架我需要ClassA的实例,它就会自动帮你创建出来。
详细依赖注入的概述网上有很多介绍文章,大家找一下就好了。
TaskHosting采用了Autofac来管理模块间的依赖关系,TaskHosting支持3种类型注册的方式:
- 编码注册
//相当于程序的Main方法所属类(后面会详细介绍)
internal class Initializer : IInitializer
{
public void ConfigureServices(ContainerBuilder builder)
{
//Autofac的方式注册
builder.RegisterType<Config>().AsSelf();
}
} - 属性注册
//通过属性注入,可指定是否单例、作用域
[Injection(AsType = typeof(Config), IsSingleInstance = true, Scope= InjectionScope.Running)]
class Config
{
//类实现...
} - 自动注册
所有实现了ITaskConfig接口的类都会自动注册为单例,例如:class Config : Mondol.Configuration, ITaskConfig
{
//类实现...
}
除了插件自己注册的类型外,框架还会注册一些接口给插件使用,例如:
- IAppEnvironment //APP环境
- IEventManager //事件管理器
- IImmediateLogger //即时日志输出接口
更多接口请参见Mondol.TaskHosting.Abstractions.dll程序集,这个程序集里全是接口定义,里面的接口将来都可以通过依赖注入获取到
小结
以上简单的介绍了TaskHosting的概念及基本用法,其目的就是让你可以更专注于功能业务的开发,而无需过多关心UI实现细节。
后续会详细介绍一些高级的使用方法及示例
使用中如有什么意见或问题欢迎随时反馈给我Q46029811
以上提到的示例代码:单击下载
TaskHosting - 开发桌面工具原来还可以这么简单的更多相关文章
- heX——基于 HTML5 和 Node.JS 开发桌面应用
heX 是网易有道团队的一个开源项目,允许你采用前端技术(HTML,CSS,JavaScript)开发桌面应用软件的跨平台解决方案.heX 是你开发桌面应用的一种新的选择,意在解决传统桌面应用开发中繁 ...
- heX:用HTML5和Node.JS开发桌面应用
国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...
- node-webkit:开发桌面+WEB混合型应用的神器
顾名思义, node -webkit就是 node js+webkit. 这样做的好处显而易见,核心奥义在于,用 node js来进行本地化调用,用webkit来解析和执行HTML+JS. 快速上手 ...
- node-webkit开发桌面应用
Node-Webkit能够做什么呢?(打开链接看discuss) github 项目源:https://github.com/rogerwang 导言 node-webkit 是一个很神奇的桌面客户端 ...
- node.js 开发桌面程序, 10个令人惊讶的NodeJS开源项目
用 node-webkit 开源框架. 做企业站,杠杠地 包括电子书和支付宝系统都是node开发的,. 接收传感器发送的数据再运算...对水泵.风机.空调这些硬件进行远程控制. 细数10个令人惊讶的N ...
- (转)heX——基于 HTML5 和 Node.JS 开发桌面应用
本文转载自:http://techblog.youdao.com/?p=685 简介:heX,一个允许你采用前端技术(HTML,CSS,JavaScript)开发桌面应用软件的跨平台解决方案.是你开发 ...
- IE开发人员工具教程
写在前面 一直非常谷歌的控制台,因为我是做前端的,谷歌浏览器在我看来是解析JS最快的浏览器,所谓的熟能生巧,用熟悉了谷歌浏览器之后就特别喜欢用谷歌的控制台调试脚本.改变样式.查看HTML.查看资源加载 ...
- go 可以开发桌面应用
go 可以开发桌面应用 go 可以开发桌面应用,但并不是很舒适.可以使用的GUI库有:1.goqt,LiteIDE作者出品,Go和QT的绑定,还未发布2.go.uik,纯Go实现的并发UI工具3.wa ...
- Java开发桌面程序学习(一)——JavaFx+Jfoenix初始以及搭建
Java开发桌面程序学习(一)--JavaFx+Jfoenix初始以及搭建 前言 想做一个Java的桌面程序,但是,使用原生的Swing感觉又十分麻烦,那个布局都是拿代码设置,看着十分的乱,偶然的情况 ...
随机推荐
- 数论 : 模运算法则(poj 1152)
题目:An Easy Problem! 题意:求给出数的最小进制. 思路:暴力WA: discuss中的idea: 给出数ABCD,若存在n 满足 (A* n^3 +B*n^2+C*n^1+D*n^0 ...
- MySQL使用小记
时间格式化: select date_format('2008-08-08 22:23:01', '%Y%m%d%H%i%s'); 去重复: use iksdb3; select distinct ` ...
- log4net详细配置说明
原文地址:http://blog.sina.com.cn/s/blog_671486bc01011rdj.html 1.概述 log4net是.Net下一个非常优秀的开源日志记录组件.log4net记 ...
- oracle ORA-00911 问题 解决
书写sql语句 using (OracleConnection conn = new OracleConnection(OracleString)) { conn.Open(); var trans ...
- 重新理解JS的6种继承方式
写在前面 一直不喜欢JS的OOP,在学习阶段好像也用不到,总觉得JS的OOP不伦不类的,可能是因为先接触了Java,所以对JS的OO部分有些抵触. 偏见归偏见,既然面试官问到了JS的OOP,那么说明这 ...
- ASP.NET Core重写个人博客站点小结
今天用ASP.NET Core重写了个人博客站点,原来是基于ASP.NET 4.5开发的.重写工作总体很顺利,最后成功发布到Ubunt+Nginx平台上.效果如下: 右边的Header信息里可以看到已 ...
- [C++] C\C++ printf 输出格式
1.转换说明符 %a(%A) 浮点数.十六进制数字和p-(P-)记数法(C99) %c 字符 %d 有符号十进制整数 % ...
- 说不尽的MVVM(5) - 消息满天飞
知识预备 阅读本文,我假定你具备以下知识: C#和WPF基础知识 Lambda表达式 清楚ViewModel的职责 如果我们的程序需要弹出一个MessageBox,我们应该怎么做? 我见过不少人在Vi ...
- [BTS] Deploy Command
BizTalkDeploymentTools.AddResource.bat @Echo OFF SET ApplicationName=%~1 SET ComponentType=%~2 SET C ...
- Atitit.列表页面and条件查询的实现最佳实践(2)------翻页 分页 控件的实现java .net php
)------翻页 分页 控件的实现java .net php 1. 关于翻页有关的几大控件::搜索框控件,显示表格控件,翻页器,数据源控件.. 1 2. 翻页的显示格式:: 1 2.1. 通常ui- ...