.NET-HILFE

C# Prioritäts-Warteschlange (So funktioniert es für Entwickler)

Veröffentlicht 18. Februar 2024
Teilen Sie:

Einführung

Die Programmierung in C# ist recht flexibel, und IronPDF ist eine leistungsstarke Bibliothek, die die Arbeit mit Dokumenten erleichtert, insbesondere bei der Erstellung und Änderung von PDF-Dateien. In diesem Beitrag wird die Idee einer Prioritäts-Warteschlange in C# erläutert und gezeigt, wie man sie mit IronPDF effektiv zur Optimierung von Dokumentenverarbeitungsprozessen nutzen kann. In diesem Artikel werden wir die C#-Prioritätswarteschlange zusammen mit IronPDF verwenden.

Wie verwendet man C# Prioritäts-Warteschlange

  1. Legen Sie ein neues C#-Projekt an und erstellen Sie ein Prioritätswarteschlangenobjekt.

  2. Elemente mit Prioritäten in die Warteschlange stellen.

  3. Dequeue-Element mit höchster Priorität.

  4. Werfen Sie einen Blick auf das Element mit der höchsten Priorität.

  5. Prüfen Sie, ob die Prioritätswarteschlange leer ist.

  6. Führen Sie den Code aus und entsorgen Sie die Objekte.

Prioritäts-Warteschlange

Eine Datenstruktur, die so genannte Prioritätswarteschlange, verfolgt mehrere Komponenten, denen jeweils eine Priorität zugewiesen ist. Das wesentliche Merkmal einer Prioritätswarteschlange ist, dass sie einen effizienten Abruf ermöglicht, da das Element mit der höchsten Priorität(oder niedrigster Prioritätswert, je nach Implementierung) steht immer an vorderster Front. In Situationen, in denen Aufgaben oder Elemente in einer bestimmten Reihenfolge entsprechend ihrer Priorität bearbeitet werden müssen, werden häufig Prioritätswarteschlangen verwendet.

Obwohl es in der C#-Standardbibliothek keine PriorityQueue-Klasse gibt, können Sie selbst eine erstellen oder Bibliotheken von Drittanbietern verwenden, die diese Datenstruktur anbieten. Der Array-Heap hat eine anfängliche Kapazität, und ein neuer Heap mit einer größeren Kapazität wird gebildet, wenn er voll ist, und wir versuchen, ein neues Element in die Warteschlange zu stellen. Zwei Komponenten werden in der gleichen Reihenfolge bedient, in der sie in die Warteschlange gestellt wurden, wenn sie die gleiche Priorität haben. Um Wettlaufsituationen zu vermeiden, müssen Sie Ihren eigenen Code entwickeln, um die Threadsicherheit zu gewährleisten.

Wenn Komponenten entsprechende Prioritäten haben und nach diesen Prioritäten behandelt werden müssen, bietet eine Prioritätswarteschlange in C# mehrere Vorteile.

Im Folgenden werden einige Vorteile der Verwendung einer Prioritätswarteschlange in C# aufgeführt;

  • Prioritätsreihenfolge: Die Elemente werden automatisch nach ihrer Priorität geordnet, wobei eine Prioritätswarteschlange verwendet wird. Dadurch wird die prioritätsbasierte Verarbeitung effizienter, da gewährleistet ist, dass Komponenten mit höheren Prioritäten vor denen mit niedrigeren Prioritäten behandelt werden.
  • Anpassbare Vergleiche: In Prioritäts-Warteschlangen können Sie benutzerdefinierte Vergleichsklassen verwenden oder benutzerdefinierte Vergleiche konstruieren, die es ermöglichen, Daten nach komplizierten Standards zu sortieren. Bei der Arbeit mit Objekten, die mehrere Merkmale oder eine benutzerdefinierte Prioritätslogik aufweisen, ist dies hilfreich.
  • Schneller Abruf: In den meisten Fällen erfordert das Abrufen des Elements mit der höchsten Priorität - oder der niedrigsten, je nachdem, wie es implementiert ist - eine gleichbleibende Zeitspanne. Dies ist besonders nützlich für Algorithmen, die das wichtigste Element schnell abrufen müssen.

Implementierung einer Prioritätswarteschlange in C#;

