基于ignite的分布式框架

項目背景

最近在做公司項目的微服務(wù)改造,在dubbo和spring-cloud這兩個主流的微服務(wù)框架之前技術(shù)選型徘徊了好久,兩個框架在性能,設(shè)計,社區(qū)支持都非常地完美。然而我為什么要選擇自己去封裝ignite去實現(xiàn)RPC服務(wù)呢?

  • 原因一:
    公司缺乏專門的運維人員去維護(hù)第三方的中間件服務(wù),例如 Redis,Zookeeper,Rabbitmq等等的這些中間件。
  • 原因二:
    應(yīng)用本身要求的高并發(fā)不多,但是需要做高可用,引入dubbo和spring-cloud視乎有點小題大做的感覺
  • 原因三:
    使用dubbo和spring-cloud對現(xiàn)有項目的代碼改造工作量有點大,本身公司項目有部分已經(jīng)基于ignite做服務(wù)網(wǎng)格了
  • 原因四:
    工作這么久了,還是要嘗試封裝一下框架,所以這個框架是我的處女做,希望大家能夠批評和指正

項目概述

底層基于apache ignite,特點是可以做到不依賴外部中間件,實現(xiàn)RPC服務(wù)分布式緩存分布式計算分布式消息等功能特性
框架也基于ignite的集群管理,實現(xiàn)了基于集群組的顆粒度的服務(wù)調(diào)用,即針對集群組的調(diào)用

  • JDK和Spring boot版本
    JDK版本為1.8
    Spring boot 版本要求1.5.3以上

框架說明

服務(wù)中啟動的 Spring boot 應(yīng)用同時啟動了ignite的server和client模式,注入到了Spring容器中

    @Autowired
    @Qualifier("igniteClient")
    private Ignite igniteClient;

    @Autowired
    @Qualifier("igniteServer")
    private Ignite igniteServer;

因此你可以無縫地使用框架沒有封裝的ignite功能,更多的ignite的功能,請參考中文官網(wǎng)
https://www.ignite-service.cn/doc/java/

功能概述

  • RPC服務(wù)
  • 分布式消息
  • 分布式廣播
  • 分布式計算

quick-start

構(gòu)建

git clone https://github.com/konglinghai123/ignite-spring-project.git
cd ignite-spring-boot-starter
mvn clean install

構(gòu)建一個基于ignite的 spring boot 項目

  • 添加依賴:
   <dependency>
        <groupId>com.github.kong.spring.boot</groupId>
        <artifactId>ignite-spring-boot-starter</artifactId>
        <version>1.0</version>
   </dependency>
  • 在application-yml添加ignite的相關(guān)配置信息,樣例配置如下:

  • zookeeper 發(fā)現(xiàn)

#zookeeper發(fā)現(xiàn)
ignite-cluster:
  name: hello_client_1 #節(jié)點名稱
  role: client
  des: 測試服務(wù)
  zookeeperUrl: 192.168.56.100:2181
  localAddress: 127.0.0.1
  localPort: 47600
  • 動態(tài)ip發(fā)現(xiàn)
ignite-cluster:
  name: hello_server #節(jié)點名稱
  role: server
  des: 測試服務(wù)端
  multicast-group: 224.0.1.111 #組播地址
  localAddress: 127.0.0.1
  localPort: 48600
  • 為了開發(fā)方便,如果Spring boot Appliction 類的不在包名com.github.kong目錄下,接下來在Spring Boot Application的上添加@ComponentScan("com.github.kong.*"),這樣idea可以通過看到一些Bean是否已經(jīng)注入了,當(dāng)然也可以不添加,框架也有寫掃描注入
@SpringBootApplication
@ComponentScan("com.github.kong.*")
public class HelloWorldServerApplication {
    public static void main(String[] args) {
        SpringApplication.run(HelloWorldServerApplication.class);
    }
}

RPC服務(wù)的創(chuàng)建與消費

發(fā)布服務(wù)基于ignite的RPC服務(wù)

  • 編寫你的ignite服務(wù),需要添加要發(fā)布的服務(wù)實現(xiàn)上添加@IgniteRpcService注解,繼承IgniteService.
  • HelloWorld 是定義的接口
