JAVA分布式開發(fā)中遇到的哪些坑(一)

一举庶、Spring使用過程中的踩坑記錄

image
  • Spring通過注解使用多數(shù)據(jù)源

坑:@Autowired 按 byType 自動(dòng)注入,而 @Resource 則默認(rèn)按 byName 自動(dòng)注入觅玻,@Primary是優(yōu)先選擇疮方。

例如悍募,在項(xiàng)目中是有兩個(gè)Redis源,這兩個(gè)Redis Bean分別為dataRedisTemplate和redisTemplate常侦。

Redis Bean1:dataRedisTemplate浇冰,clusterNodes為${data-redis.cluster.nodes}

@Bean(name = "dataRedisTemplate")
public RedisTemplate dataRedisTemplate() {
    RedisTemplate template = new RedisTemplate();
    template.setConnectionFactory(sessionLettuceConnectionFactory);
    Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
    ObjectMapper om = new ObjectMapper();
    om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
    om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
    jackson2JsonRedisSerializer.setObjectMapper(om);
    template.setKeySerializer(new StringRedisSerializer());
    template.setValueSerializer(jackson2JsonRedisSerializer);
    template.setHashKeySerializer(jackson2JsonRedisSerializer);
    template.setHashValueSerializer(new StringRedisSerializer());
    template.afterPropertiesSet();
    return template;
}

// factory
@Resource
@Qualifier(value = "dataLettuceConnectionFactory")
private RedisConnectionFactory dataLettuceConnectionFactory;

// clusterNodes
@Value("${spring.data-redis.cluster.nodes}")
private String clusterNodes;

Redis Bean2:redisTemplate予弧,clusterNodes為${redis.cluster.nodes}

@Primary
@Bean(name = "redisTemplate")
public RedisTemplate<String, Object> redisTemplate() {
    RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
    redisTemplate.setConnectionFactory(lettuceConnectionFactory);
    redisTemplate.setKeySerializer(new StringRedisSerializer());
    Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
    objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
    jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
    redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
    redisTemplate.setHashKeySerializer(new StringRedisSerializer());
    redisTemplate.setHashValueSerializer(new StringRedisSerializer());
    redisTemplate.afterPropertiesSet();
    return redisTemplate;
}

// factory
@Resource
@Qualifier(value = "lettuceConnectionFactory")
private RedisConnectionFactory lettuceConnectionFactory;

// clusterNodes
@Value("${spring.redis.cluster.nodes}")
private String clusterNodes;

在一個(gè)應(yīng)用中要把數(shù)據(jù)放入到“Redis Bean1:dataRedisTemplate”對(duì)應(yīng)的Redis中,于是我在這個(gè)應(yīng)用中使用方式如下:

@Autowired
private RedisTemplate dataRedisTemplate;

// 根據(jù)key獲取數(shù)據(jù)
Object obj = dataRedisTemplate.opsForValue().get(key);

實(shí)際上使用的是“Redis Bean2:redisTemplate”對(duì)應(yīng)的Redis湖饱。
破解方式一:把@Autowired換成@Resource 注解掖蛤。如下:

@Autowired
private RedisTemplate dataRedisTemplate;

// 把@Autowired換成@Resource

@Resource
private RedisTemplate dataRedisTemplate;

@Autowired和@Resource最大的區(qū)別就是:@Autowired 按 byType 自動(dòng)注入,而 @Resource 則默認(rèn)按 byName 自動(dòng)注入井厌。

這里還需要注意一個(gè)注解@Primary蚓庭,官方的說明如下:

Indicates that a bean should be given preference when multiple candidates are qualified to autowire a single-valued dependency. If exactly one 'primary' bean exists among the candidates, it will be the autowired value.

@Primary 優(yōu)先方案,被注解的實(shí)現(xiàn)仅仆,優(yōu)先被注入

通常情況下@Autowired是通過byType的方法注入的器赞,可是在多個(gè)實(shí)現(xiàn)類的時(shí)候,byType的方式不再是唯一墓拜,而需要通過byName的方式來注入港柜,而這個(gè)name默認(rèn)就是根據(jù)變量名來的。

