日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 运维知识 > linux >内容正文

linux

Dotnet创建Linux下的Service应用

發(fā)布時(shí)間:2023/12/4 linux 34 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Dotnet创建Linux下的Service应用 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

創(chuàng)建Service應(yīng)用,是一個(gè)服務(wù)端開發(fā)的必會(huì)技能。

?

前言

說(shuō)到服務(wù)端應(yīng)用,最常見的就是API服務(wù)。

除此之外,還有一類應(yīng)用,比方一個(gè)Socket的服務(wù)器。這類型的應(yīng)用,本身沒有Web層,當(dāng)然也不屬于API服務(wù)。

通常大家會(huì)怎么做?

不講究的做法,就是做一個(gè)Console應(yīng)用,加載到后臺(tái)一直跑著。

其實(shí),還有另外一種做法,就是把應(yīng)用加載到Services里,使應(yīng)用以一個(gè)Service來(lái)做響應(yīng)。這樣可以依托操作系統(tǒng)的Services管理器來(lái)進(jìn)行統(tǒng)一管理,自動(dòng)運(yùn)行和故障處理。

?

Dotnet做Window Service的內(nèi)容,網(wǎng)上有很多。我今天寫一個(gè)在Linux下做Service的方法。

創(chuàng)建Linux下的Service應(yīng)用

創(chuàng)建一個(gè)LInux下的Service應(yīng)用其實(shí)很簡(jiǎn)單,就分這么幾步:

1. 用 Worker 模板創(chuàng)建工程

如果習(xí)慣用VS上創(chuàng)建,就找一下Worker Service模板。

我是習(xí)慣從命令行創(chuàng)建,就一條命令:

%?dotnet?new?worker?-o?projectname

Dotnet會(huì)自動(dòng)造成工程,并自動(dòng)引用Microsoft.Extensions.Hosting包,因?yàn)檫@本身是一個(gè)Self-Hosting應(yīng)用。

2. 加入Linux Service擴(kuò)展包

其實(shí)這就是一個(gè)包:Microsoft.Extensions.Hosting.Systemd。這個(gè)包為應(yīng)用提供了在Linux下使用Systemd守護(hù)進(jìn)程的基礎(chǔ)配置。

還是命令行:

%?dotnet?add?package?Microsoft.Extensions.Hosting.Systemd

3. 修改Program.cs

其實(shí)就是一行代碼,把第二步引入的包加入應(yīng)用。修改Program.cs

public?static?IHostBuilder?CreateHostBuilder(string[]?args)?=>Host.CreateDefaultBuilder(args).UseSystemd()??//?加入的就是這一行。.ConfigureServices((hostContext,?services)?=>{services.AddHostedService<Worker>();});

?

到這兒,套路性的工作已經(jīng)完成。簡(jiǎn)單吧?

?

我們來(lái)看一下現(xiàn)在的工程:

├──?Program.cs ├──?Properties │???└──?launchSettings.json ├──?Worker.cs ├──?appsettings.Development.json ├──?appsettings.json └──?workerdemo.csproj

大家會(huì)注意到,里面多了一個(gè)Worker.cs的類文件。

看一下這個(gè)文件:

public?class?Worker?:?BackgroundService {private?readonly?ILogger<Worker>?_logger;public?Worker(ILogger<Worker>?logger){_logger?=?logger;}protected?override?async?Task?ExecuteAsync(CancellationToken?stoppingToken){while?(!stoppingToken.IsCancellationRequested){_logger.LogInformation("Worker?running?at:?{time}",?DateTimeOffset.Now);await?Task.Delay(1000,?stoppingToken);}} }

這其實(shí)就是加載到Systemd里的服務(wù)的模板。我們需要的服務(wù)代碼,需要加到ExecuteAsync(CancellationToken stoppingToken)方法中。

?

我簡(jiǎn)單做個(gè)例子,在里面加入U(xiǎn)DP服務(wù),看代碼:

public?class?Worker?:?BackgroundService {private?readonly?ILogger<Worker>?_logger;private?readonly?IConfiguration?_configuration;public?Worker(ILogger<Worker>?logger,?IConfiguration?configuration){_logger?=?logger;_configuration?=?configuration;}protected?override?async?Task?ExecuteAsync(CancellationToken?stoppingToken){_logger.LogInformation("Worker?running?at:?{time}",?DateTimeOffset.Now);UdpClient?udpClient?=?new?UdpClient(new?IPEndPoint(IPAddress.Parse("127.0.0.1"),?8000));while?(!stoppingToken.IsCancellationRequested){UdpReceiveResult?udpReceiveResult?=?await?udpClient.ReceiveAsync();string?message?=?Encoding.UTF8.GetString(udpReceiveResult.Buffer);Console.WriteLine($"{udpReceiveResult.RemoteEndPoint.ToString()}?-?{message}");await?udpClient.SendAsync(Encoding.Default.GetBytes("Got"),?3,?udpReceiveResult.RemoteEndPoint);}} }

