標籤: 南投搬家公司費用

  • 邁向碳中合歐盟 擬設統一能源部門

    摘錄自2020年7月13日台灣醒報報導

    歐洲議會推設立「統一能源部門」,促歐盟各國實現碳中合目標!奧地利的歐洲議員嘉夢日前在一場會議中,提出設立「歐洲統一能源部門」,將協助整合各國現有的能源法規、打造歐盟再生能源中心及促進工業及交通運輸轉型為使用綠色能源,獲得多數議員支持。

    歐盟許多成員國包括芬蘭、丹麥已設下2050年以前要實現「碳中合」的目標,即便是將脫歐的英國也曾承諾跟進。根據歐盟調查,為了實現碳中合目標,歐盟整體的用電量將從25%提升到50%,嘉夢說:「實現碳中合,非常需要各國政府機關朝使用綠色能源邁進,因此,我們需要一個部門負責協調整合。」

    除了統一各國現有的能源政策及法規外,該部門也將負責打造歐盟再生能源中心、提升各國的太陽能、生物能發電等基礎設施的普及率、研究並部署綠色能源科技,以及協助將乾淨能源的使用範圍擴張到傳統上依賴化石燃料的領域中,例如提倡綠色建築、工業和交通運輸。

    能源議題
    再生能源
    能源轉型
    國際新聞
    歐盟
    碳中和

    本站聲明:網站內容來源環境資訊中心https://e-info.org.tw/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    網頁設計公司推薦不同的風格,搶佔消費者視覺第一線

    ※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

    ※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

    南投搬家公司費用需注意的眉眉角角,別等搬了再說!

    ※教你寫出一流的銷售文案?

    ※回頭車貨運收費標準

    ※別再煩惱如何寫文案,掌握八大原則!

  • 伊朗火災與爆炸頻傳 伊斯法罕省電廠也傳事故

    摘錄自2020年7月19日中央社報導

    伊朗中部伊斯法罕省一座電廠今天(19日)發生爆炸。所幸沒有人員傷亡。電廠過了約兩小時恢復正常運作,伊斯法罕省供電不受影響。

    伊朗各地的軍事與民用設施近幾週頻傳火災與爆炸事故,讓伊朗境內人士懷疑可能是宿敵以色列在搞破壞。以色列指控伊朗試圖發展核彈,伊朗則堅稱本身的核子計畫完全是為了和平目的。

    首都德黑蘭(Tehran)6月下旬就發生兩起爆炸,其中一起發生在軍事用地附近,另一起衛生中心的火災則奪走19人性命。本月稍早,伊朗南部一座造船廠、德黑蘭南方一座工廠,以及伊朗中部的納坦茲(Natanz)核設施,也分別發生火災或爆炸事故。其中工廠事故奪走兩人性命。伊朗當局形容納坦茲核設施起火是意外,他們基於「安全理由」不公布事發原因及細節。

    土地利用
    能源轉型
    國際新聞
    伊朗
    火力發電廠
    火災
    災害
    核能

    本站聲明:網站內容來源環境資訊中心https://e-info.org.tw/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    USB CONNECTOR掌控什麼技術要點? 帶您認識其相關發展及效能

    台北網頁設計公司這麼多該如何選擇?

    ※智慧手機時代的來臨,RWD網頁設計為架站首選

    ※評比南投搬家公司費用收費行情懶人包大公開

    ※回頭車貨運收費標準

  • 新種海蟑螂出現!酷似星際大戰「黑武士」

    新種海蟑螂出現!酷似星際大戰「黑武士」

    摘錄自2020年7月18日自由時報報導

    新加坡與印尼研究團隊將2018年發現一種外貌如星際大戰「黑武士」的14足生物,認證為新品種海蟑螂。綜合媒體報導,新加坡國立大學教授黃䙫麟(Peter K. L. Ng)從2018年與印尼科學院合作,共同探勘印尼西爪哇外海63個地點後,發現12個未登錄在科學文獻的新物種。

    本(7)月8日,黃䙫麟與印尼團隊的論文登上生物學期刊《ZooKeys》,將發現的一種甲殼類生物命為「Bathynomus raksasa」,是「大王巨足蟲屬」(Bathynomus)的一種,其長相雖貌似陸地的蟑螂或鼠婦(woodlice of land),實際上與螃蟹、蝦子等海生動物關係更近。

    一般等足動物長約33公分,但由於天敵稀少、深海環境寒冷,「Bathynomus raksasa」的身體能夠長到50公分左右,為目前科學界已知第二長的等足動物,僅次於「大王巨足蟲」(Bathynomus giganteus)。



    新加坡與印尼研究團隊近日在學術期刊發表論文,將2018年發現一種外貌如星際大戰「黑武士」的14足生物,認證為新品種海蟑螂。圖片來源:Twitter(galamedianews.com)


    生物多樣性
    國際新聞
    印尼
    新加坡
    新物種

    本站聲明:網站內容來源環境資訊中心https://e-info.org.tw/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    網頁設計公司推薦不同的風格,搶佔消費者視覺第一線

    ※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

    ※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

    南投搬家公司費用需注意的眉眉角角,別等搬了再說!

    ※教你寫出一流的銷售文案?

  • 簡單5步,輕鬆debug K8S服務!

    簡單5步,輕鬆debug K8S服務!

    作者:
    Ram Rai,性能、可擴展性以及軟件架構的愛好者

    原文鏈接:
    https://medium.com/better-programming/debug-your-kubernetes-service-in-5-easy-steps-1457974f024c

    在Kubernetes中,服務是一個核心概念。在本文中,將介紹如何調試K8S服務,這些服務是由多個Pod組成的工作負載的抽象接口(主機+端口)。

    在我們深入探索debug方法之前,我們先簡單回顧一下網絡,這是Kubernetes服務的基礎。

    • 在一個pod中的容器共享相同的網絡空間和IP。

    • 所有的pod都能通過IP彼此通信。

    • 每個節點都能看到所有的Pod,反之亦然。

    • Pod可以看到所有的服務。

    那麼,在實踐中這些意味着什麼呢?

    在圖中:

    • 位於Pod1中的容器B可以直接作為localhost尋址容器A

    • 容器B可以通過其IP直接尋址Pod2(kubectl get pod -o wide)。我們知道當pod2出現故障時着不是一個可靠的通信渠道,並且一個新的pod可以出現在其位置中。但是我們無法追逐不斷變化的目標。

    • 接下來,容器B可以通過Service x訪問pod 2和pod 3,後者將它們的IP與負載均衡捆綁在一起;因此,在K8S上支持基於微服務的應用程序起着至關重要的作用

    儘管對Kubernetes的內部網絡結構的檢查不在本文的討論範圍內,但我稍後會發布一些參考資料以供大家進一步研究。

    對於當下,我還是鼓勵你花費一點時間在實踐中經歷和理解Kubernetes中的網絡。例如,你可以啟動一個Kubernetes測試pod並且嘗試從該pod中訪問其他pod、節點和服務。此處显示的命令將在Pod內彈出一個Linux shell。

    kubectl run -it networktest --image=alpine bin/ash --restart=Never --rm
    

    現在你在Kubernetes網絡空間內並且你可以隨意使用wegtpingnslookup之類的命令進行實驗。例如,測試你的Kubernetes集群中先前列出的網絡要求,nslookup <servicename>, ping <PodIP>

    現在讓我們回到我們的話題,troubleshooting Kubernetes服務,這實際上是一種網絡結構。

    Step1:檢查服務是否存在

    kubectl get svc
    

    如果服務不存在,應該是服務創建出現了故障,因此要去檢查你的服務定義。

    Step2:測試你的服務

    請記住,一個內部的Kubernetes ClusterIP服務是無法在集群外部訪問的。因此,有兩種方法可以對其進行測試。方法一,你可以啟動一個測試Pod,通過SSH進入該pod,然後嘗試像這樣訪問你的服務:

    kubectl run -it testpod --image=alpine bin/ash --restart=Never --rm
    

    在本文中我們啟動一個alpine Docker鏡像作為pod來從其內部測試服務:

    #works for http services
    wget <servicename>:<httpport>
    
    #Confirm there is a DNS entry for the service!
    nslookup <servicename>
    

    或者,你可以轉發到本地計算機並在本地進行測試。

    kubectl port-forward <service_name> 8000:8080
    

    現在,你可以通過localhost:8000訪問服務。

    Step3:檢查服務是否target相關Pod

    Kubernetes服務會根據標籤selector將入站流量路由到其中一個pod,流量通過其IP路由到目標Pod。所以,請檢查服務是否綁定到那些pod。

    kubectl describe service <service-name> | grep Endpoints
    

    執行上述命令之後,你應該看到與列出的工作負載相關的所有Pod的IP。如果沒有看到,請執行Step4。

    Step4:檢查Pod標籤

    確保在Kubernetes服務中的selector與pod的標籤相匹配。

    kubectl get pods --show-labels
    kubectl describe svc <service_name>
    

    從下面的截圖的中可以看到,pod的標籤在右邊。四個pod被標記為app=tinywebsitetier=frontend,這些標籤與下面“described”的服務selector相匹配。

    在這四個匹配的Pod中,只有三個正在運行,其IP在突出显示的行中被列為服務的端點(endpoint)。你還可以在IP列中看到相同的IP。

    Step5:確認服務端口與pod相匹配

    最後,確保在你的pod中的代碼能夠監聽到你為服務指定的targetPort(例如,你在上方截圖中看到的port8001)!

    這十分簡單,為了讓你更進一步深入了解和研究Kubernetes的網絡世界,歡迎你閱讀以下文章。

    • 在Kubernetes中部署一個應用程序

    • Debug服務

    • Kubernetes網絡

    本站聲明:網站內容來源於博客園,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    網頁設計一頭霧水該從何著手呢? 台北網頁設計公司幫您輕鬆架站!

    網頁設計公司推薦不同的風格,搶佔消費者視覺第一線

    ※想知道購買電動車哪裡補助最多?台中電動車補助資訊懶人包彙整

    南投搬家公司費用,距離,噸數怎麼算?達人教你簡易估價知識!

    ※教你寫出一流的銷售文案?

    ※超省錢租車方案

  • .Net Core實戰之基於角色的訪問控制的設計,.Net微服務實戰之技術架構分層篇,.Net微服務實戰之技術選型篇

    .Net Core實戰之基於角色的訪問控制的設計,.Net微服務實戰之技術架構分層篇,.Net微服務實戰之技術選型篇

    前言

      上個月,我寫了兩篇微服務的文章:《.Net微服務實戰之技術架構分層篇》與《.Net微服務實戰之技術選型篇》,微服務系列原有三篇,當我憋第三篇的內容時候一直沒有靈感,因此先打算放一放。

      本篇文章與源碼原本打算實在去年的時候完成併發布的,然而我一直忙於公司項目的微服務的實施,所以該篇文章一拖再拖。如今我花了點時間整理了下代碼,並以此篇文章描述整個實現思路,並開放了源碼給予需要的人一些參考。

      源碼:https://github.com/SkyChenSky/Sikiro.RBAC

    RBAC

      Role-Based Access Contro翻譯成中文就是基於角色的訪問控制,文章以下我都用他的簡稱RBAC來描述。

      現信息系統的權限控制大多數採取RBAC的思想進行實現,其本質思想是對系統各種的操作權限不是直接授予具體的某個用戶,而是在用戶集合與權限集合之間建立一個角色,作為間接關聯。每一種角色對應一組相應的權限。一旦用戶被分配了適當的角色后,該用戶就擁有此角色的所有操作權限。

      通過以上的描述,我們可以分析出以下信息:

    •   用戶與權限是通過角色間接關聯的
    •   角色的本質就是權限組(權限集合)

      這樣做的好處在於,不必在每次創建用戶時都進行分配權限的操作,只要分配用戶相應的角色即可,而且角色的權限變更比用戶的權限變更要少得多,這樣將簡化用戶的權限管理,減少系統的開銷。

      

    功能分析

    權限分類

    從權限的作用可以分為三種,功能權限、訪問權限、數據權限

    • 功能權限
      • 功能權限指系統用戶允許在頁面進行按鈕操作的權限。如果有權限則功能按鈕展示,否則隱藏。
    • 訪問權限
      • 訪問權限指系統用戶通過點擊按鈕後進行地址的請求訪問的權限(地址跳轉與接口請求),如果無權限訪問,則由頁面提示無權限訪問。
    • 數據權限
      • 數據權限指用戶可訪問系統的數據權限,不同的用戶可以訪問不同的數據粒度。

    數據權限的實現可大可小,大可大到對條件進行動態配置,小可小到只針對某個維度進行硬編碼。不納入這次的討論範圍。

    用例圖

    非功能性需求

      時效性,直接影響到安全性,既然是權限控制,那麼理應一修改權限后就立刻生效。曾經有同行問過我,是不是每一個請求都得去查一次數據庫是否滿足權限,如果是,數據庫壓力豈不是很大?

      安全性,每一個頁面跳轉,每一個讀寫請求都的進行一次權限驗證,不滿足的權限的功能按鈕就不需要渲染,避免樣式display:none的情況。

      開發效率,權限控制理應是框架層面的,因此盡可能作為非業務的侵入性,讓開發人員保持原有的數據善增改查與頁面渲染。

    技術選型

    LayUI

      學習門檻極低,開箱即用。其外在極簡,卻又不失飽滿的內在,體積輕盈,組件豐盈,從核心代碼到 API 的每一處細節都經過精心雕琢,非常適合界面的快速開發,它更多是為服務端程序員量身定做,無需涉足各種前端工具的複雜配置,只需面對瀏覽器本身,讓一切你所需要的元素與交互,從這裏信手拈來。作為國人的開源項目,完整的接口文檔與Demo示例讓入門者非常友好的上手,開箱即用的Api讓學習成本盡可能的低,其易用性成為快速開發框架的基礎。

    MongoDB

      主要兩大優勢,無模式與橫向擴展。對於權限模塊來說,無需SQL來寫複雜查詢和報表,也不需要使用到多表的強事務,上面提到的時效性的數據庫壓力問題也可以通過分片解決。無模式使得開發人員無需預定義存儲結構,結合MongoDB官方提供的驅動可以做到快速的開發。

    數據庫設計

     E-R圖

     

      一個管理員可以擁有多個角色,因此管理員與角色是一對多的關聯;角色作為權限組的存在,又可以選擇多個功能權限值與菜單,所以角色與菜單、功能權限值也是一對多的關係。

    類圖

    Deparment與Position屬於非核心,可以按照自己的實際業務進行擴展。

    功能權限值初始化

      隨着業務發展,需求功能是千奇百怪的,根本無法抽象出來,那麼功能按鈕就要隨着業務進行定義。在我的項目里使用了枚舉值進行定義每個功能權限,通過自定義的PermissionAttribute與響應的action進行綁定,在系統啟動時,通過反射把功能權限的枚舉值與相應的controller、action映射到MenuAction表,枚舉值對應code字段,controller與action拼接后對應url字段。

      已初始化到數據庫的權限值可以到菜單頁把相對應的菜單與權限通過用戶界面關聯起來。

    權限值綁定action

    1         [HttpPost]
    2         [Permission(PermCode.Administrator_Edit)]
    3         public IActionResult Edit(EditModel edit)
    4         {
    5             //do something
    6 
    7             return Json(result);
    8         }

    初始化權限值

     1     /// <summary>
     2     /// 功能權限
     3     /// </summary>
     4     public static class PermissionUtil
     5     {
     6         public static readonly Dictionary<string, IEnumerable<int>> PermissionUrls = new Dictionary<string, IEnumerable<int>>();
     7         private static MongoRepository _mongoRepository;
     8 
     9         /// <summary>
    10         /// 判斷權限值是否被重複使用
    11         /// </summary>
    12         public static void ValidPermissions()
    13         {
    14             var codes = Enum.GetValues(typeof(PermCode)).Cast<int>();
    15             var dic = new Dictionary<int, int>();
    16             foreach (var code in codes)
    17             {
    18                 if (!dic.ContainsKey(code))
    19                     dic.Add(code, 1);
    20                 else
    21                     throw new Exception($"權限值 {code} 被重複使用,請檢查 PermCode 的定義");
    22             }
    23         }
    24 
    25         /// <summary>
    26         /// 初始化添加預定義權限值
    27         /// </summary>
    28         /// <param name="app"></param>
    29         public static void InitPermission(IApplicationBuilder app)
    30         {
    31             //驗證權限值是否重複
    32             ValidPermissions();
    33 
    34             //反射被標記的Controller和Action
    35             _mongoRepository = (MongoRepository)app.ApplicationServices.GetService(typeof(MongoRepository));
    36 
    37             var permList = new List<MenuAction>();
    38             var actions = typeof(PermissionUtil).Assembly.GetTypes()
    39                 .Where(t => typeof(Controller).IsAssignableFrom(t) && !t.IsAbstract)
    40                 .SelectMany(t => t.GetMethods(BindingFlags.Instance | BindingFlags.Public | BindingFlags.DeclaredOnly));
    41 
    42             //遍歷集合整理信息
    43             foreach (var action in actions)
    44             {
    45                 var permissionAttribute =
    46                     action.GetCustomAttributes(typeof(PermissionAttribute), false).ToList();
    47                 if (!permissionAttribute.Any())
    48                     continue;
    49 
    50                 var codes = permissionAttribute.Select(a => ((PermissionAttribute)a).Code).ToArray();
    51                 var controllerName = action?.ReflectedType?.Name.Replace("Controller", "").ToLower();
    52                 var actionName = action.Name.ToLower();
    53 
    54                 foreach (var item in codes)
    55                 {
    56                     if (permList.Exists(c => c.Code == item))
    57                     {
    58                         var menuAction = permList.FirstOrDefault(a => a.Code == item);
    59                         menuAction?.Url.Add($"{controllerName}/{actionName}".ToLower());
    60                     }
    61                     else
    62                     {
    63                         var perm = new MenuAction
    64                         {
    65                             Id = item.ToString().EncodeMd5String().ToObjectId(),
    66                             CreateDateTime = DateTime.Now,
    67                             Url = new List<string> { $"{controllerName}/{actionName}".ToLower() },
    68                             Code = item,
    69                             Name = ((PermCode)item).GetDisplayName() ?? ((PermCode)item).ToString()
    70                         };
    71                         permList.Add(perm);
    72                     }
    73                 }
    74                 PermissionUrls.TryAdd($"{controllerName}/{actionName}".ToLower(), codes);
    75             }
    76 
    77             //業務功能持久化
    78             _mongoRepository.Delete<MenuAction>(a => true);
    79             _mongoRepository.BatchAdd(permList);
    80         }
    81 
    82         /// <summary>
    83         /// 獲取當前路徑
    84         /// </summary>
    85         /// <param name="filterContext"></param>
    86         /// <returns></returns>
    87         public static string CurrentUrl(HttpContext filterContext)
    88         {
    89             var url = filterContext.Request.Path.ToString().ToLower().Trim('/');
    90             return url;
    91         }
    92     }

    關聯菜單與功能權限

    訪問權限

      當所有權限關係關聯上后,用戶訪問系統時,需要對其所有操作進行攔截與實時的權限判斷,我們註冊一個全局的GlobalAuthorizeAttribute,其主要攔截所有已經標識PermissionAttribute的action,查詢該用戶所關聯所有角色的權限是否滿足允許通過。

      我的實現有個細節,給判斷用戶IsSuper==true,也就是超級管理員,如果是超級管理員則繞過所有判斷,可能有人會問為什麼不在角色添加一個名叫超級管理員進行判斷,因為名稱是不可控的,在代碼邏輯里並不知道用戶起的所謂的超級管理員,就是我們需要繞過驗證的超級管理員,假如他叫無敵管理員呢?

     1  /// <summary>
     2     /// 全局的訪問權限控制
     3     /// </summary>
     4     public class GlobalAuthorizeAttribute : System.Attribute, IAuthorizationFilter
     5     {
     6         #region 初始化
     7         private string _currentUrl;
     8         private string _unauthorizedMessage;
     9         private readonly List<string> _noCheckPage = new List<string> { "home/index", "home/indexpage", "/" };
    10 
    11         private readonly AdministratorService _administratorService;
    12         private readonly MenuService _menuService;
    13 
    14         public GlobalAuthorizeAttribute(AdministratorService administratorService, MenuService menuService)
    15         {
    16             _administratorService = administratorService;
    17             _menuService = menuService;
    18         } 
    19         #endregion
    20 
    21         public void OnAuthorization(AuthorizationFilterContext context)
    22         {
    23             context.ThrowIfNull();
    24 
    25             _currentUrl = PermissionUtil.CurrentUrl(context.HttpContext);
    26 
    27             //不需要驗證登錄的直接跳過
    28             if (context.Filters.Count(a => a is AllowAnonymousFilter) > 0)
    29                 return;
    30 
    31             var user = GetCurrentUser(context);
    32             if (user == null)
    33             {
    34                 if (_noCheckPage.Contains(_currentUrl))
    35                     return;
    36 
    37                 _unauthorizedMessage = "登錄失效";
    38 
    39                 if (context.HttpContext.Request.IsAjax())
    40                     NoUserResult(context);
    41                 else
    42                     LogoutResult(context);
    43                 return;
    44             }
    45 
    46             //超級管理員跳過
    47             if (user.IsSuper)
    48                 return;
    49 
    50             //賬號狀態判斷
    51             var administrator = _administratorService.GetById(user.UserId);
    52             if (administrator != null && administrator.Status != EAdministratorStatus.Normal)
    53             {
    54                 if (_noCheckPage.Contains(_currentUrl))
    55                     return;
    56 
    57                 _unauthorizedMessage = "親~您的賬號已被停用,如有需要請您聯繫系統管理員";
    58 
    59                 if (context.HttpContext.Request.IsAjax())
    60                     AjaxResult(context);
    61                 else
    62                     AuthResult(context, 403, GoErrorPage(true));
    63 
    64                 return;
    65             }
    66 
    67             if (_noCheckPage.Contains(_currentUrl))
    68                 return;
    69 
    70             var userUrl = _administratorService.GetUserCanPassUrl(user.UserId);
    71 
    72             // 判斷菜單訪問權限與菜單訪問權限
    73             if (IsMenuPass(userUrl) && IsActionPass(userUrl))
    74                 return;
    75 
    76             if (context.HttpContext.Request.IsAjax())
    77                 AuthResult(context, 200, GetJsonResult());
    78             else
    79                 AuthResult(context, 403, GoErrorPage());
    80         }
    81     }

    功能權限

      在權限驗證通過後,返回view之前,還是利用了Filter進行一個實時的權限查詢,主要把該用戶所擁有功能權限值查詢出來通過ViewData[“PermCodes”]傳到頁面,然後通過razor進行按鈕的渲染判斷。

      然而我在項目中封裝了大部分常用的LayUI控件,主要利用.Net Core的TagHelper進行了封裝,TagHelper內部與ViewData[“PermCodes”]進行判斷是否輸出HTML。

    全局功能權限值查詢

     1 /// <summary>
     2     /// 全局用戶權限值查詢
     3     /// </summary>
     4     public class GobalPermCodeAttribute : IActionFilter
     5     {
     6         private readonly AdministratorService _administratorService;
     7 
     8         public GobalPermCodeAttribute(AdministratorService administratorService)
     9         {
    10             _administratorService = administratorService;
    11         }
    12 
    13         private static AdministratorData GetCurrentUser(HttpContext context)
    14         {
    15             return context.User.Claims.FirstOrDefault(c => c.Type == ClaimTypes.UserData)?.Value.FromJson<AdministratorData>();
    16         }
    17 
    18 
    19         public void OnActionExecuting(ActionExecutingContext context)
    20         {
    21             ((Controller)context.Controller).ViewData["PermCodes"] = new List<int>();
    22 
    23             if (context.HttpContext.Request.IsAjax())
    24                 return;
    25 
    26             var user = GetCurrentUser(context.HttpContext);
    27             if (user == null)
    28                 return;
    29 
    30             if (user.IsSuper)
    31                 return;
    32 
    33             ((Controller)context.Controller).ViewData["PermCodes"] = _administratorService.GetActionCode(user.UserId).ToList();
    34         }
    35 
    36         public void OnActionExecuted(ActionExecutedContext context)
    37         {
    38         }
    39     }

    LayUI Buttom的TagHelper封裝

     1   [HtmlTargetElement("LayuiButton")]
     2     public class LayuiButtonTag : TagHelper
     3     {
     4         #region 初始化
     5         private const string PermCodeAttributeName = "PermCode";
     6         private const string ClasstAttributeName = "class";
     7         private const string LayEventAttributeName = "lay-event";
     8         private const string LaySubmitAttributeName = "LaySubmit";
     9         private const string LayIdAttributeName = "id";
    10         private const string StyleAttributeName = "style";
    11 
    12         [HtmlAttributeName(StyleAttributeName)]
    13         public string Style { get; set; }
    14 
    15         [HtmlAttributeName(LayIdAttributeName)]
    16         public string Id { get; set; }
    17 
    18         [HtmlAttributeName(LaySubmitAttributeName)]
    19         public string LaySubmit { get; set; }
    20 
    21         [HtmlAttributeName(LayEventAttributeName)]
    22         public string LayEvent { get; set; }
    23 
    24         [HtmlAttributeName(ClasstAttributeName)]
    25         public string Class { get; set; }
    26 
    27         [HtmlAttributeName(PermCodeAttributeName)]
    28         public int PermCode { get; set; }
    29 
    30         [HtmlAttributeNotBound]
    31         [ViewContext]
    32         public ViewContext ViewContext { get; set; }
    33 
    34         #endregion
    35         public override async void Process(TagHelperContext context, TagHelperOutput output)
    36         {
    37             context.ThrowIfNull();
    38             output.ThrowIfNull();
    39 
    40             var administrator = ViewContext.HttpContext.GetCurrentUser();
    41             if (administrator == null)
    42                 return;
    43 
    44             var childContent = await output.GetChildContentAsync();
    45 
    46             if (((List<int>)ViewContext.ViewData["PermCodes"]).Contains(PermCode) || administrator.IsSuper)
    47             {
    48                 foreach (var item in context.AllAttributes)
    49                 {
    50                     output.Attributes.Add(item.Name, item.Value);
    51                 }
    52 
    53                 output.TagName = "a";
    54                 output.TagMode = TagMode.StartTagAndEndTag;
    55                 output.Content.SetHtmlContent(childContent.GetContent());
    56             }
    57             else
    58             {
    59                 output.TagName = "";
    60                 output.TagMode = TagMode.StartTagAndEndTag;
    61                 output.Content.SetHtmlContent("");
    62             }
    63         }
    64     }

     

    視圖代碼

    結尾

      以上就是我本篇分享的內容,項目是以單體應用提供的,方案思路也適用於前後端分離。最後附上幾個系統效果圖

     

     

     

    本站聲明:網站內容來源於博客園,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    ※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

    網頁設計一頭霧水該從何著手呢? 台北網頁設計公司幫您輕鬆架站!

    ※想知道最厲害的網頁設計公司"嚨底家"!

    ※別再煩惱如何寫文案,掌握八大原則!

    ※產品缺大量曝光嗎?你需要的是一流包裝設計!

  • 東南亞最大湖泊洞里薩湖延遲回流 柬埔寨歸咎上游中、寮大壩

    摘錄自2020年7月22日ETtoday新聞雲報導

    位於柬埔寨的洞里薩湖(Tonle Sap)是東南亞最大的湖泊,無論旱、雨季都出產大量魚蝦,周圍地帶有約300萬人以漁業相關產業維生。

    洞里薩湖的水源—湄公河—通常會於雨季水位上漲,並回流至柬埔寨的洞里薩湖,提供豐富的魚類資源。但專家稱,近來連續2年湄公河都延遲回流,嚴重干擾了捕魚活動,影響上百萬人的糧食供應。

    據《路透社》報導,湄公河委員會認為延遲回流現象歸因於2019年降雨減少,以及湄公河上游2座寮國和11座中國水壩的運行,破壞了湄公河的自然水流,回流預計延遲到下個月(8月)才可能發生,導致漁民生計大受影響。

    生物多樣性
    國際新聞
    柬埔寨
    洞里薩湖
    水源供應
    湄公河
    大壩
    截流
    水文

    本站聲明:網站內容來源環境資訊中心https://e-info.org.tw/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    USB CONNECTOR掌控什麼技術要點? 帶您認識其相關發展及效能

    台北網頁設計公司這麼多該如何選擇?

    ※智慧手機時代的來臨,RWD網頁設計為架站首選

    ※評比南投搬家公司費用收費行情懶人包大公開

    ※回頭車貨運收費標準

  • 印尼帝汶島海灘驚現大鯨魚屍體

    摘錄自2020年7月24日法廣印尼報導

    印度尼西亞南部帝汶島沙灘附近,發現了一具上百噸重大鯨魚的屍體。當局目前還沒搞懂這頭大型哺乳動物的死因。

    當地漁民是在20日早上發現這具長達29米的大鯨魚。海事當局認為這頭海洋生物已有70歲,無生命的漂移,可能已長達數年之久。死因則尚無定論。其屍體上沒有發現任何傷痕,但當地媒體提到了上一個的案例。

    2018年,另一頭鯨魚曾擱淺在附近的另個小島Kapota。那頭鯨魚的胃裡發現了眾多塑膠垃圾:115個塑膠杯,還有4個塑膠瓶。這一發現讓人感到震驚,儘管這最終並沒有被確定是造成大鯨魚死亡的原因,但已足以引起科學界的重視。

    生物多樣性
    海洋
    國際新聞
    印尼
    鯨魚
    海洋垃圾

    本站聲明:網站內容來源環境資訊中心https://e-info.org.tw/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    網頁設計公司推薦不同的風格,搶佔消費者視覺第一線

    ※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

    ※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

    南投搬家公司費用需注意的眉眉角角,別等搬了再說!

    ※教你寫出一流的銷售文案?

  • 面試官突然問我MySQL存儲過程,我竟然連基礎都不會!(詳細)

    面試官突然問我MySQL存儲過程,我竟然連基礎都不會!(詳細)

    所有知識體系文章,GitHub已收錄,歡迎Star!再次感謝,願你早日進入大廠!

    GitHub地址: https://github.com/Ziphtracks/JavaLearningmanual

    MySQL存儲過程

    一、存儲過程

    1.1 什麼是存儲過程

    存儲過程(Stored Procedure)是在大型數據庫系統中,一組為了完成特定功能的SQL 語句集,它存儲在數據庫中,一次編譯后永久有效,用戶通過指定存儲過程的名字並給出參數(如果該存儲過程帶有參數)來執行它。存儲過程是數據庫中的一個重要對象。在數據量特別龐大的情況下利用存儲過程能達到倍速的效率提升

    1.2 數據庫存儲過程程序

    當我們了了解存儲過程是什麼之後,就需要了解數據庫中存在的這三種類型的數據庫存儲類型程序,如下:

    • 存儲過程: 存儲過程是最常見的存儲程序,存儲過程是能夠接受輸入和輸出參數並且能夠在請求時被執行的程序單元。
    • 存儲函數: 存儲函數和存儲過程很相像,但是它的執行結果會返回一個值。最重要的是存儲函數可以被用來充當標準的 SQL 語句,允許程序員有效的擴展 SQL 語言的能力。
    • 觸發器: 觸發器是用來響應激活或者觸發數據庫行為事件的存儲程序。通常,觸發器用來作為數據庫操作語言的響應而被調用,觸發器可以被用來作為數據校驗和自動反向格式化。

    注意: 其他的數據庫提供了別的數據存儲程序,包括包和類。目前MySQL不提供這種結構。

    1.3 為什麼要使用存儲程序

    雖然目前的開發中存儲程序我們使用的並不是很多,但是不一定就否認它。其實存儲程序會為我們使用和管理數據庫帶來了很多優勢:

    • 使用存儲程序更加安全。
    • 存儲程序提供了一種數據訪問的抽象機制,它能夠極大的改善你的代碼在底層數據結構演化過程中的易維護性。
    • 存儲程序可以降低網絡擁阻,因為屬於數據庫服務器的內部數據,這相比在網上傳輸數據要快的多。
    • 存儲程序可以替多種使用不同構架的外圍應用實現共享的訪問例程,無論這些構架是基於數據庫服務器外部還是內部。
    • 以數據為中心的邏輯可以被獨立的放置於存儲程序中,這樣可以為程序員帶來更高、更為獨特的數據庫編程體驗。
    • 在某些情況下,使用存儲程序可以改善應用程序的可移植性。(在另外某些情況下,可移植性也會很差!)

    這裏我大致解釋一下上述幾種使用存儲程序的優勢:

    我們要知道在Java語言中,我們使用數據庫與Java代碼結合持久化存儲需要引入JDBC來完成。會想到JDBC,我們是否還能想起SQL注入問題呢?雖然使用PreparedStatement解決SQL注入問題,那就真的是絕對安全嗎?不,它不是絕對安全的。

    這時候分析一下數據庫與Java代碼的連接操作流程。在BS結構中,一般都是瀏覽器訪問服務器的,再由服務器發送SQL語句到數據庫,在數據庫中對SQL語句進行編譯運行,最後把結果通過服務器處理再返回瀏覽器。在此操作過程中,瀏覽器對服務器每發送一次對數據庫操作的請求就會調用對應的SQL語句編譯和執行,這是一件十分浪費性能的事情,性能下降 了就說明對數據庫的操作效率低 了。

    還有一種可能是,在這個過程中進行發送傳輸的SQL語句是對真實的庫表進行操作的SQL語句,如果在發送傳輸的過程中被攔截了,一些不法分子會根據他所攔截的SQL語句推斷出我們數據庫中的庫表結構,這是一個很大的安全隱患

    關於可維護性的提高,這裏模擬一個場景。通常數據庫在公司中是由DBA來管理的,如果管理數據庫多年的DBA辭職了,此時數據庫會被下一任DBA來管理。這裏時候問題來了,數據庫中這麼多的數據和SQL語句顯然對下一任管理者不太友好。就算管理多年的DBA長時間不操作查看數據庫也會忘記點什麼東西。所以,我們在需要引入存儲程序來進行SQL語句的統一編寫和編譯,為維護提供了便利 。(其實我覺得這個例子並不生動合理,但是為了大家能理解,請體諒!)

    講了很多存儲程序的優勢演變過程,其核心就是: 需要將編譯好的一段或多段SQL語句放置在數據庫端的存儲程序中,以便解決以上問題並方便開發者直接調用。

    二、存儲過程的使用步驟

    2.1 存儲過程的開發思想

    存儲過程時數據庫的一個重要的對象,可以封裝SQL語句集,可以用來完成一些較複雜的業務邏輯,並且可以入參(傳參)、出參(返回參數),這裏與Java中封裝方式十分相似。

    而且創建時會預先編譯后保存,開發者後續的調用都不需要再次編譯。

    2.2 存儲過程的優缺點

    存儲過程使用的優缺點其實在1.3中的優勢中說到了。這裏我簡單羅列一下存儲過程的優點與缺點。

    • 優點:
    • 在生產環境下,可以通過直接修改存儲過程的方式修改業務邏輯或bug,而不用重啟服務器。
    • 執行速度快,存儲過程經過編譯之後會比單獨一條一條編譯執行要快很多。
    • 減少網絡傳輸流量。
    • 便於開發者或DBA使用和維護。
    • 在相同數據庫語法的情況下,改善了可移植性。
    • 缺點:
    • 過程化編程,複雜業務處理的維護成本高。
    • 調試不便。
    • 因為不同數據庫語法不一致,不同數據庫之間可移植性差。

    2.3 MySQL存儲過程的官方文檔

    英語好或者有能力的小夥伴可以去參考一下官方文檔。如果不參考官方文檔,沒關係,我在下面也會詳細講述MySQL存儲過程的各個知識點。

    1https://dev.mysql.com/doc/refman/5.6/en/preface.html

    2.3 存儲過程的使用語法

    1create PROCEDURE 過程名( in|out|inout 參數名 數據類型 , ...)
    2begin
    3    sql語句;
    4end;
    5call 過程名(參數值);

    in是定義傳入參數的關鍵字。out是定義出參的關鍵字。inout是定義一個出入參數都可以的參數。如果括號內什麼都不定義,就說明該存儲過程時一個無參的函數。在後面會有詳細的案例分析。

    注意: SQL語句默認的結束符為;,所以在使用以上存儲過程時,會報1064的語法錯誤。我們可以使用DELIMITER關鍵字臨時聲明修改SQL語句的結束符為//,如下:

    1-- 臨時定義結束符為"//"
    2DELIMITER //
    3create PROCEDURE 過程名( in|out 參數名 數據類型 , ...)
    4begin
    5    sql語句;
    6end//
    7-- 將結束符重新定義回結束符為";"
    8DELIMITER ;

    例如: 使用存儲過程來查詢員工的工資(無參)

    注意: 如果在特殊的必要情況下,我們還可以通過delimiter關鍵字將;結束符聲明回來使用,在以下案例中我並沒有這樣將結束符聲明回原來的;,在此請大家注意~

    為什麼我在這裏提供了drop(刪除)呢?

    是因為我們在使用的時候如果需要修改存儲過程中的內容,我們需要先刪除現有的存儲過程后,再creat重新創建。

     1# 聲明結束符為//
    2delimiter //
    3
    4# 創建存儲過程(函數)
    5create procedure se()
    6begin
    7    select salary from employee;
    8end //
    9
    10# 調用函數
    11call se() //
    12
    13# 刪除已存在存儲過程——se()函數
    14drop procedure if exists se //

    三、存儲過程的變量和賦值

    3.1 局部變量

    聲明局部變量語法: declare var_name type [default var_value];

    賦值語法:

    注意: 局部變量的定義,在begin/end塊中有效。

    使用set為參數賦值

     1# set賦值
    2
    3# 聲明結束符為//
    4delimiter //
    5
    6# 創建存儲過程
    7create procedure val_set()
    8begin
    9    # 聲明一個默認值為unknown的val_name局部變量
    10    declare val_name varchar(32) default 'unknown'
    ;
    11    # 為局部變量賦值
    12    set val_name = 'Centi';
    13    # 查詢局部變量
    14    select val_name;
    15end //
    16
    17# 調用函數
    18call val_set() //
    19

    使用into接收參數

     1delimiter //
    2create procedure val_into()
    3begin
    4    # 定義兩個變量存放name和age
    5    declare val_name varchar(32) default 'unknown'
    ;
    6    declare val_age int;
    7    # 查詢表中id為1的name和age並放在定義的兩個變量中
    8    select name,age into val_name,val_age from employee where id = 1;
    9    # 查詢兩個變量
    10    select val_name,val_age;
    11end //
    12
    13call val_into() //
    14

    3.2 用戶變量

    用戶自定義用戶變量,當前會話(連接)有效。與Java中的成員變量相似。

    • 語法: @val_name
    • 注意: 該用戶變量不需要提前聲明,使用即為聲明。
     1delimiter //
    2create procedure val_user()
    3begin
    4    # 為用戶變量賦值
    5    set @val_name = 'Lacy';
    6end //
    7
    8# 調用函數
    9call val_user() //
    10
    11# 查詢該用戶變量
    12select @val_name //

    3.3 會話變量

    會話變量是由系統提供的,只在當前會話(連接)中有效。

    語法: @@session.val_name

    1# 查看所有會話變量
    2show session variables;
    3# 查看指定的會話變量
    4select @@session.val_name;
    5# 修改指定的會話變量
    6set @@session.val_name = 0;

    這裏我獲取了一下所有的會話變量,大概有500條會話變量的記錄。等我們深入學習MySQL后,了解了各個會話變量值的作用,可以根據需求和場景來修改會話變量值。

    1delimiter //
    2create procedure val_session()
    3begin
    4    # 查看會話變量
    5    show session variables
    ;
    6end //
    7
    8call val_session() //
    9

    image-20200610112512964

    3.4 全局變量

    全局變量由系統提供,整個MySQL服務器內有效。

    語法: @@global.val_name

    1# 查看全局變量中變量名有char的記錄
    2show global variables like '%char%' //
    3# 查看全局變量character_set_client的值
    4select @@global.character_set_client //

    3.5 入參出參

    入參出參的語法我們在文章開頭已經提過了,但是沒有演示,在這裏我將演示一下入參出參的使用。

    語法: in|out|inout 參數名 數據類型 , ...

    in定義出參;out定義入參;inout定義出參和入參。

    出參in

    使用出參in時,就是需要我們傳入參數,在這裏可以對參入的參數加以改變。簡單來說in只負責傳入參數到存儲過程中,類似Java中的形參。

     1delimiter //
    2create procedure val_in(in val_name varchar(32))
    3begin
    4    # 使用用戶變量出參(為用戶變量賦參數值)
    5    set @val_name1 = val_name;
    6end //
    7
    8# 調用函數
    9call val_in('DK') //
    10
    11# 查詢該用戶變量
    12select @val_name1 //

    入參out

    在使用out時,需要傳入一個參數。而這個參數相當於是返回值,可以通過調用、接收來獲取這個參數的內容。簡單來說out只負責作返回值。

     1delimiter //
    2# 創建一個入參和出參的存儲過程
    3create procedure val_out(in val_id int,out val_name varchar(32))
    4begin
    5    # 傳入參數val_id查詢員工返回name值(查詢出的name值用出參接收並返回)
    6    select name into val_name from employee where id = val_id;
    7end //
    8
    9# 調用函數傳入參數並聲明傳入一個用戶變量
    10call val_out(1, @n) //
    11
    12# 查詢用戶變量
    13select @n //

    入參出參inout

    inout關鍵字,就是把in和out合併成了一個關鍵字使用。被關鍵字修飾的參數既可以出參也可以入參。

     1delimiter //
    2create procedure val_inout(in val_name varchar(32), inout val_age int)
    3begin
    4    # 聲明一個a變量
    5    declare a int;
    6    # 將傳入的參數賦值給a變量
    7    set a = val_age;
    8    # 通過name查詢age並返回val_age
    9    select age into val_age from employee where name = val_name;
    10    # 將傳入的a與-和查詢age結果字符串做拼接並查詢出來(concat——拼接字符串)
    11    select concat(a, '-', val_age);
    12end //
    13
    14# 聲明一個用戶變量並賦予參數為40
    15set @ages = '40' //
    16# 調用函數並傳入參數值
    17call val_inout('Ziph', @ages) //
    18# 執行結果
    19# 40-18

    四、存儲過程中的流程控制

    4.1 if 條件判斷(推薦)

    擴展: timestampdiff(unit, exp1, exp2)為exp2 – exp1得到的差值,而單位是unit。(常用於日期)

    擴展例子: select timestampdiff(year,’2020-6-6‘,now()) from emp e where id = 1;

    解釋擴展例子: 查詢員工表中id為1員工的年齡,exp2就可以為該員工的出生年月日,並以年為單位計算。

    語法:

    1IF 條件判斷 THEN 結果
    2    [ELSEIF 條件判斷 THEN 結果] ...
    3    [ELSE 結果]
    4END IF

    舉例: 傳入所查詢的id參數查詢工資標準(s<=6000為低工資標準;6000 =15000為高工資標準)

     1delimiter //
    2create procedure s_sql(in val_id int)
    3begin
    4    # 聲明一個局部變量result存放工資標準結果
    5    declare result varchar(32)
    ;
    6    # 聲明一個局部變量存放查詢得到的工資
    7    declare s double;
    8    # 根據入參id查詢工資
    9    select salary into s from employee where id = val_id;
    10    # if判斷的使用
    11    if s <= 6000 then
    12        set result = '低工資標準';
    13    elseif s <= 10000 then
    14        set result = '中工資標準';
    15    elseif s <= 15000 then
    16        set result = '中上工資標準';
    17    else
    18        set result = '高工資標準';
    19    end if;
    20    # 查詢工資標準結果
    21    select result;
    22end //
    23
    24# 調用函數,傳入參數
    25call s_sql(1);

    4.2 case條件判斷

    關於case語句,不僅僅在存儲過程中可以使用,MySQL基礎查詢語句中也有用到過。相當於是Java中的switch語句。

    語法:

     1# 語法一
    2CASE case_value
    3    WHEN when_value THEN 結果
    4    [WHEN when_value THEN 結果] ...
    5    [ELSE 結果]
    6END CASE
    7
    8# 語法二(推薦語法)
    9CASE
    10    WHEN 條件判斷 THEN 結果
    11    [WHEN 條件判斷 THEN 結果] ...
    12    [ELSE 結果]
    13END CASE

    舉例:

     1# 語法一
    2delimiter //
    3create procedure s_case(in val_id int)
    4begin
    5    # 聲明一個局部變量result存放工資標準結果
    6    declare result varchar(32);
    7    # 聲明一個局部變量存放查詢得到的工資
    8    declare s double;
    9    # 根據入參id查詢工資
    10    select salary into s from employee where id = val_id;
    11    case s
    12        when 6000 then set result = '低工資標準';
    13        when 10000 then set result = '中工資標準';
    14        when 15000 then set result = '中上工資標準';
    15        else set result = '高工資標準';
    16    end case;
    17    select result;
    18end //
    19
    20call s_case(1);
    21
    22# 語法二(推薦)
    23delimiter //
    24create procedure s_case(in val_id int)
    25begin
    26    # 聲明一個局部變量result存放工資標準結果
    27    declare result varchar(32);
    28    # 聲明一個局部變量存放查詢得到的工資
    29    declare s double;
    30    # 根據入參id查詢工資
    31    select salary into s from employee where id = val_id;
    32    case
    33        when s <= 6000 then set result = '低工資標準';
    34        when s <= 10000 then set result = '中工資標準';
    35        when s <= 15000 then set result = '中上工資標準';
    36        else set result = '高工資標準';
    37    end case;
    38    select result;
    39end //
    40
    41call s_case(1);

    4.3 loop循環

    loop為死循環,需要手動退出循環,我們可以使用leave來退出循環

    可以把leave看成Java中的break;與之對應的,就有iterate(繼續循環)也可以看成Java的continue

    語法:

    1[別名:] LOOP
    2    循環語句
    3END LOOP [別名]

    注意:別名和別名控制的是同一個標籤。

    示例1: 循環打印1~10(leave控制循環的退出)

    注意:該loop循環為死循環,我們查的1~10数字是i,在死循環中設置了當大於等於10時停止循環,也就是說先後執行了10次該循環內的內容,結果查詢了10次,生成了10個結果(1~10)。

     1delimiter //
    2create procedure s_loop()
    3begin
    4    # 聲明計數器
    5    declare i int default 1;
    6    # 開始循環
    7    num:
    8    loop
    9        # 查詢計數器記錄的值
    10        select i;
    11        # 判斷大於等於停止計數
    12        if i >= 10 then
    13            leave num;
    14        end if;
    15        # 計數器自增1
    16        set i = i + 1;
    17    # 結束循環
    18    end loop num;
    19end //
    20
    21call s_loop();

    打印結果:

    image-20200610191639524

    示例2: 循環打印1~10(iterate和leave控制循環)

    注意:這裏我們使用字符串拼接計數器結果,而條件如果用iterate就必須時 i < 10 了!

     1delimiter //
    2create procedure s_loop1()
    3begin
    4    # 聲明變量i計數器
    5    declare i int default 1
    ;
    6    # 聲明字符串容器
    7    declare str varchar(256) default '1';
    8    # 開始循環
    9    num:
    10    loop
    11        # 計數器自增1
    12        set i = i + 1;
    13        # 字符串容器拼接計數器結果
    14        set str = concat(str, '-', i);
    15        # 計數器i如果小於10就繼續執行
    16        if i < 10 then
    17            iterate num;
    18        end if;
    19        # 計數器i如果大於10就停止循環
    20        leave num;
    21    # 停止循環
    22    end loop num;
    23    # 查詢字符串容器的拼接結果
    24    select str;
    25end //
    26
    27call s_loop1();

    image-20200610193153512

    4.4 repeat循環

    repeat循環類似Java中的do while循環,直到條件不滿足才會結束循環。

    語法:

    1[別名:] REPEAT
    2    循環語句
    3UNTIL 條件
    4END REPEAT [別名]

    示例: 循環打印1~10

     1delimiter //
    2create procedure s_repeat()
    3begin
    4    declare i int default 1;
    5    declare str varchar(256default '1';
    6    # 開始repeat循環
    7    num:
    8    repeat
    9        set i = i + 1;
    10        set str = concat(str'-', i);
    11    # until 結束條件
    12    # end repeat 結束num 結束repeat循環
    13    until i >= 10 end repeat num;
    14    # 查詢字符串拼接結果
    15    select str;
    16end //
    17
    18call s_repeat();

    4.5 while循環

    while循環就與Java中的while循環很相似了。

    語法:

    1[別名] WHILE 條件 DO
    2    循環語句
    3END WHILE [別名]

    示例: 循環打印1~10

     1delimiter //
    2create procedure s_while()
    3begin
    4    declare i int default 1;
    5    declare str varchar(256default '1';
    6    # 開始while循環
    7    num:
    8    # 指定while循環結束條件
    9    while i < 10 do
    10        set i = i + 1;
    11        set str = concat(str'+', i);
    12    # while循環結束
    13    end while num;
    14    # 查詢while循環拼接字符串
    15    select str;
    16end //
    17
    18call s_while();

    4.6 流程控制語句(繼續、結束)

    至於流程控制的繼續和結束,我們在前面已經使用過了。這裏再列舉一下。

    leave:與Java中break;相似

    1leave 標籤;

    iterate:與Java中的continue;相似

    1iterate 標籤;

    五、游標與handler

    5.1 游標

    游標是可以得到某一個結果集並逐行處理數據。游標的逐行操作,導致了游標很少被使用!

    語法:

    1DECLARE 游標名 CURSOR FOR 查詢語句
    2-- 打開語法
    3OPEN 游標名
    4-- 取值語法
    5FETCH 游標名 INTO var_name [, var_name] ...
    6-- 關閉語法
    7CLOSE 游標名

    了解了游標的語法,我們開始使用游標。如下:

    示例: 使用游標查詢id、name和salary。

     1delimiter //
    2create procedure f()
    3begin
    4    declare val_id int;
    5    declare val_name varchar(32);
    6    declare val_salary double;
    7
    8    # 聲明游標
    9    declare emp_flag cursor for
    10    select idname, salary from employee;
    11
    12    # 打開
    13    open emp_flag;
    14
    15    # 取值
    16    fetch emp_flag into val_id, val_name, val_salary;
    17
    18    # 關閉
    19    close emp_flag;
    20
    21    select val_id, val_name, val_salary;
    22end //
    23
    24call f();

    執行結果:

    image-20200610203622749

    因為游標逐行操作的特點,導致我們只能使用游標來查詢一行記錄。怎麼改善代碼才可以實現查詢所有記錄呢?聰明的小夥伴想到了使用循環。對,我們試試使用一下循環。

     1delimiter //
    2create procedure f()
    3begin
    4    declare val_id int;
    5    declare val_name varchar(32);
    6    declare val_salary double;
    7
    8    # 聲明游標
    9    declare emp_flag cursor for
    10    select idname, salary from employee;
    11
    12    # 打開
    13    open emp_flag;
    14
    15    # 使用循環取值
    16    c:loop
    17        # 取值
    18        fetch emp_flag into val_id, val_name, val_salary;
    19    end loop;
    20
    21    # 關閉
    22    close emp_flag;
    23
    24    select val_id, val_name, val_salary;
    25end //
    26
    27call f();

    image-20200610204034224

    我們使用循環之後,發現有一個問題,因為循環是死循環,我們不加結束循環的條件,游標會一直查詢記錄,當查到沒有的記錄的時候,就會拋出異常1329:未獲取到選擇處理的行數

    如果我們想辦法指定結束循環的條件該怎麼做呢?

    這時候可以聲明一個boolean類型的標記。如果為true時則查詢結果集,為false時則結束循環。

     1delimiter //
    2create procedure f()
    3begin
    4    declare val_id int;
    5    declare val_name varchar(32);
    6    declare val_salary double;
    7
    8    # 聲明flag標記
    9    declare flag boolean default true;
    10
    11    # 聲明游標
    12    declare emp_flag cursor for
    13    select idname, salary from employee;
    14
    15    # 打開
    16    open emp_flag;
    17
    18    # 使用循環取值
    19    c:loop
    20        fetch emp_flag into val_id, val_name, val_salary;
    21        # 如果標記為true則查詢結果集
    22        if flag then
    23            select val_id, val_name, val_salary;
    24        # 如果標記為false則證明結果集查詢完畢,停止死循環
    25        else
    26            leave c;
    27        end if;
    28    end loop;
    29
    30    # 關閉
    31    close emp_flag;
    32
    33    select val_id, val_name, val_salary;
    34end //
    35
    36call f();

    上述代碼你會發現並沒有寫完,它留下了一個很嚴肅的問題。當flag = false時候可以結束循環。但是什麼時候才讓flag為false啊?

    於是,MySQL為我們提供了一個handler句柄。它可以幫我們解決此疑惑。

    handler句柄語法: declare continue handler for 異常 set flag = false;

    handler句柄可以用來捕獲異常,也就是說在這個場景中當捕獲到1329:未獲取到選擇處理的行數時,就將flag標記的值改為false。這樣使用handler句柄就解決了結束循環的難題。讓我們來試試吧!

    終極版示例: 解決了多行查詢以及結束循環問題。

     1delimiter //
    2create procedure f()
    3begin
    4    declare val_id int;
    5    declare val_name varchar(32);
    6    declare val_salary double;
    7
    8    # 聲明flag標記
    9    declare flag boolean default true;
    10
    11    # 聲明游標
    12    declare emp_flag cursor for
    13    select idname, salary from employee;
    14
    15    # 使用handler句柄來解決結束循環問題
    16    declare continue handler for 1329 set flag = false;
    17
    18    # 打開
    19    open emp_flag;
    20
    21    # 使用循環取值
    22    c:loop
    23        fetch emp_flag into val_id, val_name, val_salary;
    24        # 如果標記為true則查詢結果集
    25        if flag then
    26            select val_id, val_name, val_salary;
    27        # 如果標記為false則證明結果集查詢完畢,停止死循環
    28        else
    29            leave c;
    30        end if;
    31    end loop;
    32
    33    # 關閉
    34    close emp_flag;
    35
    36    select val_id, val_name, val_salary;
    37end //
    38
    39call f();

    執行結果:

    image-20200610210925964

    在執行結果中,可以看出查詢結果以多次查詢的形式,分佈显示到了每一個查詢結果窗口中。

    注意: 在語法中,變量聲明、游標聲明、handler聲明是必須按照先後順序書寫的,否則創建存儲過程出錯。

    5.2 handler句柄

    語法:

    1DECLARE handler操作 HANDLER
    2    FOR 情況列表...(比如:異常錯誤情況)
    3    操作語句

    注意:異常情況可以寫異常錯誤碼、異常別名或SQLSTATE碼。

    handler操作:

    • CONTINUE: 繼續
    • EXIT: 退出
    • UNDO: 撤銷

    異常情況列表:

    • mysql_error_code
    • SQLSTATE [VALUE] sqlstate_value
    • condition_name
    • SQLWARNING
    • NOT FOUND
    • SQLEXCEPTION

    注意: MySQL中各種異常情況代碼、錯誤碼、別名和SQLSTATEM碼可參考官方文檔:

    https://dev.mysql.com/doc/refman/5.6/en/server-error-reference.html

    寫法示例:

    1    DECLARE exit HANDLER FOR SQLSTATE '3D000' set flag = false;
    2    DECLARE continue HANDLER FOR 1050 set flag = false;
    3    DECLARE continue HANDLER FOR not found set flag = false;

    六、循環創建表

    需求: 創建下個月的每天對應的表,創建的表格式為:comp_2020_06_01、comp_2020_06_02、...

    描述: 我們需要用某個表記錄很多數據,比如記錄某某用戶的搜索、購買行為(注意,此處是假設用數據庫保存),當每天記錄較多時,如果把所有數據都記錄到一張表中太龐大,需要分表,我們的要求是,每天一張表,存當天的統計數據,就要求提前生產這些表——每月月底創建下一個月每天的表!

    預編譯: PREPARE 數據庫對象名 FROM 參數名

    執行: EXECUTE 數據庫對象名 [USING @var_name [, @var_name] ...]

    通過數據庫對象創建或刪除表: {DEALLOCATE | DROP} PREPARE 數據庫對象名

    關於時間處理的語句:

    1-- EXTRACT(unit FROM date)               截取時間的指定位置值
    2-- DATE_ADD(date,INTERVAL expr unit)     日期運算
    3-- LAST_DAY(date)                          獲取日期的最後一天
    4-- YEAR(date)                             返回日期中的年
    5-- MONTH(date)                            返回日期的月
    6-- DAYOFMONTH(date)                        返回日

    代碼:

     1-- 思路:循環構建表名 comp_2020_06_01 到 comp_2020_06_30;並執行create語句。
    2delimiter //
    3create procedure sp_create_table()
    4begin
    5    # 聲明需要拼接表名的下一個月的年、月、日
    6    declare next_year int;
    7    declare next_month int;
    8    declare next_month_day int;
    9
    10    # 聲明下一個月的月和日的字符串
    11    declare next_month_str char(2);
    12    declare next_month_day_str char(2);
    13
    14    # 聲明需要處理每天的表名
    15    declare table_name_str char(10);
    16
    17    # 聲明需要拼接的1
    18    declare t_index int default 1;
    19    # declare create_table_sql varchar(200);
    20
    21    # 獲取下個月的年份
    22    set next_year = year(date_add(now(),INTERVAL 1 month));
    23    # 獲取下個月是幾月 
    24    set next_month = month(date_add(now(),INTERVAL 1 month));
    25    # 下個月最後一天是幾號
    26    set next_month_day = dayofmonth(LAST_DAY(date_add(now(),INTERVAL 1 month)));
    27
    28    # 如果下一個月月份小於10,就在月份的前面拼接一個0
    29    if next_month < 10
    30        then set next_month_str = concat('0',next_month);
    31    else
    32        # 如果月份大於10,不做任何操作
    33        set next_month_str = concat('',next_month);
    34    end if;
    35
    36    # 循環操作(下個月的日大於等於1循環開始循環)
    37    while t_index <= next_month_day do
    38
    39        # 如果t_index小於10就在前面拼接0
    40        if (t_index < 10)
    41            then set next_month_day_str = concat('0',t_index);
    42        else
    43            # 如果t_index大於10不做任何操作
    44            set next_month_day_str = concat('',t_index);
    45        end if;
    46
    47        # 拼接標命字符串
    48        set table_name_str = concat(next_year,'_',next_month_str,'_',next_month_day_str);
    49        # 拼接create sql語句
    50        set @create_table_sql = concat(
    51                    'create table comp_',
    52                    table_name_str,
    53                    '(`grade` INT(11) NULL,`losal` INT(11) NULL,`hisal` INT(11) NULL) COLLATE=\'utf8_general_ci\' ENGINE=InnoDB');
    54        # 預編譯
    55        # 注意:FROM後面不能使用局部變量!
    56        prepare create_table_stmt FROM @create_table_sql;
    57        # 執行
    58        execute create_table_stmt;
    59        # 創建表
    60        DEALLOCATE prepare create_table_stmt;
    61
    62        # t_index自增1
    63        set t_index = t_index + 1;
    64
    65    end while;  
    66end//
    67
    68# 調用函數
    69call sp_create_table()

    七、其他

    7.1 characteristic

    在MySQL存儲過程中,如果沒有显示的定義characteristic,它會隱式的定義一系列特性的默認值來創建存儲過程。

    • LANGUAGE SQL

    • 存儲過程語言,默認是sql,說明存儲過程中使用的是sql語言編寫的,暫時只支持sql,後續可能會支持其他語言

    • NOT DETERMINISTIC

    • 是否確定性的輸入就是確定性的輸出,默認是NOT DETERMINISTIC,只對於同樣的輸入,輸出也是一樣的,當前這個值還沒有使用

    • CONTAINS SQL

    • 提供子程序使用數據的內在信息,這些特徵值目前提供給服務器,並沒有根據這些特徵值來約束過程實際使用數據的情況。有以下選擇:

      • CONTAINS SQL表示子程序不包含讀或者寫數據的語句
      • NO SQL 表示子程序不包含sql
      • READS SQL DATA 表示子程序包含讀數據的語句,但是不包含寫數據的語句
      • MODIFIES SQL DATA 表示子程序包含寫數據的語句。
    • SQL SECURITY DEFINER

    • MySQL存儲過程是通過指定SQL SECURITY子句指定執行存儲過程的實際用戶。所以次值用來指定存儲過程是使用創建者的許可來執行,還是執行者的許可來執行,默認值是DEFINER

      • DEFINER 創建者的身份來調用,對於當前用戶來說:如果執行存儲過程的權限,且創建者有訪問表的權限,當前用戶可以成功執行過程的調用的
      • INVOKER 調用者的身份來執行,對於當前用戶來說:如果執行存儲過程的權限,以當前身份去訪問表,如果當前身份沒有訪問表的權限,即便是有執行過程的權限,仍然是無法成功執行過程的調用的。
    • COMMENT ”

    • 存儲過程的註釋性信息寫在COMMENT裏面,這裏只能是單行文本,多行文本會被移除到回車換行等

    7.2 死循環處理

    如有死循環處理,可以通過下面的命令查看並殺死(結束)

    1show processlist;
    2kill id;

    7.3 select語句中書寫case

    1select 
    2    case
    3        when 條件判斷 then 結果
    4        when 條件判斷 then 結果
    5        else 結果
    6    end 別名,
    7    *
    8from 表名;

    7.4 複製表和數據

    1CREATE TABLE dept SELECT * FROM procedure_demo.dept;
    2CREATE TABLE emp SELECT * FROM procedure_demo.emp;
    3CREATE TABLE salgrade SELECT * FROM procedure_demo.salgrade;

    7.5 臨時表

     1create temporary table 表名(
    2  字段名 類型 [約束],
    3  name varchar(20
    4)Engine=InnoDB default charset utf8;
    5
    6-- 需求:按照部門名稱查詢員工,通過select查看員工的編號、姓名、薪資。(注意,此處僅僅演示游標用法)
    7delimiter $$
    8create procedure sp_create_table02(in dept_name varchar(32))
    9begin
    10    declare emp_no int;
    11    declare emp_name varchar(32);
    12    declare emp_sal decimal(7,2);
    13    declare exit_flag int default 0;
    14
    15    declare emp_cursor cursor for
    16        select e.empno,e.ename,e.sal
    17        from emp e inner join dept d on e.deptno = d.deptno where d.dname = dept_name;
    18
    19    declare continue handler for not found set exit_flag = 1;
    20
    21    -- 創建臨時表收集數據
    22    CREATE temporary TABLE `temp_table_emp` (
    23        `empno` INT(11NOT NULL COMMENT '員工編號',
    24        `ename` VARCHAR(32NULL COMMENT '員工姓名' COLLATE 'utf8_general_ci',
    25        `sal` DECIMAL(7,2NOT NULL DEFAULT '0.00' COMMENT '薪資',
    26        PRIMARY KEY (`empno`USING BTREE
    27    )
    28    COLLATE='utf8_general_ci'
    29    ENGINE=InnoDB;  
    30
    31    open emp_cursor;
    32
    33    c_loop:loop
    34        fetch emp_cursor into emp_no,emp_name,emp_sal;
    35
    36
    37        if exit_flag != 1 then
    38            insert into temp_table_emp values(emp_no,emp_name,emp_sal); 
    39        else
    40            leave c_loop;
    41        end if;
    42
    43    end loop c_loop;
    44
    45    select * from temp_table_emp;
    46
    47    select @sex_res; -- 僅僅是看一下會不會執行到
    48    close emp_cursor;
    49
    50end$$
    51
    52call sp_create_table02('RESEARCH');

    本站聲明:網站內容來源於博客園,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    USB CONNECTOR掌控什麼技術要點? 帶您認識其相關發展及效能

    台北網頁設計公司這麼多該如何選擇?

    ※智慧手機時代的來臨,RWD網頁設計為架站首選

    ※評比南投搬家公司費用收費行情懶人包大公開

    ※回頭車貨運收費標準

  • 當年考車沒掛科?看完包你再也不敢裝老司機了…

    當年考車沒掛科?看完包你再也不敢裝老司機了…

    學員在上課日本駕校的模擬駕駛訓練你以為學了這麼久,總算能拿證了。呵呵,天真。上面說的都只是駕校的必修課程,上完前面說的那些課之後,你會拿到一張類似准考證的東西。你需要做的是,拿着這張證,去參加正式的考試,萬一掛科了,就要回去駕校重。

    每天當打開電腦,準備一覽天下,指點江山的時候,經常都會看到各類交通事故的彈窗新聞。點進去之後往評論區一看,這幾種回答絕對跑不掉:

    “國產車鐵皮薄,就是不耐撞!”

    “肯定是女司機引發的吧,馬路殺手啊……”

    “戳我,海量資源,看妹妹自拍羞羞視頻妹妹&¥#…#%¥……”

    其實不管是男是女,是國產車還是進口車,大多數情況下,發生交通事故的原因都是駕駛操作本身出現了問題。花了那麼多錢和時間,被教練“親切問候”了那麼多次,到頭來拿了駕照卻還是不會開車?難道是國內的駕考規則還不夠嚴?

    要弄清楚這個問題,那就得深入探究一下,各國的駕考規則到底有什麼不同。所以今天就讓大家看一下,世界各國的人們到底是怎麼考駕照的。

    最全面

    德 國

    德國人一向以嚴謹著稱,考駕照也不例外。在他們看來,開車不僅僅是一個人的事情,還關乎他人的生命,所以安全是絕對是被放在第一位的。

    在德國,考駕照之前要先學8個小時的急救知識,止血、心肺復蘇等全部都要練得滾瓜爛熟。然後才能開始上車練,每次最少練1個半小時,一共練12次,其中包括4次高速公路練習和3次夜間道路練習。

    在正式考試的時候沒有固定線路,要聽考官指揮怎麼走,基本上從繁忙道路到高速公路等各種路況都得走一遍。考核標準也比較人性化,比如考試過程中遇到突發障礙物,不管是繞行,剎車,還是停車後下車查看,這都是允許的,因為安全最重要。不會像國內的电子駕考,半路突然跑出個熊孩子,你一腳急剎車,反而算你掛科了。

    值得一提的是,除了安全只是和駕駛技術,德國駕考還會把汽車知識也一起考了。考官會打開發動機蓋,問你這個零件是幹嘛的,那個零件壞了怎麼辦。注重安全之餘,也可見德國人對汽車真的是愛到骨子里。

    缺點是學車價格比較貴,均價在1500歐左右,摺合人民幣10898元。

    最麻煩

    日 本

    在日本學車也不是一件容易事,首先學費就很貴,一般費用在15000到20000人民幣之間。

    重點是在日本學車,上課的壓力不比高考黨低。先要學交通安全知識10個小時,然後學駕駛理論15個小時,再然後是上車實操練習19個小時,最後還要16個小時的專業進修。

    學員在上課

    日本駕校的模擬駕駛訓練

    你以為學了這麼久,總算能拿證了?呵呵,天真!上面說的都只是駕校的必修課程,上完前面說的那些課之後,你會拿到一張類似准考證的東西。你需要做的是,拿着這張證,去參加正式的考試,萬一掛科了,就要回去駕校重!新!學!

    所以為了照顧那些需要快速拿證的學員,一些駕校推出了包含住宿在內駕考套餐,吃喝拉撒都在駕校里,最快兩個星期就能拿證。

    最嚴格

    芬 蘭

    芬蘭出名的除了芬蘭浴和諾基亞之外,芬蘭人會開車也是另外一塊大招牌。芬蘭素來盛產冠軍車手,像哈基寧,萊科寧這兩位F1冠軍,都是芬蘭人。

    哈基寧(左)和萊科寧(右)

    用一句話概括芬蘭人的車技就是:沒人不會漂移。因為芬蘭地處北歐,常年積雪,車子行駛的時候附着力很差,所以對駕駛者的技術要求很高,偶爾需要用上漂移來過彎真的一點都沒有誇張的成分。

    因此芬蘭的駕考標準相當嚴格,而且難度奇高。比如有一項考試就是讓學員在濕滑的地面上快速行駛,然後急剎車,看學員能不能在保持車身穩定的同時把車子停下來,稍有偏差就是掛科。由於芬蘭森林覆蓋率很高,路上隔三差五就有動物跳出來,所以考試里還有一項是用電腦模擬野生動物突然出現,用來測試學員的緊急反應能力。

    有趣的是,芬蘭的交通罰款是按照駕駛者的收入來決定罰款高低的,跟收稅一樣。同樣的違規現象,越有錢的罰款越高。曾經有一次就是諾基亞的副總裁在限速50km/h的路上開到了75km/h,結果被罰了11萬6千歐元,摺合人民幣約84萬……

    最簡單

    韓 國

    韓國駕考容易通過相信不少國內的駕考學員都知道,韓國的駕考是不會硬性要求學員去駕校學習的。先是進行筆試,內容非常簡單,60分就可以過,而且支持中文作答。

    實操考試方面,只要你覺得你會開車,就可以去考了。考試內容也十分簡單,基本就是停停車,開開燈,然後在固定線路跑一遍就行了。如果你是有一點駕駛基礎的,提前在考試路段練幾遍基本都能過。因為可以選擇不上駕校,所以費用相對來說非常低,光是考駕照的花費來說的話,一千來塊人民幣就能搞定了。

    因此之前不少國人由於擔心國內駕照考不過,或者嫌等考試的時間太長,都會報個十天團去韓國專門考駕照。可是這條門路現在已經基本被封起來了,因為從今年4月1日起,公安部就出台新規定,中國內地居民取得境外駕駛證時,在核發國家或地區連續居留不足三個月的,換領國內駕駛證時,需要參加全部考試科目。

    總結

    考駕照的目的就是為了讓自己成為一名合格的駕駛員,無論是對車內乘客,對路人,還是對自己,都是最起碼的責任。每個國家的國情和路況都有所不同,所以不能一概而論哪國的駕考標準更好。根據我國道路情況,認為現有的駕考規則還是比較能考驗出學員的駕駛水平的。建議大家都按照規矩來好好學車,別老想着走後門,畢竟考試掛總好過路上“掛”。本站聲明:網站內容來源於http://www.auto6s.com/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    USB CONNECTOR掌控什麼技術要點? 帶您認識其相關發展及效能

    台北網頁設計公司這麼多該如何選擇?

    ※智慧手機時代的來臨,RWD網頁設計為架站首選

    ※評比南投搬家公司費用收費行情懶人包大公開

    ※回頭車貨運收費標準

  • 購車新生代都在關注啥車型?別忘了奔騰的高顏值SUV

    購車新生代都在關注啥車型?別忘了奔騰的高顏值SUV

    只有帥氣外觀還不夠,撩妹還要看內功。打開車門,新款奔騰X80的內飾設計作出了更多優化。8英寸的懸浮式多媒體觸摸屏、簡化實體按鍵的中控面板,清晰易讀帶彩色显示屏的儀錶盤,都是新款奔騰X80提升乘客滿意度的一大法寶。

    都說現在是90后小鮮肉的時代,踏入社會後工作穩定時,就應買台顏值出眾的SUV來好好的撩妹。想着每天開着高顏值的SUV上班,在路旁的年輕MM的注視下,展示你最帥的一面……別YY了,趕緊找來一款高顏值配置實用的SUV,這車帥得分分鐘讓你按耐不住馬上買買買!

    顏值出眾的SUV,粉絲們也許首先會想到某些百萬豪車。但90后購入的第一款車,太貴的車咱先不考慮。其實在我們身邊的自主品牌,也有不少讓人眼前一亮的佳作。先賣個關子,曾有一款SUV,憑着車身C柱上的獨特“X”字形優美線條,以及紮實底盤,獲得一致好評。

    現在,TA帶着更加年輕運動化的外觀,以及更多實用配置升級,回!來!了!

    一汽新款奔騰X80

    實力提升的一汽新款奔騰X80,將為兵家必爭之地的10~15萬自主緊湊型SUV車市,再添上一把火。值得一提的是,新款奔騰X80的外觀年輕運動化改造,吸睛能力簡直是棒棒噠,顏值控們可不要錯過了。

    為照顧外貌協會,先從新款奔騰X80的重頭戲——外觀改造說起。重新設計的大燈線條,配合更具稜角的中網,讓車頭顯得更加精神及具有運動感。車身的最大特色,車側和車尾交界處標誌性的“X”字形優美設計得以保留。

    來到車尾,熏黑尾燈組及雙邊單出橢圓排氣管,呼應前臉的運動化設計,視覺體驗更加年輕帥氣。此時也化身迷妹,實在是太帥啦!

    新款奔騰X80在尺寸方面,與同級對手對比不落下風,乘坐舒適有保證。擁有一款外觀運動高顏值SUV,新款奔騰X80讓你撩妹實力大增!

    只有帥氣外觀還不夠,撩妹還要看內功。打開車門,新款奔騰X80的內飾設計作出了更多優化。8英寸的懸浮式多媒體觸摸屏、簡化實體按鍵的中控面板,清晰易讀帶彩色显示屏的儀錶盤,都是新款奔騰X80提升乘客滿意度的一大法寶。

    最值得一提是,吸睛能力MAX的8英寸懸浮式多媒體觸摸屏,還支持小鮮肉們喜聞樂見的Apple Carplay及百度Carlife,另外全景影像,一鍵啟動,胎壓監測,ESp車身穩定系統,前後泊車雷達及六安全氣囊等實用配置統統沒有缺席。新款奔騰X80真的有料!

    手機與汽車無縫連接真方便,願意給一百個贊。YY一下,開着新款奔騰X80,用手機連接Carplay,放着喜歡的音樂,與妹子愉快的自駕游去,畫面太美了!

    高顏值,配置實用,那就更少不了大空間。新款奔騰X80的空間表現,即使185cm的長腿歐巴坐進車內,也不會覺得局促。前排座椅針對大腿和肩部的包裹進行了優化,後排的座墊足夠長,乘坐真舒適。不像部分車型,為了大空間把後排改成了“小板凳”。這樣的後排,估計妹子上車,兩分鐘就投訴了!而在奔騰新款X80的大空間幫助下,完美駕馭舒適座椅,這是適合長腿歐巴乘坐的後排啊。

    別忘了新款奔騰X80的動力部分,保持1.8T/2.0L的動力,與發動機匹配的是愛信的6擋手自一體變速箱,2.0L部分車款還搭配了6擋手動變速箱。對於一款緊湊型SUV,2.0L的動力輸出能比競品的小排量渦輪發動機,表現更加從容;1.8T則能滿足性能控對強勁動力的需求。兩款動力組合,相信能滿足小鮮肉們對動力的任性需求。

    總結:

    一汽新款奔騰X80的硬實力提升明顯,在高顏值、大空間及配置升級的針對性的升級后,能更好的滿足90后小鮮肉消費者的購車需求。2016年也拼搏一年了,買輛新款奔騰X80獎勵自己,好好撩妹再適合不過。年底打算要買車的粉絲,要留意了,2016年12月31日前購車,更享5000元購置稅補貼。本站聲明:網站內容來源於http://www.auto6s.com/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    網頁設計公司推薦不同的風格,搶佔消費者視覺第一線

    ※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

    ※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

    南投搬家公司費用需注意的眉眉角角,別等搬了再說!

    ※教你寫出一流的銷售文案?