也就是說咳榜,如果沒有在redisTemplate()上面增加@Primary的話是沒有問題的夏醉,因?yàn)橛卸鄠€(gè)實(shí)現(xiàn)時(shí),@Autowired是會(huì)通過byName的方式來注入的涌韩,但是按照上面說的畔柔,因?yàn)橛辛薂Primary,@Autowired注解會(huì)優(yōu)先使用Bean redisTemplate臣樱。

還有一種解決方案是增加@Qualifier(value = "dataRedisTemplate")靶擦,如下:

@Autowired
private RedisTemplate dataRedisTemplate;

// 換成:增加@Qualifier(value = "dataRedisTemplate")

@Autowired
@Qualifier(value = "dataRedisTemplate")
private RedisTemplate dataRedisTemplate;
  • Spring事務(wù)@Transactional失效問題

坑:若同一類中的其他沒有@Transactional 注解的方法內(nèi)部調(diào)用有@Transactional 注解的方法,有@Transactional 注解的方法的事務(wù)被忽略雇毫,不會(huì)發(fā)生回滾玄捕。

FooService.class

public interface FooService {
    void insertRecord();
    void insertThenRollback() throws Exception;
    void invokeInsertThenRollback() throws Exception;
    void invokeInsertThenRollbackTwo() throws Exception;
}

FooServiceImpl.class

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional;

@Component
public class FooServiceImpl implements FooService {
    @Autowired
    private JdbcTemplate jdbcTemplate;
    @Autowired
    private FooService fooService;

    @Override
    @Transactional
    public void insertRecord() {
        jdbcTemplate.execute("INSERT INTO FOO (BAR) VALUES ('AAA')");
    }

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void insertThenRollback() throws Exception {
        jdbcTemplate.execute("INSERT INTO FOO (BAR) VALUES ('BBB')");
        throw new Exception();
    }

    @Override
    public void invokeInsertThenRollback() throws Exception {
        insertThenRollback();
    }

    @Override
    public void invokeInsertThenRollbackTwo() throws Exception {
        fooService.insertThenRollback();
    }
}

執(zhí)行

import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.AdviceMode;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.transaction.annotation.EnableTransactionManagement;

@SpringBootApplication
@EnableTransactionManagement(mode = AdviceMode.PROXY)
@Slf4j
public class DeclarativeTransactionDemoApplication implements CommandLineRunner {
    @Autowired
    private FooService fooService;
    @Autowired
    private JdbcTemplate jdbcTemplate;

    public static void main(String[] args) {
        SpringApplication.run(DeclarativeTransactionDemoApplication.class, args);
    }

    @Override
    public void run(String... args) throws Exception {
        // 標(biāo)記1:輸出 AAA 1
        fooService.insertRecord();
        log.info("AAA {}",jdbcTemplate.queryForObject("SELECT COUNT(*) FROM FOO WHERE BAR='AAA'", Long.class));
        
        // 標(biāo)記2:輸出 BBB 0,事務(wù)生效        
        try {
            fooService.insertThenRollback();
        } catch (Exception e) {
            log.info("BBB {}",jdbcTemplate.queryForObject("SELECT COUNT(*) FROM FOO WHERE BAR='BBB'", Long.class));
        }

        // 標(biāo)記3:輸出 BBB 1棚放,事務(wù)未生效
        // ***這個(gè)地方就是最容易踩坑的地方C墩场!席吴!***        
        try {
            fooService.invokeInsertThenRollback();
        } catch (Exception e) {
            log.info("BBB {}",jdbcTemplate.queryForObject("SELECT COUNT(*) FROM FOO WHERE BAR='BBB'", Long.class));
        }

        // 標(biāo)記4:輸出 BBB 1赌结,事務(wù)生效(如果把標(biāo)記3的代碼注釋掉,輸出 BBB 0)
        // ***這是避免踩坑的一種方式***        
        try {
            fooService.invokeInsertThenRollbackTwo();
        } catch (Exception e) {
            log.info("BBB {}",jdbcTemplate.queryForObject("SELECT COUNT(*) FROM FOO WHERE BAR='BBB'", Long.class));
        }
    }
}

