- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
其实很多公司或者资深的开发都有自己快速创建项目的脚手架的,有的是魔改代码生成器实现,有的直接基于T4,RazorEngine等模板引擎打造;但无论如何,其最终目的其实就是搭建一个自定义项目模板(脚手架).
今天我们聊聊:如何基于官方的cli donet new 命令创建自己的项目模板.
我想用一个命令来说明:
dotnet new list
到这里大家就非常熟悉了,原来大家平时创建项目都是基于已有的模板创建的(红圈部分大家应该不陌生);我们今天目的就是创建一个这样的模板,并在vs新建项目时可供选择创建项目,或者使用cli命令直接创建; 。
当然,还有公开模板:
https://dotnetnew.azurewebsites.net/ 。
这里准备的项目就是平时普通的项目,后面会以这个项目为蓝本创建模板;因为我最近使用Azure Function类型项目比较多,我就以Function项目为例,其他类型项目同理的; 。
项目结构图:
项目文件结构:
D:.
│ appsettings.CI.json
│ appsettings.Development.json
│ appsettings.json
│ appsettings.Production.json
│ Dockerfile
│ Function1.cs
│ host.json
│ local.settings.json
│ MyCompany.Cutapi.FunctionTemp.csproj #这个名字后面要被替换的
│ settings.CI.yaml
│ settings.Production.yaml
│ Startup.cs
│
├─build
│ CD.yaml
│ CI.yaml
│ _deploy.yaml
│
└─deploy
│ kustomization.yaml
│
├─base
│ deploy.yaml
│ kustomization.yaml
│
├─ci
│ deploy.yaml
│ kustomization.yaml
│
└─prod
deploy.yaml
kustomization.yaml
可以看到其实有很多跟构建,部署等有关的配置文件; 。
Function1.cs 。
#模板项目的命名空间
namespace MyCompany.Cutapi.FunctionTemp
{
public class Function1
{
private readonly Stopwatch _sw;
private readonly IExtractSegmentService _extractSegmentService;
private readonly ILogger<Function1> _logger;
public Function1(IExtractSegmentService extractSegmentService, ILogger<Function1> logger)
{
_sw = new Stopwatch();
_extractSegmentService = extractSegmentService;
_logger = logger;
}
#模板项目的FunctionName 和一些跟队列有关的配置,这些后面都要
[FunctionName("function1")]
[return: ServiceBus("cutapi-queue1-notify", Connection = "ServiceBusConnection")]
public async Task<VideoTranscodeNotify> Run([ServiceBusTrigger("cutapi-queue1", Connection = "ServiceBusConnection")] ServiceBusReceivedMessage message
, string messageId
, ServiceBusMessageActions messageActions
, Int32 deliveryCount
, DateTime enqueuedTimeUtc
, ILogger log
)
{
_sw.Start();
var messageBody = Encoding.UTF8.GetString(message.Body);
log.LogInformation($"{Environment.MachineName} -> function1 begin ->{messageId}: {messageBody}");
await messageActions.CompleteMessageAsync(message);
var result = new VideoTranscodeNotify();
try
{
//todo...
}
catch (Exception ex)
{
log.LogError(ex, $"{Environment.MachineName} -> {messageId}:function1 Exception:{ex.Message}");
}
_sw.Stop();
log.LogInformation($"{Environment.MachineName} function1 Over ->{messageId} Elapsed: {_sw.Elapsed}");
return result;
}
}
}
以这个文件为例,模板项目里很多文件内容都可以按自定义参数被替换;当然文件名也可以替换; 。
在项目根目录下创建配置文件:/.template.config/template.json 。
结构如下:
├─.template.config │ template.json 。
内容:
{
"author": "Heiner Wang", //作者
"classifications": [ "Azure Functions" ], //项目归类 classifications 还会出现在“Tags”列中
"name": "Heiner Function", //项目全名,用户应看到的模板名称。
"identity": "HeinerFunction", //项目唯一id
"shortName": "hfunc", //项目简写
"tags": {
"language": "C#",
"type": "project"
},
"sourceName": "MyCompany.Cutapi.FunctionTemp", //运行模板时使用 -n 或 --name 选项提供要替换的值,不写了话项目名称不变
"preferNameDirectory": true, //创建项目的目录层级;
"symbols": { //自定义语法
//自定义参数,新项目命名空间
"Namespace": {
"type": "parameter",
"dataType": "text", //文本类型
"defaultValue": "Heiner.Function",
"replaces": "MyCompany.Cutapi.FunctionTemp" //项目里这个值将会被替换掉
//"fileRename": "MyCompany.Cutapi.FunctionTemp" //也可以指定替换文件名
},
"FunctionName": {
"type": "parameter",
"dataType": "text",
"defaultValue": "function1",
"replaces": "function1"
},
"QueueName": {
"type": "parameter",
"dataType": "text",
"defaultValue": "cutapi-queue1",
"replaces": "cutapi-queue1"
},
"EnableRedis": {
"type": "parameter",
"dataType": "bool", #布尔类型的
"defaultValue": "true"
}
}
}
更多参数请参考:https://github.com/dotnet/templating/wiki/Reference-for-template.json 。
cs文件 。
//EnableRedis是自定义参数
#if (EnableRedis)
ConnectionMultiplexer redisConnection = ConnectionMultiplexer.Connect(AppSettings.GetConnectionString("Redis"));
builder.Services.AddSingleton<IConnectionMultiplexer>(redisConnection);
builder.Services.AddSingleton<IDatabase>(c => redisConnection.GetDatabase());
#endif
项目文件 。
<ItemGroup>
<PackageReference Include="Microsoft.Azure.Functions.Extensions" Version="1.1.0" />
<PackageReference Include="Microsoft.Azure.WebJobs.Extensions.DurableTask" Version="2.9.0" />
<PackageReference Include="Microsoft.Azure.WebJobs.Extensions.ServiceBus" Version="5.9.0" />
<PackageReference Include="Microsoft.NET.Sdk.Functions" Version="4.1.1" />
<PackageReference Include="Microsoft.VisualStudio.Azure.Containers.Tools.Targets" Version="1.19.5" />
</ItemGroup>
<ItemGroup Condition="'$(EnableRedis)' == 'True' ">
<PackageReference Include="StackExchange.Redis" Version="2.6.48" />
</ItemGroup>
模板文件加入如下配置 。
"symbols":{...},
"sources": [
{
"modifiers": [
{
"condition": "(!EnableRedis)", //EnableRedis!=true
"exclude": [ //排除下面的文件(这里仅做示例),后面的模板项目当设置参数:EnableRedis==false时,下面的文件就被过滤掉了
"src/MyCompany.Cutapi.FunctionTemp/Redis.cs",
]
}
]
}
]
这一步是将根据配置文件,将普通项目安装成一个项目模板,理论上创建自定义模板到这步就完成了; 。
项目根目录执行:
dotnet new install .
这里命令后面的`.` 是安装当前目录的项目的意思;
dotnet new install D:\MyCompany.Cutapi.FunctionTemp
也可以这样,用绝对路径
强制覆盖安装 。
dotnet new install . --force
先删除再安装 。
#先删除
dotnet new uninstall .
#重新安装
dotnet new install .
后面的.都代表在项目根目录执行,后面不再赘述; 。
dotnet new list
无论用cli还是vs 都可以看到我们项目模板了,创建模板成功; 。
参考 。
这步是可选的! 注意!很多内部模板要脱密处理后再执行推送,请勿将机密信息推送到公网; 。
MyCompany.Cutapi.FunctionTemp.nuspec
<?xml version="1.0"?>
<package >
<metadata>
<id>HeinerFunction</id>
<version>1.0.0</version>
<authors>Heiner Wang</authors>
<owners>Heiner Wang</owners>
<requireLicenseAcceptance>false</requireLicenseAcceptance>
<description>xxx 公司 Azure Function 快速模板.</description>
<tags>dotnet-new;template</tags>
</metadata>
<files>
<file src="**\*" target="content"/>
</files>
</package>
在项目根目录执行 。
nuget pack MyCompany.Cutapi.FunctionTemp.nuspec
生成nuget包:
HeinerFunction.1.0.0.nupkg 。
nuget push HeinerFunction.1.0.0.nupkg -Source https://api.nuget.org/v3/index.json -ApiKey YOUR_API_KEY
这步的--Source参数,如果你有搭建好自己的nuget服务端的话改成你自己的; 。
模板有了,怎么用这个就简单了; 。
在创建项目时直接选择自定义模板 。
不过这样的话,自定义参数都是用默认值,所以我还是更推荐用命令行方式; 。
大家做demo的时候都应该执行过这样的命令,其实这就是使用了官方shotname为console的模板 。
dotnet new console -n MyConsoleApp1
一样,自定义模板命令为:
#默认参数
dotnet new hfunc -n MyCompany.Heiner.Test
#指定参数
dotnet new hfunc -n MyCompany.Heiner.Test --Namespace MyCompany.Heiner.Test --FunctionName function-live-record --QueueName cutapi-live-record --EnableRedis false
创建成功 。
https://learn.microsoft.com/zh-cn/dotnet/core/tools/custom-templates 。
https://cloud.tencent.com/developer/article/2319366 。
https://github.com/dotnet/templating/wiki/Reference-for-template.json 。
最后此篇关于C#如何创建一个可快速重复使用的项目模板的文章就讲到这里了,如果你想了解更多关于C#如何创建一个可快速重复使用的项目模板的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我在网上搜索但没有找到任何合适的文章解释如何使用 javascript 使用 WCF 服务,尤其是 WebScriptEndpoint。 任何人都可以对此给出任何指导吗? 谢谢 最佳答案 这是一篇关于
我正在编写一个将运行 Linux 命令的 C 程序,例如: cat/etc/passwd | grep 列表 |剪切-c 1-5 我没有任何结果 *这里 parent 等待第一个 child (chi
所以我正在尝试处理文件上传,然后将该文件作为二进制文件存储到数据库中。在我存储它之后,我尝试在给定的 URL 上提供文件。我似乎找不到适合这里的方法。我需要使用数据库,因为我使用 Google 应用引
我正在尝试制作一个宏,将下面的公式添加到单元格中,然后将其拖到整个列中并在 H 列中复制相同的公式 我想在 F 和 H 列中输入公式的数据 Range("F1").formula = "=IF(ISE
问题类似于this one ,但我想使用 OperatorPrecedenceParser 解析带有函数应用程序的表达式在 FParsec . 这是我的 AST: type Expression =
我想通过使用 sequelize 和 node.js 将这个查询更改为代码取决于在哪里 select COUNT(gender) as genderCount from customers where
我正在使用GNU bash,版本5.0.3(1)-发行版(x86_64-pc-linux-gnu),我想知道为什么简单的赋值语句会出现语法错误: #/bin/bash var1=/tmp
这里,为什么我的代码在 IE 中不起作用。我的代码适用于所有浏览器。没有问题。但是当我在 IE 上运行我的项目时,它发现错误。 而且我的 jquery 类和 insertadjacentHTMl 也不
我正在尝试更改标签的innerHTML。我无权访问该表单,因此无法编辑 HTML。标签具有的唯一标识符是“for”属性。 这是输入和标签的结构:
我有一个页面,我可以在其中返回用户帖子,可以使用一些 jquery 代码对这些帖子进行即时评论,在发布新评论后,我在帖子下插入新评论以及删除 按钮。问题是 Delete 按钮在新插入的元素上不起作用,
我有一个大约有 20 列的“管道分隔”文件。我只想使用 sha1sum 散列第一列,它是一个数字,如帐号,并按原样返回其余列。 使用 awk 或 sed 执行此操作的最佳方法是什么? Accounti
我需要将以下内容插入到我的表中...我的用户表有五列 id、用户名、密码、名称、条目。 (我还没有提交任何东西到条目中,我稍后会使用 php 来做)但由于某种原因我不断收到这个错误:#1054 - U
所以我试图有一个输入字段,我可以在其中输入任何字符,但然后将输入的值小写,删除任何非字母数字字符,留下“。”而不是空格。 例如,如果我输入: 地球的 70% 是水,-!*#$^^ & 30% 土地 输
我正在尝试做一些我认为非常简单的事情,但出于某种原因我没有得到想要的结果?我是 javascript 的新手,但对 java 有经验,所以我相信我没有使用某种正确的规则。 这是一个获取输入值、检查选择
我想使用 angularjs 从 mysql 数据库加载数据。 这就是应用程序的工作原理;用户登录,他们的用户名存储在 cookie 中。该用户名显示在主页上 我想获取这个值并通过 angularjs
我正在使用 autoLayout,我想在 UITableViewCell 上放置一个 UIlabel,它应该始终位于单元格的右侧和右侧的中心。 这就是我想要实现的目标 所以在这里你可以看到我正在谈论的
我需要与 MySql 等效的 elasticsearch 查询。我的 sql 查询: SELECT DISTINCT t.product_id AS id FROM tbl_sup_price t
我正在实现代码以使用 JSON。 func setup() { if let flickrURL = NSURL(string: "https://api.flickr.com/
我尝试使用for循环声明变量,然后测试cols和rols是否相同。如果是,它将运行递归函数。但是,我在 javascript 中执行 do 时遇到问题。有人可以帮忙吗? 现在,在比较 col.1 和
我举了一个我正在处理的问题的简短示例。 HTML代码: 1 2 3 CSS 代码: .BB a:hover{ color: #000; } .BB > li:after {
我是一名优秀的程序员,十分优秀!