@Service
@IgniteRpcService(des = "這是一個例子")
public class HelloWorldService extends IgniteService implements HelloWorld {


}
  • @IgniteRpcService 注解的定義
/**
 * 服務(wù)提供者注解
 */
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface IgniteRpcService {

    /**
     * @return
     */
    String version() default "1.0";

    //接口描述
    String des() default "";

    //單個節(jié)點部署的實例數(shù)
    int maxPerNodeCount() default 1;

    //整個集群部署的最大實例數(shù),0:無限制
    int total() default 0;

}
  • 啟動你的Spring Boot應(yīng)用,觀察控制臺,可以看到ignite啟動相關(guān)信息.

調(diào)用已經(jīng)發(fā)布的RPC服務(wù)

  • Spring boot 應(yīng)用配置同上,唯一不同的是,需要更改配置
#zookeeper發(fā)現(xiàn)
ignite-cluster:
  name: hello_client_1 #節(jié)點名稱 (必須在集群中唯一)
  • 通過@IgniteRpcReference注入需要使用的interface.
@Controller
public class HelloWorldController {

    @IgniteRpcReference
    private HelloWorld helloWorldService;

    @RequestMapping("/helloworld")
    @ResponseBody
    public String test(){
        return helloWorldService.sayHello("kong");
    }

}
  • 調(diào)用不同版本的RPC服務(wù)
 @IgniteRpcReference(version = "1.1")
    private HelloWorld helloWorldService;
  • @IgniteRpcReference 注解的定義
/**
 * 網(wǎng)格服務(wù)注入注解
 */
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface IgniteRpcReference {

    String version() default "1.0";

    //默認(rèn)使用負(fù)載均衡
    boolean isLoadbalance() default true;

    //默認(rèn)不設(shè)超時
    long timeout() default 0;
}

分布式消息

分布式消息是基于內(nèi)存的消息訂閱系統(tǒng),如果需要持久化請使用外部的消息系統(tǒng)

定義話題消費者

@Service
@IgniteMessageListener(topic = "hello",isBroadcast = false)
public class HelloWorldMessage implements IgniteMessageRecevicer<String> {

    @Override
    public boolean apply(UUID uuid, MessageModel<String> messageModel) {
        System.out.println(messageModel);
        return true;
    }
}
  • @IgniteMessageListener 注解的定義
/**
 * 服務(wù)提供者注解
 */
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface IgniteMessageListener {


    //消息主題
    String topic();

    //消息描述
    String des() default "";

    //是否針對集群內(nèi)的所有節(jié)點(是否允許重復(fù)消費)
    boolean isBroadcast() default true;

}
  • MessageModel 是一個消息封裝,發(fā)送消息時必須用它來發(fā)送

發(fā)送話題消息

@Controller
@RequestMapping("/message")
public class MessageController {

    @Autowired
    private IgniteMessageSender sender;

    @RequestMapping("/sayHello")
    @ResponseBody
    public String test(){
        sender.toRemote("hello", new MessageModel<>("1212"));
        return "1212";
    }
}
  • IgniteMessageSender是框架注入的Bean,可以直接引用

<span id="BroadCast">分布式廣播</span>

分布式廣播是指:對集群組的所有節(jié)點發(fā)送消息,然后獲取所有節(jié)點返回的結(jié)果,原來是基于ignite的分布式閉包利用反射機制調(diào)用spring容器內(nèi)Bean的方法

發(fā)送一個分布式廣播

@Controller
@RequestMapping("/broadcast")
public class BroadcastController {

    @Autowired
    private BroadcastServiceExecutor broadcastServiceExecutor;

    @RequestMapping("/sayHello")
    @ResponseBody
    public List<String> test(){
        return (List<String>) broadcastServiceExecutor.broadcast("server", TestBroadService.class,"sayHello","12123");
    }
}
  • BroadcastServiceExecutor是框架注入的Bean,可以直接引用

  • broadcast 方法提供3個方法定義

   /**
     * 向其他集群廣播
     * @param targetRole 集群標(biāo)識
     * @param targetClass api類
     * @param methodName 方法名稱
     * @param args 參數(shù)
     * @return
     */
    public List broadcast(String targetRole,Class targetClass,String methodName,Object... args){...}

      /**
         * 向遠(yuǎn)端集群廣播消息
         * @param targetClass
         * @param methodName
         * @param args
         * @return
         */
     public List broadcastRemote(Class targetClass,String methodName,Object... args){...}


        /**
          * 向集群內(nèi)廣播消息
          * @param targetClass
          * @param methodName
          * @param args
          * @return
          */
      public List broadcastLocal(Class targetClass,String methodName,Object... args){...}

