篇(8)-Asp.Net Core入門實戰-實現分頁,排序和篩選功能其實很簡單

来源:https://www.cnblogs.com/mushaobai/archive/2022/11/12/16880929.html
-Advertisement-
Play Games

核心思路是使用Region的求交集和並集的結果與原始Region對比 Winform項目自帶這個類庫,如果使用控制台,需要先在nuget安裝System.Drawing.Common /// <summary> /// 計算兩個形狀的關係 /// </summary> /// <param name ...


入門實戰-實現分頁功能其實很簡單

分頁功能是項目開發中必須掌握的技能,網路上也有很多組件實現該功能,但是在Asp.Net Core中的分頁,我們學習時,還是要掌握其最基本的寫法,我計劃寫2種EF的分頁代碼。

1.第一種寫法,使用EF中的Skip()和Take()來實現分頁,Skip和Take是Linq的擴展方法,Skip(n)跳過n條數據, Take(n)獲取n條數據,只要是分頁就必須是相關的計算和變數:總記錄數,每頁條數,頁數;還以Menu的相關功能頁面來進行分頁代碼演練。

(1).先查看Menu的列表頁,也就是Index視圖,計劃在表格的下麵增加分頁功能

(2).先在視圖頁中(Index.cshtml)中規劃一下,在顯示分頁的位置上寫上代碼

(3).修改Index() Action的代碼,原來的代碼預設是不分頁的,現在需要修改為可以分頁的功能:

在Action中對一些變數的存儲使用管理ViewBag;有個小知識點註意一下:

  1. ViewData是字典類型,賦值方式用字典方式,通過key值讀取對應的value,ViewData[“myName”]
  2. ViewBag是動態類型,使用時直接添加屬性賦值即可ViewBag.myName
  3. ViewBag和ViewData只在當前Action中有效,等同於View
  4. ViewData和ViewBag中的值可以互相訪問,因為ViewBag的實現中包含了ViewData

(4).修改Index視圖頁,將其分頁代碼編寫完成。增加了一個設置:當是第一頁時,[上一頁]不可用,當時最後一頁是,[下一頁]不可用。

註意,有些是將<a>標簽調用Js函數代碼來實現Action跳轉和提交參數,如果是2個以上的參數,可以在增加一個比如asp-route-pageSize=“@ViewBag.pageSize”這樣的形式,也可以用Js,我考慮的是單一參數,先簡單的實現。<a>標簽的Action傳遞參數時,如果後臺Index的參數名是pageIndex,那麼前臺就是asp-route-pageIndex;如果後臺Index的參數名是Id,那麼前臺就是asp-route-Id;

(5).預覽分頁效果

2.第二種分頁寫法,我將微軟官方提供的分頁教程演練一下,很有特點,包括查詢框,排序和分頁。

(1).我們看下微軟官方的分頁實例圖片效果:

其教程地址為:https://learn.microsoft.com/zh-cn/aspnet/core/data/ef-mvc/sort-filter-page?view=aspnetcore-3.1

(2).先在Db文件夾下建立一個公共的分頁類;

 

代碼如下
/// <summary>
/// 分頁代碼實現設置
/// </summary>
//public class PaginatedList<T> : List<T>
public class PaginatedList<T> : List<T> where T : class
{
public int PageIndex { get; set; }
public int TotalPages { get; set; }
public PaginatedList(List<T> items, int count, int pageIndex, int pageSize)
{
PageIndex = pageIndex;
TotalPages = (int)Math.Ceiling(count / (double)pageSize);
this.AddRange(items);
}

//上一頁
public bool HasPreviousPage
{
get
{
return (PageIndex > 1);
}
}
//下一頁
public bool HasNextPage
{
get
{
return (PageIndex < TotalPages);
}
}

public static PaginatedList<T> CreateAsync(IQueryable<T> source, int pageIndex, int pageSize)
{
var count = source.Count();
var items = source.Skip((pageIndex - 1) * pageSize).Take(pageSize).ToList();
return new PaginatedList<T>(items, count, pageIndex, pageSize);
}
}

 

 

關於代碼第一行定義:class A<T> where T : new()這種類型的解釋,我引用一個網友的說明:

這是類型參數約束,where表名了對類型變數T的約束關係。where T:A 表示類型變數是繼承於A的,或者是A本身。where T : new()指明瞭創建T的實例應該使用的構造函數。

.NET支持的類型參數約束有以下五種:

where T: struct T必須是一個結構類型

where T: class T必須是一個類(class)類型,不是結構(structure)類型

where T: new() T必須要有一個無參構造函數

where T: NameOfBaseClass T必須繼承名為NameOfBaseClass的類

where T: NameOfInterface T必須實現名為NameOfInterface的介面

 

