您好,登錄后才能下訂單哦!
PhysicalDBNode 是Mycat集群(Datanode)的對(duì)應(yīng),引用一個(gè)連接池對(duì)象 PhysicalDBPool,
PhysicalDBPool 里面引用了真正的連接池對(duì)象 PhysicalDatasource,PhysicalDBPool 里面把該
集群的讀節(jié)點(diǎn),寫節(jié)點(diǎn)寫入各自的 PhysicalDatasource 數(shù)組,通過負(fù)載均衡決定走哪個(gè)節(jié)點(diǎn)
負(fù)載均衡策略:隨機(jī)選擇,按權(quán)重設(shè)置隨機(jī)概率
代碼:randomSelect
節(jié)點(diǎn)權(quán)重計(jì)算公式String weightStr = node.getAttribute("weight");
int weight = "".equals(weightStr) ? PhysicalDBPool.WEIGHT : Integer.parseInt(weightStr) ;
負(fù)載均衡:offset -= okSources.get(i).getConfig().getWeight();
沒明白為什么這么分配,難道可用達(dá)到權(quán)重越大,分配可能性越小???
public PhysicalDatasource randomSelect(ArrayList<PhysicalDatasource> okSources) {
if (okSources.isEmpty()) {
return this.getSource();
} else {
int length = okSources.size(); // 總個(gè)數(shù)
int totalWeight = 0; // 總權(quán)重
boolean sameWeight = true; // 權(quán)重是否都一樣
for (int i = 0; i < length; i++) {
int weight = okSources.get(i).getConfig().getWeight();
totalWeight += weight; // 累計(jì)總權(quán)重
if (sameWeight && i > 0
&& weight != okSources.get(i-1).getConfig().getWeight() ) { // 計(jì)算所有權(quán)重是否一樣
sameWeight = false;
}
}
if (totalWeight > 0 && !sameWeight ) {
// 如果權(quán)重不相同且權(quán)重大于0則按總權(quán)重?cái)?shù)隨機(jī)
int offset = random.nextInt(totalWeight);
// 并確定隨機(jī)值落在哪個(gè)片斷上
for (int i = 0; i < length; i++) {
offset -= okSources.get(i).getConfig().getWeight();
if (offset < 0) {
return okSources.get(i);
}
}
}
// 如果權(quán)重相同或權(quán)重為0則均等隨機(jī)
return okSources.get( random.nextInt(length) );
//int index = Math.abs(random.nextInt()) % okSources.size();
//return okSources.get(index);
}
}
PhysicalDatasource 連接池對(duì)象保存該連接的可用連接使用的數(shù)據(jù)結(jié)構(gòu)是,ConMap,主要功能是獲取當(dāng)前節(jié)點(diǎn)的可用連接,首先從當(dāng)前database上獲取可用連接,如果沒有,則從其他 database 上獲取可用連接
public BackendConnection tryTakeCon(final String schema, boolean autoCommit) {
final ConQueue queue = items.get(schema);
BackendConnection con = tryTakeCon(queue, autoCommit);
if (con != null) {
return con;
} else {
for (ConQueue queue2 : items.values()) {
if (queue != queue2) {
con = tryTakeCon(queue2, autoCommit);
if (con != null) {
return con;
}
}
}
}
return null;
}
private BackendConnection tryTakeCon(ConQueue queue, boolean autoCommit) {
BackendConnection con = null;
if (queue != null && ((con = queue.takeIdleCon(autoCommit)) != null)) {
return con;
} else {
return null;
}
}
database的可用連接是存放在數(shù)據(jù)結(jié)構(gòu)ConQueue中的,可用連接分為自動(dòng)提交,手動(dòng)提交,所以ConQueue由2個(gè)ConcurrentLinkedQueue組成,autoCommitCons 自動(dòng)提交隊(duì)列,manCommitCons 手動(dòng)提交隊(duì)列
分配可用連接:先從提交方式隊(duì)列隊(duì)首分配,分配失敗,從另一個(gè)隊(duì)列分配,分配失敗,從其他databse 分配。猜想:此處分配完成應(yīng)該不是最種的可用連接,還需要做事務(wù)隔離級(jí)別、事務(wù)模式、字符集、Database 等等處理和校驗(yàn),才能執(zhí)行具體的 sql 指令,這些應(yīng)該是在MySQLConnection 類中進(jìn)行的
public BackendConnection takeIdleCon(boolean autoCommit) {
ConcurrentLinkedQueue<BackendConnection> f1 = autoCommitCons;
ConcurrentLinkedQueue<BackendConnection> f2 = manCommitCons;
if (!autoCommit) {
f1 = manCommitCons;
f2 = autoCommitCons;
}
BackendConnection con = f1.poll();
if (con == null || con.isClosedOrQuit()) {
con = f2.poll();
}
if (con == null || con.isClosedOrQuit()) {
return null;
} else {
return con;
}
}
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。