Verwenden wir einen binären Heap, um ein einfaches C#-Warteschlangensystem mit Priorität aufzubauen. Denken Sie daran, dass Sie möglicherweise bereits vorhandene Bibliotheken nutzen oder einen komplexeren Ansatz für den Produktionseinsatz in Erwägung ziehen möchten.

public class PriorityQueue<T>
{
    private List<T> elements;
    private readonly IComparer<T> comparer;
    public PriorityQueue(IComparer<T> comparer)
    {
        this.elements = new List<T>();
        this.comparer = comparer;
    }
    public int Count => elements.Count;
    public void Enqueue(T item)
    {
        elements.Add(item);
        int index = Count - 1;
        while (index > 0)
        {
            int parentIndex = (index - 1) / 2;
            if (comparer.Compare(elements[parentIndex], elements[index]) <= 0)
                break;
            Swap(index, parentIndex);
            index = parentIndex;
        }
    }
    public T Dequeue()
    {
        if (Count == 0)
            throw new InvalidOperationException("Queue is empty.");
        T front = elements[0];
        elements[0] = elements[Count - 1];
        elements.RemoveAt(Count - 1);
        int index = 0;
        while (true)
        {
            int leftChild = 2 * index + 1;
            if (leftChild >= Count)
                break;
            int rightChild = leftChild + 1;
            int minChild = (rightChild < Count && comparer.Compare(elements[rightChild], elements[leftChild]) < 0)
                ? rightChild
                : leftChild;
            if (comparer.Compare(elements[index], elements[minChild]) <= 0)
                break;
            Swap(index, minChild);
            index = minChild;
        }
        return front;
    }
    private void Swap(int i, int j)
    {
        T temp = elements[i];
        elements[i] = elements[j];
        elements[j] = temp;
    }
}
public class PriorityQueue<T>
{
    private List<T> elements;
    private readonly IComparer<T> comparer;
    public PriorityQueue(IComparer<T> comparer)
    {
        this.elements = new List<T>();
        this.comparer = comparer;
    }
    public int Count => elements.Count;
    public void Enqueue(T item)
    {
        elements.Add(item);
        int index = Count - 1;
        while (index > 0)
        {
            int parentIndex = (index - 1) / 2;
            if (comparer.Compare(elements[parentIndex], elements[index]) <= 0)
                break;
            Swap(index, parentIndex);
            index = parentIndex;
        }
    }
    public T Dequeue()
    {
        if (Count == 0)
            throw new InvalidOperationException("Queue is empty.");
        T front = elements[0];
        elements[0] = elements[Count - 1];
        elements.RemoveAt(Count - 1);
        int index = 0;
        while (true)
        {
            int leftChild = 2 * index + 1;
            if (leftChild >= Count)
                break;
            int rightChild = leftChild + 1;
            int minChild = (rightChild < Count && comparer.Compare(elements[rightChild], elements[leftChild]) < 0)
                ? rightChild
                : leftChild;
            if (comparer.Compare(elements[index], elements[minChild]) <= 0)
                break;
            Swap(index, minChild);
            index = minChild;
        }
        return front;
    }
    private void Swap(int i, int j)
    {
        T temp = elements[i];
        elements[i] = elements[j];
        elements[j] = temp;
    }
}
Public Class PriorityQueue(Of T)
	Private elements As List(Of T)
	Private ReadOnly comparer As IComparer(Of T)
	Public Sub New(ByVal comparer As IComparer(Of T))
		Me.elements = New List(Of T)()
		Me.comparer = comparer
	End Sub
	Public ReadOnly Property Count() As Integer
		Get
			Return elements.Count
		End Get
	End Property
	Public Sub Enqueue(ByVal item As T)
		elements.Add(item)
		Dim index As Integer = Count - 1
		Do While index > 0
			Dim parentIndex As Integer = (index - 1) \ 2
			If comparer.Compare(elements(parentIndex), elements(index)) <= 0 Then
				Exit Do
			End If
			Swap(index, parentIndex)
			index = parentIndex
		Loop
	End Sub
	Public Function Dequeue() As T
		If Count = 0 Then
			Throw New InvalidOperationException("Queue is empty.")
		End If
		Dim front As T = elements(0)
		elements(0) = elements(Count - 1)
		elements.RemoveAt(Count - 1)
		Dim index As Integer = 0
		Do
			Dim leftChild As Integer = 2 * index + 1
			If leftChild >= Count Then
				Exit Do
			End If
			Dim rightChild As Integer = leftChild + 1
			Dim minChild As Integer = If(rightChild < Count AndAlso comparer.Compare(elements(rightChild), elements(leftChild)) < 0, rightChild, leftChild)
			If comparer.Compare(elements(index), elements(minChild)) <= 0 Then
				Exit Do
			End If
			Swap(index, minChild)
			index = minChild
		Loop
		Return front
	End Function
	Private Sub Swap(ByVal i As Integer, ByVal j As Integer)
		Dim temp As T = elements(i)
		elements(i) = elements(j)
		elements(j) = temp
	End Sub