(3).按照圖例的效果,對Menu功能的視圖和Action進行改造,先進性Controller的Index()的改造:

 

public IActionResult Index(string sortOrder, string currentFilter, string searchString, int? pageNumber)
{
ViewData["CurrentSort"] = sortOrder;
ViewData["NameSortParm"] = String.IsNullOrEmpty(sortOrder) ? "name_desc" : "";
ViewData["DateSortParm"] = sortOrder == "Date" ? "date_desc" : "Date";
if (searchString != null)
{
  pageNumber = 1;
}
else
{
  searchString = currentFilter;
}
ViewData["CurrentFilter"] = searchString;
var menus = from s in _appDbContext.Menu
select s;
if (!string.IsNullOrEmpty(searchString))
{
  menus = menus.Where(s => s.DisplayName.Contains(searchString));
}
switch (sortOrder)
{
  case "name_desc":
    menus = menus.OrderByDescending(s => s.DisplayName);
    break;
  case "Date":
    menus = menus.OrderBy(s => s.AddTime);
    break;
  case "date_desc":
    menus = menus.OrderByDescending(s => s.AddTime);
    break;
  default:
    menus = menus.OrderBy(s => s.DisplayName);
    break;
}
int pageSize = 2;
return View(PaginatedList<Menu>.CreateAsync(menus.AsNoTracking(), pageNumber ?? 1, pageSize));
#region 舊代碼

//int pageSize = 2;//每頁顯示數量
//var items = _appDbContext.Menu.ToList();
//var res = _appDbContext.Menu
// .OrderBy(m => m.Id)
// .Skip((pageIndex - 1) * pageSize)
// .Take(pageSize)
// .ToList();
//ViewBag.pageIndex = pageIndex;
//ViewBag.pageSize = pageSize;
//ViewBag.totalCount = items.Count;
//ViewBag.totalPage = Math.Ceiling(items.Count * 1.0 / pageSize);
//return View(res);
#endregion

#region 舊代碼
//1.第一種查詢寫法lamada表達式方法
//var items = _appDbContext.Set<Menu>().Where(m=>m.IsDelete==false).ToList();
//var items = _appDbContext.Menu.Where(m => m.IsDelete == false).OrderByDescending(m => m.AddTime).ToList();
//2.第二種查詢寫法linq
//var items = from m in _appDbContext.Menu
// where m.IsDelete == false
// orderby m.AddTime descending
// select m;
//3.自定義sql
//var items = _appDbContext.Menu.FromSqlInterpolated($"select * from Menu order by addtime desc").ToList();
//return View(items);
#endregion
}

 

 

(4).再對Views下的視圖進行改造

 

@*
本視圖頁實現Menu的查詢功能:列出表數據和分頁,排序,篩選
*@
@using RjCms.Db;
@model PaginatedList<Menu>
<form asp-action="Index" method="get">
  <a asp-action="Create" asp-controller="Menu">新建</a>
  <input type="button" name="submit" onclick="location.href='@Url.Action("Create","Menu",new { id=3})'" value="新建" />
  <label name="SearchString">查找名稱:</label>
  <input type="text" name="SearchString" value="@ViewData["CurrentFilter"]" />
  <input type="submit" value="查詢" />
</form>

<table border="1">
<tr>
  <td>編號</td>
  <td><a asp-action="Index" asp-route-sortOrder="@ViewData["NameSortParm"]" asp-route-currentFilter="@ViewData["CurrentFilter"]">名稱</a></td>
  <td>Url</td>
  <td><a asp-action="Index" asp-route-sortOrder="@ViewData["DateSortParm"]" asp-route-currentFilter="@ViewData["CurrentFilter"]">時間</a></td>
  <td>操作</td>