這個(gè)代碼中,有兩件事需要注意:

  • 在前邊Program.cs中加入U(xiǎn)seSystemd()時(shí),已經(jīng)注入了IConfiguration。因此,可以在這個(gè)方法中直接引入并使用。換句話說(shuō),就是可以直接讀取例如appsetting.json的內(nèi)容;

  • 是上邊提到的,真正的服務(wù)響應(yīng)在ExecuteAsync(CancellationToken stoppingToken)中。這兒沒什么特別的,就是正常的寫法。

  • ?

    上面這個(gè),是服務(wù)端的程序,是響應(yīng)。

    下面我簡(jiǎn)單做個(gè)客戶端的請(qǐng)求,供測(cè)試用。就不解釋了,只列出步驟:

  • 創(chuàng)建一個(gè)工程

  • %?dotnet?new?console?-o?democlient
  • 修改Program.cs

  • static?async?Task?Main(string[]?args) {UdpClient?udpClient?=?new?UdpClient();for?(int?i?=?0;?i?<?10000;?i++){byte[]?buffer?=?new?byte[8?*?1024];await?Task.Run(()?=>{udpClient.SendAsync(buffer,?buffer.Length,?new?IPEndPoint(IPAddress.Parse("127.0.0.1"),?8000));});}while?(true){UdpReceiveResult?udpReceiveResult?=?await?udpClient.ReceiveAsync();string?message?=?Encoding.UTF8.GetString(udpReceiveResult.Buffer);Console.WriteLine($"{udpReceiveResult.RemoteEndPoint.ToString()}?-?{message}");}Console.ReadKey(); }

    運(yùn)行一下,看看效果。

    ?

    到這里,Service應(yīng)用開發(fā)的工作已經(jīng)完成。

    下面是部署。

    部署Service應(yīng)用

    Linux下面部署一個(gè)Service應(yīng)用,只有兩個(gè)步驟:

    1. 創(chuàng)建Service定義

    Linux下的每個(gè)Service,都會(huì)有個(gè)定義文件。這個(gè)文件存在于/etc/systemd/system目錄下。

    下面我給出一個(gè)簡(jiǎn)單的Service模板:

    [Unit] Description=DemoProject[Service] Type=notify ExecStart=dotnet?/yourfolder/yourproject.dll[Install] WantedBy=multi-user.target

    把這個(gè)內(nèi)容保存為一個(gè)文件,例如叫demo.service。然后把這個(gè)文件復(fù)制到/etc/systemd/system下,并改為可執(zhí)行。

    簡(jiǎn)單說(shuō)一下這個(gè)文件的一些項(xiàng):

    • Description,是服務(wù)的名字。不重要,啟動(dòng)時(shí),你用到的是文件名demo.service;

    • Type,服務(wù)類型,使用Dotnet加載時(shí),只能是這種類型。如果把程序編譯為自包含程序,這個(gè)類型可以是simple;

    • ExecStart,啟動(dòng)程序的命令,是全路徑的,要確保能找得到這個(gè)程序。上面例子中,dotnet /yourfolder/yourproject.dll,是因?yàn)閐otnet命令是有PATH變量支持的。

    這個(gè)文件的配置項(xiàng)有很多,包括定義是否需要自動(dòng)重啟、重啟間隔等。如果需要,可以去這里查詢。

    2. 啟動(dòng)Service

    有兩種方法。

    第一種是刷新Service守護(hù)

    %?systemctl?daemon-reload

    刷新守護(hù)時(shí),守護(hù)進(jìn)程會(huì)去/etc/systemd/system目錄下,尋找新加入的Service文件,并啟動(dòng)。

    第二種是單獨(dú)啟動(dòng),有一系列命令:

    • 啟動(dòng)

    %?systemctl?start?demo.service
    • 停止

    %?systemctl?stop?demo.service
    • 重啟

    %?systemctl?restart?demo.service
    • 查詢狀態(tài)

    %?systemctl?status?demo.service

    嗯。這就是服務(wù)加載和停止了。

    ?

    注意,這種方式加載的Service,是完全系統(tǒng)的服務(wù),會(huì)沒有任何輸出。

    如果需要調(diào)試,一種方式是加文件日志,另一種方式是用另一個(gè)命令啟動(dòng):

    %?journalctl?-u?dnsserver.service

    當(dāng)然,這種方式只用于調(diào)試。正式運(yùn)行時(shí),還應(yīng)該是上面的方式。

    ?

    這就是今天的內(nèi)容,希望能幫到大家。感覺有用的話,給個(gè)三連唄~

    喜歡就來(lái)個(gè)三連,讓更多人因你而受益

    總結(jié)

    以上是生活随笔為你收集整理的Dotnet创建Linux下的Service应用的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。