End Class
VB   C#

IronPDF

Mit Hilfe der .NET-Bibliothek IronPDF können Programmierer die Sprache C# verwenden, um PDF-Dokumente zu erzeugen, zu bearbeiten und zu verändern. Die Software bietet eine Reihe von Werkzeugen und Funktionen zur Erleichterung verschiedener Vorgänge mit PDF-Dateien, einschließlich, aber nicht beschränkt auf die Erstellung von PDFs aus HTML, die Konvertierung von HTML in PDF, das Kombinieren oder Teilen von PDF-Dokumenten und das Anhängen von Text, Fotos und Anmerkungen an bereits vorhandene PDFs. Weitere Informationen über IronPDF finden Sie in derIronPDF-Dokumentation.

Das Hauptmerkmal von IronPDF ist seineHTML zu PDF Konvertierung funktion, die Layouts und Stile verwaltet. Es wandelt Webinhalte in PDF-Dateien um, ideal für Berichte, Rechnungen und Dokumentationen. Dazu gehört die Umwandlung von HTML-Dateien, URLs und HTML-Strings in PDFs.

using IronPdf;

class Program
{
    static void Main(string[] args)
    {
        var renderer = new ChromePdfRenderer();

        // 1. Convert HTML String to PDF
        var htmlContent = "<h1>Hello, IronPDF!</h1><p>This is a PDF from an HTML string.</p>";
        var pdfFromHtmlString = renderer.RenderHtmlAsPdf(htmlContent);
        pdfFromHtmlString.SaveAs("HTMLStringToPDF.pdf");

        // 2. Convert HTML File to PDF
        var htmlFilePath = "path_to_your_html_file.html"; // Specify the path to your HTML file
        var pdfFromHtmlFile = renderer.RenderHtmlFileAsPdf(htmlFilePath);
        pdfFromHtmlFile.SaveAs("HTMLFileToPDF.pdf");

        // 3. Convert URL to PDF
        var url = "http://ironpdf.com"; // Specify the URL
        var pdfFromUrl = renderer.RenderUrlAsPdf(url);
        pdfFromUrl.SaveAs("URLToPDF.pdf");
    }
}
using IronPdf;

class Program
{
    static void Main(string[] args)
    {
        var renderer = new ChromePdfRenderer();

        // 1. Convert HTML String to PDF
        var htmlContent = "<h1>Hello, IronPDF!</h1><p>This is a PDF from an HTML string.</p>";
        var pdfFromHtmlString = renderer.RenderHtmlAsPdf(htmlContent);
        pdfFromHtmlString.SaveAs("HTMLStringToPDF.pdf");

        // 2. Convert HTML File to PDF
        var htmlFilePath = "path_to_your_html_file.html"; // Specify the path to your HTML file
        var pdfFromHtmlFile = renderer.RenderHtmlFileAsPdf(htmlFilePath);
        pdfFromHtmlFile.SaveAs("HTMLFileToPDF.pdf");

        // 3. Convert URL to PDF
        var url = "http://ironpdf.com"; // Specify the URL
        var pdfFromUrl = renderer.RenderUrlAsPdf(url);
        pdfFromUrl.SaveAs("URLToPDF.pdf");
    }
}
Imports IronPdf