</tr>
@foreach(var item in Model)
{
<tr>
  <td>@item.Id</td>
  <td>@item.DisplayName</td>
  <td>@item.LinkUrl</td>
  <td>@item.AddTime</td>
  <td>
    <a asp-action="Details" asp-controller="Menu" asp-route-id="@item.Id">Details</a>
    <a asp-action="Edit" asp-controller="Menu" asp-route-id="@item.Id">Edit</a>
    <a asp-action="Delete" asp-controller="Menu" asp-route-id="@item.Id">Delete</a>
  </td>
</tr>
}
</table>
<div>
@{
  var prevDisabled = !Model.HasPreviousPage ? "disabled" : "";
  var nextDisabled = !Model.HasNextPage ? "disabled" : ""; ;
}
<a asp-action="Index"
asp-route-sortOrder="@ViewData["CurrentSort"]"
asp-route-pageNumber="@(Model.PageIndex - 1)"
asp-route-currentFilter="@ViewData["CurrentFilter"]"
class="btn btn-default @prevDisabled">
上一頁
</a>
<a asp-action="Index"
asp-route-sortOrder="@ViewData["CurrentSort"]"
asp-route-pageNumber="@(Model.PageIndex + 1)"
asp-route-currentFilter="@ViewData["CurrentFilter"]"
class="btn btn-default @nextDisabled">
下一頁
</a>
</div>
@*<div>
共 @ViewBag.totalCount 條記錄,每頁2條:&nbsp;
<a asp-action="Index" asp-route-pageIndex="1">首頁</a>
@if (ViewBag.pageIndex == 1)
{
  <a asp-action="Index" style="pointer-events: none;cursor: default;color:gray;" >上一頁</a>
}
else
{
  <a asp-action="Index" asp-route-pageIndex="@(ViewBag.pageIndex-1)">上一頁</a>
}
@if (ViewBag.pageIndex == ViewBag.totalPage)
{
  <a asp-action="Index" style="pointer-events: none;cursor: default;color:gray;">下一頁</a>
}
else
{
  <a asp-action="Index" asp-route-pageIndex="@(ViewBag.pageIndex + 1)">下一頁</a>
}
<a asp-action="Index" asp-route-pageIndex="@ViewBag.totalPage">尾頁</a>
</div>*@

 

 

3.總結說明

兩種方式,分頁都是萬變不離其宗,頁碼,頁數,總數等概念搞清楚。然後連接指向同一個action的頁碼參數,進行變化。Skip()函數在Sql-Server 2008不被支持,有些可能需要在startup.cs文件中配置,增加UseRowNumberForPaging();但是我嘗試在.Net Core3.1中這樣修改是不行的。Skip()在Sql2012+版本是支持的。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、通過官網進行創建 官網創建鏈接:https://start.spring.io/ 1.官網頁面配置: Project:我們需要選擇項目類型,我使用的是Maven項目; Language:使用的是Java語言,選擇Java; SpringBoot:選擇版本; Project Metadata:元數 ...
  • 前言 嗨嘍,大家好呀~這裡是愛看美女的茜茜吶 又到了學Python時刻~ 我又來了!今天整個好玩的,你們肯定喜歡~ 咱們上班累了,不得好好犒勞一下自己,是吧 ! 於是我整了一手採集附近洗jio的店子,浴皇大帝們,沖鴨~ 話不多說,沖!兄弟們,都是正規的 正規的! 使用環境 python 3.8 解釋 ...
  • 一、SpringBoot是什麼 在瞭解SpringBoot之前,我們需要先來回顧一下Spring。 Spring是一個輕量級的WEB應用開發的框架,它的初衷是為了簡化企業級應用開發的複雜性,說白了就是為了簡化開發。但是隨著Spring的不斷發展,它擴展的東西越來越多,導致其配置方面也越來越複雜,在每 ...
  • 哈嘍兄弟麽,今天咱們來複習一下列表相加的兩個方法。 利用操作符+ +操作符 對象是list a = [1,2,3] b = [4,5,6] c = a+b print(c) #c的結果:[1,2,3,4,5,6] 利用extend extend 操作對象是list a = [1,2,3] b = [ ...
  • 原創:扣釘日記(微信公眾號ID:codelogs),歡迎分享,轉載請保留出處。 簡介 java8之後,常用的Map介面中添加了一些非常實用的函數,可以大大簡化一些特定場景的代碼編寫,提升代碼可讀性,一起來看看吧。 computeIfAbsent函數 比如,很多時候我們需要對數據進行分組,變成Map< ...
  • Test 作業 提交學習筆記 編寫一個業務介面(登錄,查找學生等) 打包成jar包 在本機上面執行相應的命令,通過瀏覽器訪問介面獲得數據。 (可選)將該jar包放在伺服器上面運行,通過公網IP能夠訪問 我選擇實現一個登錄的介面 這是我的項目結構 以下是我的代碼實現 項目中的pojo的實現 即mode ...
  • 1、Mybatis 是什麼 MyBatis 是一款優秀的持久層框架,它支持自定義 SQL、存儲過程以及高級映射。MyBatis 免除了幾乎所有的 JDBC 代碼以及設置參數和獲取結果集的工作。MyBatis 可以通過簡單的 XML 或註解來配置和映射原始類型、介面和 Java POJO(Plain ...
  • JZ61 撲克牌順子 描述 現在有2副撲克牌,從撲克牌中隨機五張撲克牌,我們需要來判斷一下是不是順子。 有如下規則: 1. A為1,J為11,Q為12,K為13,A不能視為14 2. 大、小王為 0,0可以看作任意牌 3. 如果給出的五張牌能組成順子(即這五張牌是連續的)就輸出true,否則就輸出f ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...