分布式計算

分布式計算允許用戶執(zhí)行基于內(nèi)存的Map-Reduce任務(wù)

  • 創(chuàng)建 Map-Reduce 任務(wù) ,需繼承ComputeTaskSplitAdapter(import org.apache.ignite.compute.ComputeTaskSplitAdapter),泛型<T,R>
  • T:入?yún)ⅲ琑:返回類型
//字?jǐn)?shù)統(tǒng)計測試
@Service
public class MapExampleCharacterCountTask  extends ComputeTaskSplitAdapter<List<String>,Integer>  {


    @Nullable
    @Override
    public Integer reduce(List<ComputeJobResult> results) throws IgniteException {
        return results.stream().mapToInt(ComputeJobResult::<Integer>getData).sum();
    }

    @Override
    protected Collection<? extends ComputeJob> split(int gridSize, List<String> arg) throws IgniteException {
        LinkedList jobs = new LinkedList();

        List<List<String>> list = CollectionUtils.split(arg,10000);

        for(final List<String> words : list){
            jobs.add(new ComputeJobAdapter() {
                @Override
                public Object execute() throws IgniteException {
                    int i = 0;
                    for(String s : words){
                        i = i + s.length();
                    }
                    return i;
                }
            });
        }

        return jobs;
    }
}

  • 執(zhí)行 Map-Reduce 任務(wù)
@Controller
@RequestMapping("/mr")
public class MRTestController {

    @Autowired
    private MapReduceTaskExecutor<List<String>,Integer> mapReduceTaskExecutor;

    @RequestMapping("/test")
    @ResponseBody
    public Object test(){
        try {

            List<String> records = new ArrayList<>();
            // 創(chuàng)建CSV讀對象
            CsvReader csvReader = new CsvReader(new FileInputStream("D:\\data\\cs2.csv"), Charset.forName("GBK"));

            while (csvReader.readRecord()){
                // 讀一整行
                records.add(csvReader.getRawRecord());
            }

            List<String> bigRecords = new ArrayList<>();
            for(int i = 0; i < 5; i++){
                bigRecords.addAll(records);
            }
            return mapReduceTaskExecutor.execute(MapExampleCharacterCountTask.class,bigRecords);
        } catch (IOException e) {
            e.printStackTrace();
        }

        return "";
    }

}
  • MapReduceTaskExecutor是框架注入的Bean,可以直接引用

集群管理api

框架注入了IgniteManager這個bean,可以實現(xiàn)以下功能

public interface IgniteManager {

    /**
     * 獲取節(jié)點列表
     *
     * @return
     */
    List<NodeInfo> list();

    /**
     * 獲取節(jié)點的詳細(xì)信息
     *
     * @param nodeId
     * @return
     */
    ClusterMetrics info(String nodeId);

    /**
     * 獲取微服務(wù)的基本信息
     *
     * @return
     */
    List<ServiceInfo> servieInfos();

    /**
     * 集群消息信息
     * @return
     */
    List<MessageInfo> messagInfos();

}
  • 使用@Autowired 注入即可
@Controller
@RequestMapping("/admin")
public class AdminCotroller {

    @Autowired
    private IgniteManager igniteManager;

    @RequestMapping("/nodes")
    @ResponseBody
    public List<NodeInfo> nodes(){
        return igniteManager.list();
    }


    @RequestMapping("/nodeInfo/{id}")
    @ResponseBody
    public ClusterMetrics info(@PathVariable("id") String id){
        return igniteManager.info(id);
    }

    @RequestMapping("/services")
    @ResponseBody
    public List<ServiceInfo> services(){
        return igniteManager.servieInfos();
    }

}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,431評論 6 544
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,637評論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,555評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,900評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 72,629評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,976評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,976評論 3 448
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,139評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,686評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,411評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,641評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,129評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,820評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,233評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,567評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,362評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 48,604評論 2 380

推薦閱讀更多精彩內(nèi)容