Friend Class Program
	Shared Sub Main(ByVal args() As String)
		Dim renderer = New ChromePdfRenderer()

		' 1. Convert HTML String to PDF
		Dim htmlContent = "<h1>Hello, IronPDF!</h1><p>This is a PDF from an HTML string.</p>"
		Dim pdfFromHtmlString = renderer.RenderHtmlAsPdf(htmlContent)
		pdfFromHtmlString.SaveAs("HTMLStringToPDF.pdf")

		' 2. Convert HTML File to PDF
		Dim htmlFilePath = "path_to_your_html_file.html" ' Specify the path to your HTML file
		Dim pdfFromHtmlFile = renderer.RenderHtmlFileAsPdf(htmlFilePath)
		pdfFromHtmlFile.SaveAs("HTMLFileToPDF.pdf")

		' 3. Convert URL to PDF
		Dim url = "http://ironpdf.com" ' Specify the URL
		Dim pdfFromUrl = renderer.RenderUrlAsPdf(url)
		pdfFromUrl.SaveAs("URLToPDF.pdf")
	End Sub
End Class
VB   C#

Merkmale von IronPDF

  • HTML-zu-PDF-Konvertierung: IronPDF kann jede Art von HTML-Daten, wie z. B. Dateien, URLs und HTML-Code-Strings, in PDF-Dokumente konvertieren.
  • PDF-Erzeugung: Die Programmiersprache C# kann verwendet werden, um Text, Grafiken und andere Objekte programmatisch in PDF-Dokumente einzufügen.
  • PDF-Bearbeitung: IronPDF kann bestehende PDF-Dateien verändern und eine PDF-Datei in mehrere Dateien aufteilen. Es kann viele PDF-Dateien zu einer einzigen Datei zusammenfassen.
  • PDF-Formulare: Die Bibliothek ist hilfreich in Situationen, in denen Formulardaten gesammelt und verarbeitet werden müssen, da sie es Benutzern ermöglicht, PDF-Formulare zu erstellen und auszufüllen.
  • Sicherheitsfunktionen: IronPDF unterstützt Passwort- und Berechtigungssicherheit sowie die Verschlüsselung von PDF-Dokumenten.

Vorrangige Warteschlange mit IronPDF

Nachdem wir nun die Grundlagen einer Prioritätswarteschlange kennen, wollen wir uns ansehen, wie IronPDF und diese zusammenarbeiten können, um Dokumente schneller zu verarbeiten. Stellen Sie sich eine Situation vor, in der Sie PDF-Dokumente mit unterschiedlichen Prioritäten oder Dringlichkeitsstufen erstellen müssen.

Vorteile der Prioritätswarteschlange mit IronPDF

  • Dynamische Erstellung von Dokumenten: Sie können PDF-Dokumente je nach Dringlichkeit oder Priorität dynamisch erstellen, indem Sie eine neue Prioritätswarteschlange verwenden.
  • Effektives Workflow-Management: Um die Effizienz bei der Dokumentenerstellung zu maximieren, sorgt die Prioritätswarteschlange dafür, dass Aufträge mit höherer Priorität vor denen mit niedrigerer Priorität erledigt werden.
  • Anpassbare Priorisierung: Durch Ändern der Prioritätswerte/-stufen und -kriterien können Sie die Prioritätswarteschlange schnell an die jeweiligen Umstände anpassen.
  • Reibungslose Integration: Die prioritätsbasierte Dokumentenerzeugung kann problemlos in Ihre Anwendung integriert werden, indem IronPDF in Verbindung mit der gleichen Prioritätswarteschlange verwendet wird.
  • Skalierbarkeit: Die neue Prioritätswarteschlange wächst mit der Größe des Programms und kann mehr Aktionen im Zusammenhang mit der Erstellung von PDFs verarbeiten.

    Nachfolgend finden Sie den Beispielcode für die Prioritätswarteschlange mit IronPDF.