二孝冒、RocketMQ使用過程中的踩坑記錄

image
  • Rocket默認(rèn)開啟了VIP通道導(dǎo)致10909failed問題
    坑:Rocket默認(rèn)開啟了VIP通道柬姚,VIP通道端口為10911-2=10909。若Rocket服務(wù)器未啟動(dòng)端口10909庄涡,則報(bào)connect to <:10909> failed量承。

解決方案:不走VIP通道。

producer.setVipChannelEnabled(false);
consumer.setVipChannelEnabled(false);
  • Rocket instanceName參數(shù)未做配置導(dǎo)致重復(fù)消費(fèi)問題

坑:一個(gè)是Rocket如果沒有配置instanceName,那么會(huì)使用pid做instanceName撕捍,如果instanceName一樣會(huì)重復(fù)消費(fèi)拿穴,因?yàn)榧合M(fèi)模式是按instanceName做為唯一消費(fèi)實(shí)例。

查看源碼忧风,如果沒有指定instanceName默認(rèn)會(huì)把pid做為instanceName默色,如下:

if (this.instanceName.equals("DEFAULT")) {
    this.instanceName = String.valueOf(UtilAll.getPid());
}

public static int getPid() {
    RuntimeMXBean runtime = ManagementFactory.getRuntimeMXBean();
    String name = runtime.getName(); // format: "pid@hostname"
    try {
        return Integer.parseInt(name.substring(0, name.indexOf('@')));
    }
    catch (Exception e) {
        return -1;
    }
}

解決方案:運(yùn)維配置有$MQ_INSTANCE_NAME 環(huán)境變量,不同機(jī)器不一樣狮腿,所以可以使用:mq.consumer.instanceName:${MQ_INSTANCE_NAME:默認(rèn)值}進(jìn)行配置腿宰。

@Value("${rocketmq.consumer.instanceName:${MQ_INSTANCE_NAME:fota}}")
private String clientInstanceName;

consumer.setInstanceName(clientInstanceName);

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市缘厢,隨后出現(xiàn)的幾起案子吃度,更是在濱河造成了極大的恐慌,老刑警劉巖贴硫,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件椿每,死亡現(xiàn)場離奇詭異,居然都是意外死亡英遭,警方通過查閱死者的電腦和手機(jī)间护,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來贪绘,“玉大人兑牡,你說我怎么就攤上這事央碟∷肮啵” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵亿虽,是天一觀的道長菱涤。 經(jīng)常有香客問我,道長洛勉,這世上最難降的妖魔是什么粘秆? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮收毫,結(jié)果婚禮上攻走,老公的妹妹穿的比我還像新娘。我一直安慰自己此再,他們只是感情好昔搂,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著输拇,像睡著了一般摘符。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天逛裤,我揣著相機(jī)與錄音瘩绒,去河邊找鬼。 笑死带族,一個(gè)胖子當(dāng)著我的面吹牛锁荔,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蝙砌,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼堕战,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了拍霜?” 一聲冷哼從身側(cè)響起嘱丢,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎祠饺,沒想到半個(gè)月后越驻,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡道偷,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年缀旁,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片勺鸦。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡并巍,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出换途,到底是詐尸還是另有隱情懊渡,我是刑警寧澤,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布军拟,位于F島的核電站剃执,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏懈息。R本人自食惡果不足惜肾档,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望辫继。 院中可真熱鬧怒见,春花似錦、人聲如沸姑宽。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽低千。三九已至配阵,卻和暖如春馏颂,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背棋傍。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來泰國打工救拉, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人瘫拣。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓亿絮,卻偏偏與公主長得像,于是被迫代替她去往敵國和親麸拄。 傳聞我的和親對(duì)象是個(gè)殘疾皇子派昧,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

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