標籤: 網頁設計公司

  • 菲律賓巴丹群島群島 規模5以上地震連三起

    菲律賓巴丹群島群島 規模5以上地震連三起

    摘錄自2019年7月27日新唐人亞太台、自由時報報導

    27日清晨,菲律賓最北邊的巴丹群島接連發生三起規模五以上的地震,根據美國地質調查所數據,三起地震的規模分別為5.4、5.9和5.7,震源深度僅9.1公里,屬於極淺層地震;菲律賓地震局表示第二次地震規模為6.4。當地許多建築物倒塌,現全島因電力線的損壞處於停電狀態,目前估計已造成5人死亡,9人輕重傷。

    當局沒有發布海嘯警報。目前已知至少8人死亡,60人受傷。事發之後,民眾住家內的物品散亂一地。還有居民從瓦礫堆當中搜救生還者。

    巴丹群島位於台灣和呂宋島之間,菲律賓當局也派遣醫療救難隊前往巴丹群島。



    菲律賓最北方巴丹群(Batanes)今日清晨發生連續兩次規模5以上的地震。(推特 )

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

    【其他文章推薦】

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

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

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

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

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

  • 日本26年首見豬瘟案例 疑遭進口豬肉感染

    摘錄自2018年9月9日蘋果日報日本報導

    日本農業部門今(9)日證實,當地出現26年來首起豬瘟感染病例,懷疑是進口豬肉或野豬肉帶入病毒導致感染。

    發現疫情的是一間位於岐阜市的養豬場,3日到8日發現有80頭豬死亡。檢驗後證實,這些豬隻感染豬瘟,此病毒與最近在中國爆發的非洲豬瘟並不相同,也不會傳染給人類。

    根據日本家畜傳染病預防法規定,只要養豬場發現有豬隻感染豬瘟,就必須撲殺養豬場內所有豬隻。岐阜縣政府今日上午開始撲殺養豬場內的610頭豬隻,作業將持續到明天上午6時。由於豬瘟在豬隻間具傳染性,岐阜縣為預防豬瘟疫情擴大,將這間養豬場半徑10公里內劃為「禁止搬出區域」;並禁止這個區域內的其他3間養豬場出貨。

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

    【其他文章推薦】

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

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

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

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

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

  • 山葉 YAMAHA 與 Gogoro 㩗手合作!將用 Gogoro 電池交換結構在台推新車型

    山葉 YAMAHA 與 Gogoro 㩗手合作!將用 Gogoro 電池交換結構在台推新車型

    台灣知名電動機車品牌 Gogoro 將與日本著名機車品牌 YAMAHA 共同合作研發機車,此共同合作的車種將於 2019 年夏天發售,由台灣山葉機車發售,新車型將採用 Gogoro 的充換電結構與網路,Gogoro 這次與老牌機車廠合作無疑給其換電系統打了劑強心針,甚至有可能藉由 YAMAHA 之助在世界各地推動其系統,這個合作勢必也將對光陽(Kymco)與尚未選擇何種充換電系統的三陽(SYM)帶來更多壓力。

    根據兩公司發出的資料顯示,該合作計畫的內容是有關電動機車的產品開發、委託製造以及電池交換系統的使用,雙方預計於今年內簽署正式合約。屆時將以 Gogoro 市面銷售的車款為基礎,進行 YAMAHA 品牌的電動機車設計,並委由 Gogoro 生產。所完成的車輛則交由台灣山葉機車的銷售通路進行銷售,預計 2019 年夏季將推出第一個車款。

    同時,兩家公司的事業合作夥伴「住友商事株式會社」,在促成雙方合作上也扮演了重要的角色。

    YAMAHA 自 1966 年起投入台灣機車市場,目前以台灣山葉機車所生產的機車為主,年間販賣 29 萬台(2017 年實績)。同時也製造、銷售電動機車 E-VINO,並輸出到日本。以開發機能來說,YAMAHA 在台灣設有 YAMAHA Motor R&D Taiwan,主要是負責台灣市場的速克達機車開發。此次和 Gogoro 的合作案,不只是想要在台灣市場擴充其包含燃油車種在內的產品種類,在電動車的部分,也想透過運用 Gogoro 所擁有的電池交換站網路,提高消費者使用上的便利性。

    Gogoro 自 2015 年起投入台灣機車市場,以製造自有品牌的智慧雙輪和可方便交換電池的電池交換站,開展全新的電動機車商業模式。Gogoro 能源網路目前在台灣已設置超過 750 個電池交換站,預計 2019 年初將超過 1,000 站。並且,自產品上市以來,已達到 1,700 萬次以上的電池交換次數,透過此次的合作,Gogoro 可望擴大電動機車的產能。

    YAMAHA 發動機株式會社 MC 事業本部長木下拓也表示:「此次和台灣 Gogoro 的合作,不只增加顧客對移動工具的選擇,同時透過共用先進的電池交換系統,將對新的移動工具服務和創造市場帶來挑戰。」

    Gogoro 創辦人暨執行長陸學森表示:「Gogoro 為推動能源開放平台的創新品牌,並運用能源網路的建置來推進大型智慧城市的轉型。這次我們非常榮幸能與 YAMAHA 合作,向實現能源願景的目標邁出重要的一步。」

    (合作媒體:。首圖來源:)

    本站聲明:網站內容來源於EnergyTrend https://www.energytrend.com.tw/ev/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    ※帶您來了解什麼是 USB CONNECTOR  ?

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

    ※如何讓商品強力曝光呢? 網頁設計公司幫您建置最吸引人的網站,提高曝光率!

    ※綠能、環保無空污,成為電動車最新代名詞,目前市場使用率逐漸普及化

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

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

  • Gogoro 宣布與新北市政府合作擴充電池交換站,擴充大台北充電網路

    Gogoro 宣布與新北市政府合作擴充電池交換站,擴充大台北充電網路

    Gogoro 致力於電池交換型式的供電系統,提供比傳統機車更為綠能的選擇。13 日 Gogoro 宣布與新北市政府合作建置電池交換站,加上上個月台北市年底前建置的千座電池交換站,大台北地區將逐漸追上加油站數量。

    Gogoro 在新北市於明年第一季前預計完成建置 20 座電池交換站、兩座太陽能電池交換站,建置範圍涵蓋三重區、三峽區、中和區、汐止區、板橋區、林口區、新店區、新莊區、樹林區、蘆洲區等新北都會區域。

    Gogoro 營運副總潘璟倫表示:「目前雙北市平均每月電池交換次數高達近 28 萬次,為全台灣使用率最高的地區。為了加速提供車主的能源需求,Gogoro 不斷積極的佈建電池交換站以及尋找適合的地點,這次非常感謝新北市政府給予機會讓我們建置足夠的電池換電站提供消費者使用,希望透過這樣的合作持續帶動電動機車的發展。」

    Gogoro 在台北市電池交換站已經超越傳統加油站的數量,而與台北市政府合作的 1,000 座電池交換站將拉大差距。本月在大台北地區新開台北天母忠誠、新北新店民權兩家門市。大台北地區將共有 35 家銷售服務據點同時提供完整的服務,完善台北地區服務網路。

    配合開學季,Gogoro 也於 9/15 – 10/14 推試騎送限量「All Pass 文具組」,只要消費者到門市體驗試乘,即可獲得此試騎好禮。

    (合作媒體:。首圖來源:)

    本站聲明:網站內容來源於EnergyTrend https://www.energytrend.com.tw/ev/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

    ※為什麼 USB CONNECTOR 是電子產業重要的元件?

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

    ※台北網頁設計公司全省服務真心推薦

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

    ※推薦評價好的iphone維修中心

  • 台達加入國際電動車倡議 EV100,推動全球低碳交通

    台達加入國際電動車倡議 EV100,推動全球低碳交通

    電源管理及能源基礎解決方案廠商台達 26 日宣布響應國際電動車倡議 EV100,成為台灣第一家及全球第一家電動車能源基礎設施提供者的會員。台達承諾於 2030 年前,將在主要營運據點廣設電動車充電設施,提供員工及客戶電動車使用誘因,協助推動低碳交通轉型以對抗氣候變遷。

    EV100 為氣候組織(The Climate Group)2017 年發起的全球倡議,目的為透過全球具有影響力的企業及政府組織,加速交通運輸之低碳轉型,以呼應全球升溫低於 2℃ 的聯合國目標。目前已有 HP、聯合利華、Ikea、百度等 20 家國際企業響應。

    台達發言人暨企業永續發展周志宏資深協理表示,氣候變遷日益急劇,降低交通設施帶來的溫室氣體排放為另一項重要課題。電動車為低碳城市的解決方案之一,近年來受到外界許多關注,台達以「環保 節能 愛地球」為企業使命,致力為永續城市提供解決方案。運用核心能力持續研發高效能電源及整合式系統,為城市的智慧能源設施布局,並且成為全球領先的電動車製造商提供關鍵車用零組件。在企業內部,我們希望透過友善及便利的電動車充電樁與服務,提供員工及客戶使用電動車的誘因,減少環境負荷。台達至今已於全球總部、營運據點,以及全球生產據點,設置超過 40 支電動車充電樁,支援不同規格的充電需求,並已在廠區提供電動巴士作為交通車,預計將全面使用電動巴士,以降低員工交通的碳排。

    因應全球電動車與充電網絡蓬勃發展的趨勢,台達整合內部 40 年以上的電源轉換與管理的技術能力,提供業界先進的電動車充電解決方案。日前更榮獲美國能源部研發專案經費,合作開發充電輸出功率可達 400kW 的高速電動車充電機(Extreme Fast Charger,XFC),預期不到 10 分鐘的快速充電即可為未來電動車款提供 180 英里的行駛里程(約 288 公里)。除了電動車充電技術的提升,此研究專案的數據和成果將能幫助汽車製造商、相關技術提供者、城市政府、與電力公司更加了解電動車高速充電如何影響電力需量反應規劃,以及充電站如何整合可再生能源,以避免大量的高速充電對電網基礎設施造成壓力。

    (資訊、圖片來源:)

    本站聲明:網站內容來源於EnergyTrend https://www.energytrend.com.tw/ev/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

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

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

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

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

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

    ※超省錢租車方案

  • 川普關稅威脅奏效,SK Innovation 擬赴美設電動車電池廠

    川普關稅威脅奏效,SK Innovation 擬赴美設電動車電池廠

    南韓能源業者 SK Innovation 週一宣布,為鞏固美國客戶的訂單,正在考慮赴美設置電動車電池廠。美國為全球最大電動車市場之一,電池需求可期。

    美國總統川普五月根據 232 條款,以國安之名就進口汽車與零件啟動調查,雖然還未決定是否開徵 25% 的懲罰性關稅,但 SK Innovation 顯然已未雨綢繆,提早思考因應對策與未來投資佈局。

    路透社報導,SK Innovation 發言人表示,設廠位置目前鎖定在美國南部地區,已有二至三個州列入考慮,但確切時間與其它設廠細節則還未敲定。

    SK Innovation 競爭對手 LG Chem 目前已在美國設有生產據點,通用是主要客戶。

    根據國際能源總署(International Energy Agency)五月底發佈報告顯示,2017 年全球電動車加油電混合動力車突破三百萬輛、年增 54%,中國為驅動成長主力,美國與北歐國家也快速成長中。(businessgreen.com)

    (本文內容由 授權使用。首圖來源:)

    本站聲明:網站內容來源於EnergyTrend https://www.energytrend.com.tw/ev/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

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

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

    ※Google地圖已可更新顯示潭子電動車充電站設置地點!!

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

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

    網頁設計最專業,超強功能平台可客製化

  • 抗暖化不遺餘力,丹麥 2030 年將禁售汽柴油車

    抗暖化不遺餘力,丹麥 2030 年將禁售汽柴油車

    丹麥首相拉斯穆森(Lars Løkke Rasmussen)日前向國會提案,計劃在 2030 年禁售汽柴油車,並同時達成 100 萬輛電動車與油電混合車販售目標,盼可藉此響應巴黎協議,進一步展現緩解氣候變遷的決心。

    拉斯穆森指出,「這是一個具有挑戰性的目標,但這正是我們需要嘗試的原因。」,丹麥將在短短 12 年內禁止銷售新汽柴油車,而在 17 年內新販售車輛也都得是電動車與其他零排放車款。顯示混合動力車也有可能在 2035 年逐步停止販售。

    丹麥可說是全球能源轉型先鋒,將於2020年把碳排放量降到1990年的66%,更設定在 2030 年將再生能源發電佔比提升至 55%,2050 年達成發電、交通、供暖製冷全再生能源供應。為了達路上零排放,丹麥也在電動車充電站、重工業氫氣與天然氣設備撥出 7,000 萬丹麥克朗(約 3.3 億新台幣)補助。

    不過該提案尚未獲得國會批准,詳細內容則會在下週公布,若要讓國會通過新法案,丹麥首先得面臨是否要調整電動車補貼政策等難題。

    為解決空氣污染與提升電動車購買率,各國通常會以補貼或是減稅等措施提供購買誘因,像是丹麥原先透過免除最高達 180% 進口稅來刺激電動車買氣,只不過由於受到傳統車廠抗議,丹麥於 2016 年調整補貼政策,此舉導致該國電動車購買量雪崩式下滑。

    以電動車銷售為例,2015 年有高達 4,762 輛電動車登記銷售,但政策公布後,該數據在短短幾年內下降到 1,428,最後在跌至 1,000 以下、僅售出 913 輛,丹麥 2017年僅賣出 500 輛電動車,對此拉斯穆森在 2018 年 5 月也表示,由於電動車銷售量急遽下滑,政府已考慮是否調整電動車補貼政策,也不排除提高獎勵金額。

    在節能減碳與環保世界趨勢下,目前各國紛紛設立禁售汽柴油車時間表,其中由挪威率先吹響號角,於 2016 年設定 2025 年禁售汽柴油車,之後德國、愛爾蘭、印度與荷蘭等國也計劃在 2030 年達標,法國、英國、台灣則預計在 2040 年達成路上零排放,希望可藉由禁售燃油車規範達成改善空氣污染、減緩全球暖化等目的。

    彭博能源財經(BNEF)也預估,2020~2030 年電動車價格將可與傳統汽車相當,2030 年全球電動車銷售量有望突破 3,000 萬輛。

    (首圖來源: CC BY-SA 2.0。文/DaisyChuang)

    本站聲明:網站內容來源於EnergyTrend https://www.energytrend.com.tw/ev/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

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

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

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

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

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

  • 光陽 Ionex 車能網商業版正式登台,柯勝峯:下一步進軍印度市場

    光陽 Ionex 車能網商業版正式登台,柯勝峯:下一步進軍印度市場

    光陽工業(KYMCO)12 日正式在台發表 Ionex 車能網商業版,提供 4 大套裝方案,切入全球商用電動機車領域;董事長柯勝峯更宣布進軍印度這個全球最大二輪車市場,並且全面拓展中國市場。

    隨著環保意識不斷提升,電動化成為現代交通工具最重要的轉變,有效落實城市永續經營的目標。其中,商用電動車扮演極為關鍵的角色,據統計全球約有 1.5 億輛機車,其中約 500 萬輛機車為商業用途,商用機車佔比雖低,但每日騎乘距離是一般機車的 7 倍,維修換車頻率則是一般機車的 2 倍,能源消耗佔比較整體機車多 2 成。因此採行電動車,能夠真正協助企業進行營運模式的發展與轉型。

    Ionex 車能網商業版 4 套裝方案公開

    Ionex 車能網商業版可針對任何企業或政府機構的特定需求,進行量身訂製並快速導入;主要向客戶提供「基礎設施」、「商用車隊」、「共享車輛」、「大眾運輸」4 個套裝方案,內容含括電動機車、抽取式電池、能源交換站、作業系統、管理軟體、手機 App 以及其他客製化服務等等,視需求做出調整,以展現 Ionex 車能網的系統彈性與發展能力:

    • 基礎設施套裝方案:提供欲建置能源基礎建設以對特定市場提供充電服務的企業或政府所設計。
    • 商用車隊套裝方案:為需要商用電動車隊的企業所設計,尤其適合具有配送服務的物流或零售等業者。
    • 共享車輛套裝方案:針對共享業者所設計,加速共享業者將電動車導入市場的時間。
    • 大眾運輸套裝方案:有效協助政府將共享電動車租賃服務導入大眾運輸網路,提升大眾運輸工具的使用率。

    今日的發表會上更展示 2 款商用電動機車、以及 1 款電動輔助自行車。

    進軍印度電動機車市場

    目前在企業客戶方面,光陽已有大型零售的盒馬鮮生、車輛共享的騎電科技、能源服務的張飛充電;而在發展智慧城市,光陽與中國的江蘇常州市、福建寧德市、浙江衢州市、廣東深圳市合作。光陽更將在下週於印度首都新德里,正式宣布進軍印度市場,與當地的合作夥伴加速拓展 Ionex 車能網。

    (合作媒體:。圖片來源:)

    本站聲明:網站內容來源於EnergyTrend https://www.energytrend.com.tw/ev/,如有侵權,請聯繫我們,我們將及時處理

    【其他文章推薦】

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

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

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

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

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

  • 分佈式事務

    分佈式事務

    分佈式事務

    分佈式環境下的事務

    要了解分佈式事務,首先要了解分佈式環境

    分佈式

    如一網站,訪問一個服務A(查詢自己用戶信息), 提供服務A的服務器分別有A1(上海)A2(廣州) A3(新加坡)

    同一個服務分佈在三個區域的服務器上,這就是分佈式。你可以訪問 上海的服務器,廣州的或者新加坡的,但是

    三個服務器之前通信是有延遲的,所以數據同步需要一定時間

    分佈式中的問題

    舉例一個分佈式場景

    如果用戶Y個人信息 名字為 “南柯一夢” Y改為 “南柯夢”, 同一時間,Y用戶好友查看Y的名字,好友查詢的結果是”南柯一夢” 還是 “南柯夢” 這是分佈式系統常見的問題(數據修改發生在上海,訪問發生在新加坡)。

    CAP原則

    CAP原則又稱CAP定理,指的是在一個分佈式系統中,一致性(Consistency)、可用性(Availability)、分區容錯性(Partition tolerance)。CAP 原則指的是,這三個要素最多只能同時實現兩點,不可能三者兼顧。

    參考文章
    An Illustrated Proof of the CAP Theorem
    CAP 定理的含義

    以為網絡有延遲和不可測故障,因此分佈式系統是保持服務穩定的常用手段,但是分佈式因為服務機器分佈在不同地點,因此也會有分佈式的特點問題。

    分佈式中 一致性 可用性 容災性 是三個指標

    Consistency

    數據一致性,分佈式環境下,不同地點的服務器,數據庫數據同步一致。

    Availability

    服務可用性,分佈式環境下,調用服務,都可用

    Partition tolerance

    分區容錯性,容災能力

    分佈式環境多台服務器運行,其中一部分機器故障了,整個系統仍然可以正常運行提供服務

    CAP不能同時滿足

    必須滿足P

    首先分佈式環境,系統需要穩定運行,一台服務器意外斷電,不應該影響系統整體功能正常,另一台或多台服務器還能穩定提供服務,所以分區容錯是必須要滿足。

    滿足C

    數據一致性,所指的是同一個服務所在不同服務器的數據是同步的。如上改名字的場景 南柯一夢 改為 南柯夢 (在上海的數據庫被修改) 那麼系統要做到滿足數據一致性,必須馬上同步廣州和新加坡的數據庫,這樣才能滿足廣州或者新加坡的訪問者獲得的結果也一致是 “南柯夢” 而不是”南柯一夢”

    滿足A

    服務可用性,指任何時候訪問服務,都返回結果

    A與C是衝突的,上海服務器南柯一夢改為南柯夢后,為了服務可用,此時間訪問新加坡和廣州的服務器,返回的結果應該是南柯一夢(任何時候服務都返回結果) 但是嚴格上講,數據是錯誤的,因為用戶已經改了名字,改為南柯夢,但是數據在上海的才是正確的。

    滿足數據一致性必須犧牲服務可用性 或者相反

    要達到數據一致性的要求,必須在上海服務器修改數據的同時,同步廣州和新加坡的數據庫,並且在數據同步完成之前,訪問廣州和新加坡的數據庫中這條數據需要等待,返回同步后的結果(一致性)。

    失去了服務可用性(這裏服務是等待數據同步完成才返回結果,而不是立刻返回)

    因此CAP 要麼 滿足AP (分區服務可用)要麼 CP (分區數據一致)

    分佈式中事務

    商品購買中的事務

    以商品購買生成訂單為例子

    網絡上用戶A 購買 一雙鞋子 價格50 付款後生成消費訂單

    事務中包含子的服務

    這裏簡單設為三個服務,他們是事務相關的

    1.商品信息服務

    提供商品信息等服務

    鞋子 顏色 價格 庫存數量等信息 這裏設 價格price為 50 庫存數 num 9

    2.商家賬號收款服務

    提供金額收入信息等服務

    用戶購買鞋子,需要付款50元到商家賬號

    3.用戶消費訂單服務

    提供購買消費憑證信息等服務

    首先分析用戶購買鞋子,三個服務分別要做什麼

    @1 鞋子庫存減1

    @2 商家賬號金額增加50

    @3 生成 用戶購買鞋子的訂單記錄, 包括數量金額等信息

    事務特性

    原子性

    @1 @2 @3 要麼同時發生,要麼都不發生

    一致性

    鞋子庫存減少1,收入增加50

    隔離性

    鞋子庫存減1,後續用戶最多只能購買(9-1=8)雙鞋子

    持久性

    動作執行成功后,訂單生效,收入新增50生效,庫存減1生效

    上述三個服務他們可以在不同的地點,不同機器上部署的,並很常見。

    保證數據正確

    開啟事務

    確定要執行的服務,每個服務的數據庫事務開啟

    執行業務

    調用庫存減1,轉賬,生成訂單等子服務

    提交

    業務執行過程中沒有意外,各子服務的數據庫提交事務,生效數據修改

    回退

    回退,如果服務調用出現了差錯,或者某個子服務執行失敗,可以通過回滾所有數據庫達到數據正確。

    補償

    某些情況下,某個子服務執行失敗,但是不影響整體業務,也可以提交事務,後續補償機制將失敗的子服務重新執行。

    補償機制

    個人認為就商品購買而言,補償機制多數情況可以使用且實用。(對強一致要求沒那麼高的情況下)

    @1 庫存減1

    @2 收入增加50

    @ 3生成訂單記錄

    如果這次執行的動作, 只有@3失敗,@1 @2成功 說明金額交易,商品庫存業務都沒問題,只是訂單記錄失敗,這是可以提交事務的,訂單錯誤可以生成一條記錄(攜帶商品,金額等信息),發送到MQ消息隊列(或者其他設計)通過消息隊列通知訂單相關服務,補償重新執行生成訂單,達到最終一致性。

    分佈式事務控制問題

    不同服務在不同區運行

    不管是從安全性,穩定性,還是服務粒度細化方便維護等多因素考慮,都是很有必要讓不同的服務分開在不同服務區運行。

    單體數據庫的事務不被支持,購買商品到生成訂單所有操作加起來算一個事務,涉及的數據在不同一服務(不同的數據庫),並且同一個服務可能運行在多台服務器上。

    數據庫開啟事務針對的是單台服務器,多個服務多個數據庫,並不支持數據庫的事務,需要額外設計處理數據一致性問題(或者最終一致性)

    同一個服務運行在多個區

    不同服務不在一個服務器,同樣的,分佈式為穩定性可用而生,因此,一個服務大多有在多個區的服務器上運行,開啟事務的時候,如何保證事務開啟提交等事務相關命令每次發送到同一個區的同一個服務器,也是一定要考慮的問題。

    分佈式事務處理方式

    如上所述分佈式服務代表多個數據庫,不支持數據庫的事務,

    如何保證事務中涉及的數據庫數據修改都提交生效或者都回滾。

    建立控制中心

    控制中心在執行業務時,統一發送開始事務的命令給三個服務,返回狀態

    狀態沒問題執行數據修改,

    都沒問題就發送給三個服務,提交事務,否在回滾事務

    消息機制事務

    MQ消息隊列,達到控制事務正確目的,項目中kafka聽的比較多,可在高併發環境下穩定運行,可以通過消息機制發送事務處理結果到子服務,子服務收到消息,通過分析消息內容,做出對應的操作,達到事務一致性或者最終一致性等目的
    思考圖:

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

    【其他文章推薦】

    ※帶您來了解什麼是 USB CONNECTOR  ?

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

    ※如何讓商品強力曝光呢? 網頁設計公司幫您建置最吸引人的網站,提高曝光率!

    ※綠能、環保無空污,成為電動車最新代名詞,目前市場使用率逐漸普及化

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

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

  • 一時技癢,擼了個動態線程池,源碼放Github了

    一時技癢,擼了個動態線程池,源碼放Github了

    闡述背景

    線程池在日常工作中用的還挺多,當需要異步,批量處理一些任務的時候我們會定義一個線程池來處理。

    在使用線程池的過程中有一些問題,下面簡單介紹下之前遇到的一些問題。

    場景一:實現一些批量處理數據的功能,剛開始線程池的核心線程數設的比較小,然後想調整下,只能改完后重啟應用。

    場景二:有一個任務處理的應用,會接收 MQ 的消息進行任務的處理,線程池的隊列也允許緩存一定數量的任務。當任務處理的很慢的時候,想看看到底有多少沒有處理完不是很方便。當時為了快速方便,就直接啟動了一個線程去循環打印線程池隊列的大小。

    正好之前在我公眾號有轉發過美團的一篇線程池應用的文章(https://mp.weixin.qq.com/s/tIWAocevZThfbrfWoJGa9w),覺得他們的思路非常好,就是沒有開放源碼,所以自己就抽時間在我的開源項目 Kitty 中增加了一個動態線程池的組件,支持了 Cat 監控,動態變更核心參數,任務堆積告警等。今天就給大家分享一下實現的方式。

    項目源代碼地址:https://github.com/yinjihuan/kitty

    使用方式

    添加依賴

    依賴線程池的組件,目前 Kitty 未發布,需要自己下載源碼 install 本地或者私有倉庫。

    <dependency>
        <groupId>com.cxytiandi</groupId>
        <artifactId>kitty-spring-cloud-starter-dynamic-thread-pool</artifactId>
    </dependency>
    

    添加配置

    然後在 Nacos 配置線程池的信息,我的這個整合了 Nacos。推薦一個應用創建一個單獨的線程池配置文件,比如我們這個叫 dataId 為 kitty-cloud-thread-pool.properties,group 為 BIZ_GROUP。

    內容如下:

    kitty.threadpools.nacosDataId=kitty-cloud-thread-pool.properties
    kitty.threadpools.nacosGroup=BIZ_GROUP
    kitty.threadpools.accessToken=ae6eb1e9e6964d686d2f2e8127d0ce5b31097ba23deee6e4f833bc0a77d5b71d
    kitty.threadpools.secret=SEC6ec6e31d1aa1bdb2f7fd5eb5934504ce09b65f6bdc398d00ba73a9857372de00
    kitty.threadpools.owner=尹吉歡
    kitty.threadpools.executors[0].threadPoolName=TestThreadPoolExecutor
    kitty.threadpools.executors[0].corePoolSize=4
    kitty.threadpools.executors[0].maximumPoolSize=4
    kitty.threadpools.executors[0].queueCapacity=5
    kitty.threadpools.executors[0].queueCapacityThreshold=5
    kitty.threadpools.executors[1].threadPoolName=TestThreadPoolExecutor2
    kitty.threadpools.executors[1].corePoolSize=2
    kitty.threadpools.executors[1].maximumPoolSize=4
    

    nacosDataId,nacosGroup

    監聽配置修改的時候需要知道監聽哪個 DataId,值就是當前配置的 DataId。

    accessToken,secret

    釘釘機器人的驗證信息,用於告警。

    owner

    這個應用的負責人,告警的消息中會显示。

    threadPoolName

    線程池的名稱,使用的時候需要關注。

    剩下的配置就不一一介紹了,跟線程池內部的參數一致,還有一些可以查看源碼得知。

    注入使用

    @Autowired
    private DynamicThreadPoolManager dynamicThreadPoolManager;
    dynamicThreadPoolManager.getThreadPoolExecutor("TestThreadPoolExecutor").execute(() -> {
        log.info("線程池的使用");
        try {
            Thread.sleep(30000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }, "getArticle");
    

    通過 DynamicThreadPoolManager 的 getThreadPoolExecutor 方法獲取線程池對象,然後傳入 Runnable,Callable 等。第二個參數是這個任務的名稱,之所以要擴展一個參數是因為如果任務沒有標識,那麼無法區分任務。

    這個線程池組件默認集成了 Cat 打點,設置了名稱可以在 Cat 上查看這個任務相關的監控數據。

    擴展功能

    任務執行情況監控

    在 Cat 的 Transaction 報表中會以線程池的名稱為類型显示。

    詳情中會以任務的名稱显示。

    核心參數動態修改

    核心參數目前只支持 corePoolSize,maximumPoolSize,queueCapacity(隊列類型為 LinkedBlockingDeque 才可以修改),rejectedExecutionType,keepAliveTime,unit 這些參數的修改。

    一般 corePoolSize,maximumPoolSize,queueCapacity 是最常要動態改變的。

    需要改動的話直接在 Nacos 中將對應的配置值修改即可,客戶端會監聽配置的修改,然後同步修改先線程池的參數。

    隊列容量告警

    queueCapacityThreshold 是隊列容量告警的閥值,如果隊列中的任務數量超過了 queueCapacityThreshold 就會告警。

    拒絕次數告警

    當隊列容量滿了后,新進來的任務會根據用戶設置的拒絕策略去選擇對應的處理方式。如果是採用 AbortPolicy 策略,也會進行告警。相當於消費者已經超負荷了。

    線程池運行情況

    底層對接了 Cat,所以將線程的運行數據上報給了 Cat。我們可以在 Cat 中查看這些信息。

    如果你想在自己的平台去展示,我這邊暴露了/actuator/thread-pool 端點,你可以自行拉取數據。

    {
    	threadPools: [{
    		threadPoolName: "TestThreadPoolExecutor",
    		activeCount: 0,
    		keepAliveTime: 0,
    		largestPoolSize: 4,
    		fair: false,
    		queueCapacity: 5,
    		queueCapacityThreshold: 2,
    		rejectCount: 0,
    		waitTaskCount: 0,
    		taskCount: 5,
    		unit: "MILLISECONDS",
    		rejectedExecutionType: "AbortPolicy",
    		corePoolSize: 4,
    		queueType: "LinkedBlockingQueue",
    		completedTaskCount: 5,
    		maximumPoolSize: 4
    	}, {
    		threadPoolName: "TestThreadPoolExecutor2",
    		activeCount: 0,
    		keepAliveTime: 0,
    		largestPoolSize: 0,
    		fair: false,
    		queueCapacity: 2147483647,
    		queueCapacityThreshold: 2147483647,
    		rejectCount: 0,
    		waitTaskCount: 0,
    		taskCount: 0,
    		unit: "MILLISECONDS",
    		rejectedExecutionType: "AbortPolicy",
    		corePoolSize: 2,
    		queueType: "LinkedBlockingQueue",
    		completedTaskCount: 0,
    		maximumPoolSize: 4
    	}]
    }
    

    自定義拒絕策略

    平時我們使用代碼創建線程池可以自定義拒絕策略,在構造線程池對象的時候傳入即可。這裏由於創建線程池都被封裝好了,我們只能在 Nacos 配置拒絕策略的名稱來使用對應的策略。默認是可以配置 JDK 自帶的 CallerRunsPolicy,AbortPolicy,DiscardPolicy,DiscardOldestPolicy 這四種。

    如果你想自定義的話也是支持的,定義方式跟以前一樣,如下:

    @Slf4j
    public class MyRejectedExecutionHandler implements RejectedExecutionHandler {
        @Override
        public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {
            log.info("進來了。。。。。。。。。");
        }
    }
    

    要讓這個策略生效的話使用的是 SPI 的方式,需要在 resources 下面創建一個 META-INF 的文件夾,然後創建一個 services 的文件夾,再創建一個 java.util.concurrent.RejectedExecutionHandler 的文件,內容為你定義的類全路徑。

    自定義告警方式

    默認是內部集成了釘釘機器人的告警方式,如果你不想用也可以將其關閉。或者將告警信息對接到你的監控平台去。

    如果沒有告警平台也可以在項目中實現新的告警方式,比如短信等。

    只需要實現 ThreadPoolAlarmNotify 這個類即可。

    /**
     * 自定義短信告警通知
     *
     * @作者 尹吉歡
     * @個人微信 jihuan900
     * @微信公眾號 猿天地
     * @GitHub https://github.com/yinjihuan
     * @作者介紹 http://cxytiandi.com/about
     * @時間 2020-05-27 22:26
     */
    @Slf4j
    @Component
    public class ThreadPoolSmsAlarmNotify implements ThreadPoolAlarmNotify {
        @Override
        public void alarmNotify(AlarmMessage alarmMessage) {
            log.info(alarmMessage.toString());
        }
    }
    

    代碼實現

    具體的就不講的很細了,源碼在https://github.com/yinjihuan/kitty/tree/master/kitty-dynamic-thread-pool,大家自己去看,並不複雜。

    創建線程池

    根據配置創建線程池,ThreadPoolExecutor 是自定義的,因為需要做 Cat 埋點。

    /**
     * 創建線程池
     * @param threadPoolProperties
     */
    private void createThreadPoolExecutor(DynamicThreadPoolProperties threadPoolProperties) {
        threadPoolProperties.getExecutors().forEach(executor -> {
            KittyThreadPoolExecutor threadPoolExecutor = new KittyThreadPoolExecutor(
                    executor.getCorePoolSize(),
                    executor.getMaximumPoolSize(),
                    executor.getKeepAliveTime(),
                    executor.getUnit(),
                    getBlockingQueue(executor.getQueueType(), executor.getQueueCapacity(), executor.isFair()),
                    new KittyThreadFactory(executor.getThreadPoolName()),
                    getRejectedExecutionHandler(executor.getRejectedExecutionType(), executor.getThreadPoolName()), executor.getThreadPoolName());
            threadPoolExecutorMap.put(executor.getThreadPoolName(), threadPoolExecutor);
        });
    }
    

    刷新線程池

    首先需要監聽 Nacos 的修改。

    /**
     * 監聽配置修改,spring-cloud-alibaba 2.1.0版本不支持@NacosConfigListener的監聽
     */
    public void initConfigUpdateListener(DynamicThreadPoolProperties dynamicThreadPoolProperties) {
        ConfigService configService = nacosConfigProperties.configServiceInstance();
        try {
            configService.addListener(dynamicThreadPoolProperties.getNacosDataId(), dynamicThreadPoolProperties.getNacosGroup(), new AbstractListener() {
                @Override
                public void receiveConfigInfo(String configInfo) {
                    new Thread(() -> refreshThreadPoolExecutor()).start();
                    log.info("線程池配置有變化,刷新完成");
                }
            });
        } catch (NacosException e) {
            log.error("Nacos配置監聽異常", e);
        }
    }
    

    然後再刷新線程池的參數信息,由於監聽事件觸發的時候,這個時候配置其實還沒刷新,所以我就等待了 1 秒鐘,讓配置完成刷新然後直接從配置類取值。

    雖然有點挫還是可以用,其實更好的方式是解析 receiveConfigInfo 那個 configInfo,configInfo 就是改變之後的整個配置內容。因為不太好解析成屬性文件,就沒做,後面再改吧。

    /**
     * 刷新線程池
     */
    private void refreshThreadPoolExecutor() {
        try {
            // 等待配置刷新完成
            Thread.sleep(1000);
        } catch (InterruptedException e) {
        }
        dynamicThreadPoolProperties.getExecutors().forEach(executor -> {
            ThreadPoolExecutor threadPoolExecutor = threadPoolExecutorMap.get(executor.getThreadPoolName());
            threadPoolExecutor.setCorePoolSize(executor.getCorePoolSize());
            threadPoolExecutor.setMaximumPoolSize(executor.getMaximumPoolSize());
            threadPoolExecutor.setKeepAliveTime(executor.getKeepAliveTime(), executor.getUnit());
            threadPoolExecutor.setRejectedExecutionHandler(getRejectedExecutionHandler(executor.getRejectedExecutionType(), executor.getThreadPoolName()));
            BlockingQueue<Runnable> queue = threadPoolExecutor.getQueue();
            if (queue instanceof ResizableCapacityLinkedBlockIngQueue) {
                ((ResizableCapacityLinkedBlockIngQueue<Runnable>) queue).setCapacity(executor.getQueueCapacity());
            }
        });
    }
    

    其他的刷新都是線程池自帶的,需要注意的是線程池隊列大小的刷新,目前只支持 LinkedBlockingQueue 隊列,由於 LinkedBlockingQueue 的大小是不允許修改的,所以按照美團那篇文章提供的思路,自定義了一個可以修改的隊列,其實就是把 LinkedBlockingQueue 的代碼複製了一份,改一下就可以。

    往 Cat 上報運行信息

    往 Cat 的 Heartbeat 報表上傳數據的代碼如下,主要還是 Cat 本身提供了擴展的能力。只需要定時去調用下面的方式上報數據即可。

    public void registerStatusExtension(ThreadPoolProperties prop, KittyThreadPoolExecutor executor) {
        StatusExtensionRegister.getInstance().register(new StatusExtension() {
            @Override
            public String getId() {
                return "thread.pool.info." + prop.getThreadPoolName();
            }
            @Override
            public String getDescription() {
                return "線程池監控";
            }
            @Override
            public Map<String, String> getProperties() {
                AtomicLong rejectCount = getRejectCount(prop.getThreadPoolName());
                Map<String, String> pool = new HashMap<>();
                pool.put("activeCount", String.valueOf(executor.getActiveCount()));
                pool.put("completedTaskCount", String.valueOf(executor.getCompletedTaskCount()));
                pool.put("largestPoolSize", String.valueOf(executor.getLargestPoolSize()));
                pool.put("taskCount", String.valueOf(executor.getTaskCount()));
                pool.put("rejectCount", String.valueOf(rejectCount == null ? 0 : rejectCount.get()));
                pool.put("waitTaskCount", String.valueOf(executor.getQueue().size()));
                return pool;
            }
        });
    }
    

    定義線程池端點

    通過自定義端點來暴露線程池的配置和運行的情況,可以讓外部的監控系統拉取數據做對應的處理。

    @Endpoint(id = "thread-pool")
    public class ThreadPoolEndpoint {
        @Autowired
        private DynamicThreadPoolManager dynamicThreadPoolManager;
        @Autowired
        private DynamicThreadPoolProperties dynamicThreadPoolProperties;
        @ReadOperation
        public Map<String, Object> threadPools() {
            Map<String, Object> data = new HashMap<>();
            List<Map> threadPools = new ArrayList<>();
            dynamicThreadPoolProperties.getExecutors().forEach(prop -> {
                KittyThreadPoolExecutor executor = dynamicThreadPoolManager.getThreadPoolExecutor(prop.getThreadPoolName());
                AtomicLong rejectCount = dynamicThreadPoolManager.getRejectCount(prop.getThreadPoolName());
                Map<String, Object> pool = new HashMap<>();
                Map config = JSONObject.parseObject(JSONObject.toJSONString(prop), Map.class);
                pool.putAll(config);
                pool.put("activeCount", executor.getActiveCount());
                pool.put("completedTaskCount", executor.getCompletedTaskCount());
                pool.put("largestPoolSize", executor.getLargestPoolSize());
                pool.put("taskCount", executor.getTaskCount());
                pool.put("rejectCount", rejectCount == null ? 0 : rejectCount.get());
                pool.put("waitTaskCount", executor.getQueue().size());
                threadPools.add(pool);
            });
            data.put("threadPools", threadPools);
            return data;
        }
    }
    

    Cat 監控線程池中線程的執行時間

    本來是將監控放在 KittyThreadPoolExecutor 的 execute,submit 方法里的。後面測試下來發現有問題,數據在 Cat 上確實有了,但是執行時間都是 1 毫秒,也就是沒生效。

    不說想必大家也知道,因為線程是後面單獨去執行的,所以再添加任務的地方埋點沒任務意義。

    後面還是想到了一個辦法來實現埋點的功能,就是利用線程池提供的 beforeExecute 和 afterExecute 兩個方法,在線程執行之前和執行之後都會觸發這兩個方法。

    @Override
    protected void beforeExecute(Thread t, Runnable r) {
        String threadName = Thread.currentThread().getName();
        Transaction transaction = Cat.newTransaction(threadPoolName, runnableNameMap.get(r.getClass().getSimpleName()));
        transactionMap.put(threadName, transaction);
        super.beforeExecute(t, r);
    }
    @Override
    protected void afterExecute(Runnable r, Throwable t) {
        super.afterExecute(r, t);
        String threadName = Thread.currentThread().getName();
        Transaction transaction = transactionMap.get(threadName);
        transaction.setStatus(Message.SUCCESS);
        if (t != null) {
            Cat.logError(t);
            transaction.setStatus(t);
        }
        transaction.complete();
        transactionMap.remove(threadName);
    }
    

    後面的代碼大家自己去看就行了,本文到這裏就結束了。如果感覺本文還不錯的記得轉發下哦!

    多謝多謝。

    最後感謝美團技術團隊的那篇文章,雖然沒有分享源碼,但是思路什麼的和應用場景都講的很明白。

    感興趣的 Star 下唄:https://github.com/yinjihuan/kitty

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

    【其他文章推薦】

    ※為什麼 USB CONNECTOR 是電子產業重要的元件?

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

    ※台北網頁設計公司全省服務真心推薦

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

    ※推薦評價好的iphone維修中心