using IronPdf;
using System;
public class PdfGenerator
{
    static void Main()
    {
        // Create a priority queue of PDF tasks
        PriorityQueue<PdfTask> pdfTaskQueue = new PriorityQueue<PdfTask>(new PdfTaskComparer());
        // Enqueue PDF tasks with different priorities
        pdfTaskQueue.Enqueue(new PdfTask("High Priority Document", Priority.High));
        pdfTaskQueue.Enqueue(new PdfTask("Medium Priority Document", Priority.Medium));
        pdfTaskQueue.Enqueue(new PdfTask("Low Priority Document", Priority.Low));
        // Process PDF tasks in priority order
        while (pdfTaskQueue.Count > 0)
        {
            PdfTask nextTask = pdfTaskQueue.Dequeue();
            GeneratePdf(nextTask);
        }
    }
    static void GeneratePdf(PdfTask pdfTask)
    {
        // Create a new PDF document using IronPDF
        IronPdf.HtmlToPdf renderer = new IronPdf.HtmlToPdf();
        PdfDocument pdf = renderer.RenderHtmlAsPdf($"<h1>{pdfTask.Content}</h1>");
        // Save the PDF to a file
        string pdfFilePath = $"{pdfTask.Priority}_{Guid.NewGuid()}.pdf";
        pdf.SaveAs(pdfFilePath);
        // Display a message with the file path
        Console.WriteLine($"PDF generated successfully. File saved at: {pdfFilePath}");
    }
}
public class PdfTask
{
    public string Content { get; }
    public Priority Priority { get; }
    public PdfTask(string content, Priority priority)
    {
        Content = content;
        Priority = priority;
    }
}
public enum Priority
{
    Low,
    Medium,
    High
}
public class PdfTaskComparer : IComparer<PdfTask>
{
    public int Compare(PdfTask x, PdfTask y)
    {
        // Higher priority should come first
        return y.Priority.CompareTo(x.Priority);
    }
}
using IronPdf;
using System;
public class PdfGenerator
{
    static void Main()
    {
        // Create a priority queue of PDF tasks
        PriorityQueue<PdfTask> pdfTaskQueue = new PriorityQueue<PdfTask>(new PdfTaskComparer());
        // Enqueue PDF tasks with different priorities
        pdfTaskQueue.Enqueue(new PdfTask("High Priority Document", Priority.High));
        pdfTaskQueue.Enqueue(new PdfTask("Medium Priority Document", Priority.Medium));
        pdfTaskQueue.Enqueue(new PdfTask("Low Priority Document", Priority.Low));
        // Process PDF tasks in priority order
        while (pdfTaskQueue.Count > 0)
        {
            PdfTask nextTask = pdfTaskQueue.Dequeue();
            GeneratePdf(nextTask);
        }
    }
    static void GeneratePdf(PdfTask pdfTask)
    {
        // Create a new PDF document using IronPDF
        IronPdf.HtmlToPdf renderer = new IronPdf.HtmlToPdf();
        PdfDocument pdf = renderer.RenderHtmlAsPdf($"<h1>{pdfTask.Content}</h1>");
        // Save the PDF to a file
        string pdfFilePath = $"{pdfTask.Priority}_{Guid.NewGuid()}.pdf";
        pdf.SaveAs(pdfFilePath);
        // Display a message with the file path
        Console.WriteLine($"PDF generated successfully. File saved at: {pdfFilePath}");
    }
}
public class PdfTask
{
    public string Content { get; }
    public Priority Priority { get; }
    public PdfTask(string content, Priority priority)
    {
        Content = content;
        Priority = priority;
    }
}
public enum Priority
{
    Low,
    Medium,
    High
}
public class PdfTaskComparer : IComparer<PdfTask>
{
    public int Compare(PdfTask x, PdfTask y)
    {
        // Higher priority should come first
        return y.Priority.CompareTo(x.Priority);
    }
}
Imports IronPdf
Imports System
Public Class PdfGenerator
	Shared Sub Main()
		' Create a priority queue of PDF tasks
		Dim pdfTaskQueue As New PriorityQueue(Of PdfTask)(New PdfTaskComparer())
		' Enqueue PDF tasks with different priorities
		pdfTaskQueue.Enqueue(New PdfTask("High Priority Document", Priority.High))
		pdfTaskQueue.Enqueue(New PdfTask("Medium Priority Document", Priority.Medium))
		pdfTaskQueue.Enqueue(New PdfTask("Low Priority Document", Priority.Low))
		' Process PDF tasks in priority order
		Do While pdfTaskQueue.Count > 0
			Dim nextTask As PdfTask = pdfTaskQueue.Dequeue()
			GeneratePdf(nextTask)
		Loop
	End Sub
	Private Shared Sub GeneratePdf(ByVal pdfTask As PdfTask)
		' Create a new PDF document using IronPDF
		Dim renderer As New IronPdf.HtmlToPdf()
		Dim pdf As PdfDocument = renderer.RenderHtmlAsPdf($"<h1>{pdfTask.Content}</h1>")
		' Save the PDF to a file
		Dim pdfFilePath As String = $"{pdfTask.Priority}_{Guid.NewGuid()}.pdf"
		pdf.SaveAs(pdfFilePath)
		' Display a message with the file path
		Console.WriteLine($"PDF generated successfully. File saved at: {pdfFilePath}")
	End Sub
End Class
Public Class PdfTask
	Public ReadOnly Property Content() As String
	Public ReadOnly Property Priority() As Priority
	Public Sub New(ByVal content As String, ByVal priority As Priority)
		Me.Content = content
		Me.Priority = priority
	End Sub
End Class
Public Enum Priority
	Low
	Medium
	High
End Enum
Public Class PdfTaskComparer
	Implements IComparer(Of PdfTask)

	Public Function Compare(ByVal x As PdfTask, ByVal y As PdfTask) As Integer Implements IComparer(Of PdfTask).Compare
		' Higher priority should come first
		Return y.Priority.CompareTo(x.Priority)
	End Function
End Class
VB   C#

In diesem Fall werden PDF-Aufträge mit unterschiedlichen Prioritäten in die Prioritäts-Warteschlange eingereiht(pdfTaskQueue) durch die Klasse "PdfGenerator". Aufträge mit höherer Priorität werden dank der PriorityQueue zuerst bearbeitet. Wir verwenden die Enqueue-Methode, um das Element zur Prioritätswarteschlange hinzuzufügen. Wir können auch die Dequeue-Methode verwenden, um die Werte mit der höchsten Priorität zu entfernen und abzurufen. Wir können die Peek-Methode verwenden, um die höchste Priorität anzuzeigen, ohne das Element zu entfernen.

C# Prioritäts-Warteschlange (So funktioniert es für Entwickler): Abbildung 1

Beispielhafte Ausgabedatei:

C# Prioritäts-Warteschlange (So funktioniert es für Entwickler): Abbildung 2

Basierend auf dem Inhalt des Auftrags erstellt die Funktion GeneratePdf mit Hilfe von IronPDF ein PDF-Dokument, das anschließend in einer Datei gespeichert wird. Wenn Sie mehr über den IronPDF-Code erfahren möchten, lesen Sie bitte dieIronPDF HTML zu PDF Beispiele.

Schlussfolgerung

Wenn IronPDF und eine Prioritätswarteschlange in einer C#-Anwendung kombiniert werden, ist es möglich, Dokumente schnell und dynamisch nach verschiedenen Prioritäten oder Dringlichkeitsstufen zu erzeugen. Diese Methode funktioniert besonders gut in Situationen, in denen einige Dokumente vor anderen bearbeitet und bereitgestellt werden müssen.

Ihre Anwendung kann Aufträge zur Dokumentenerstellung auf flexible, skalierbare und nach Prioritäten geordnete Weise verwalten, indem sie die HTML-zu-PDF-Konvertierungsfunktionen von IronPDF nutzt und eine Prioritätswarteschlange zur Aufgabenverwaltung integriert. Der Rahmen für die Integration dieser Ideen wurde in diesem Artikel abgesteckt; können zusätzliche Anpassungen vorgenommen werden, um den besonderen Anforderungen Ihrer Dokumentenverarbeitung gerecht zu werden. Durch die Kombination von IronPDF mit einer Prioritäts-Warteschlange erhalten Sie eine effektive und priorisierte Lösung für die Handhabung von Dokumenten in C#, unabhängig davon, ob Sie ein Berichtstool, ein Dokumentenverwaltungssystem oder eine andere Anwendung entwickeln, die PDFs erzeugt.

Die $749 Lite-Edition von IronPDF beinhaltet ein Jahr Software-Wartung, Upgrade-Optionen und eine permanente Lizenz. Während des Testzeitraums mit Wasserzeichen können die Benutzer das Produkt in realen Situationen testen. Weitere Informationen zu den Kosten, der Lizenzierung und der kostenlosen Testversion von IronPDF finden Sie in derIronPDF-Lizenzierungsinformationen. Weitere Informationen über Iron Software finden Sie auf derIron Software Website.

< PREVIOUS
C# LINQ Distinct (Wie es für Entwickler funktioniert)
NÄCHSTES >
C# Sealed Class (Wie es für Entwickler funktioniert)

Sind Sie bereit, loszulegen? Version: 2024.12 gerade veröffentlicht

Gratis NuGet-Download Downloads insgesamt: 11,810,873 Lizenzen anzeigen >