Java開發技術大雜燴(三)之電商項目優化、rabbitmq、Git、OSI、VIM、Intellj IDEA、HTTP、JS、Java

前言

最近公司讓我維護Spring+Servlet+Hibernate+Spring Security+Jsp的老項目,正好能夠鍛鍊個人業務邏輯和掌控項目的能力。雖然項目很老,可是其中仍是有不少值我學習的地方。javascript

電商項目優化

1.咱們大體優化的點是秒殺接口:redis預減庫存,減小數據庫訪問;內存標記較少redis的訪問;rabbitmq隊列緩衝,異步下單,加強用戶體驗。那麼具體步驟以下。html

1.處理秒殺業務的Controller在Spring容器週期內加載就緒。也就是實現InitializingBean,在afterPropertiesSet()方法中把商品庫存加載到redis中,而且設置在內存中設置商品是否秒殺結束的flag。java

/**
     * 內存標記初始化
     * @throws Exception
     */
    @Override
    public void afterPropertiesSet() throws Exception {
        List<GoodsVo> goodsVoList = goodsService.listGoodsVo();

        if (CollectionUtils.isEmpty(goodsVoList)) {
            return;
        }

        goodsVoList.forEach(goodsVo -> {
            redisService.set(GoodsKey.getMiaoshaGoodsStock, "" + goodsVo.getId(), goodsVo.getStockCount());
            localOverMap.put(goodsVo.getId(), false);
        });
    }
複製代碼

2.後臺收到秒殺請求,首先查看內存flag標記,而後減小redis中的商品庫存。若是商品秒殺結束,在內存中設置秒殺結束的flag。若是商品秒殺還在進行中,那麼進入下一步。nginx

3.把秒殺商品的消息進行入隊緩衝,直接返回。這裏並非返回成功,而是返回到排隊中。此時,前臺不能直接提示秒殺成功,而是啓動定時器,過一段時間再去查看是否成功。git

4.消息出隊,修改db中的庫存,建立秒殺訂單。web

2.分佈式Session的解決方案是生成惟一token,token標識用戶,把token寫到Cookie中,而後把token+用戶信息寫進Redis,token在redis的失效時間要和Cookie失效時間保持一致。每當用戶登陸一次,要延遲Session的有效期和Cookie有效期。redis

3.從緩存的角度來講,咱們能夠進行頁面緩存+URL緩存+對象緩存來達到優化的目的。咱們能夠手動渲染Thymeleaf模板,把商品詳情頁和商品列表頁緩存到redis中,這裏用商品列表頁舉例。spring

@RequestMapping(value = "/to_list", produces = "text/html;charset=UTF-8")
    @ResponseBody
    public String list(MiaoshaUser miaoshaUser) throws IOException {
        modelMap.addAttribute("user", miaoshaUser);
        //取緩存
        String htmlCached = redisService.get(GoodsKey.getGoodsList, "", String.class);
        if (!StringUtils.isEmpty(htmlCached)) {
            return htmlCached;
        }
        List<GoodsVo> goodsVoList = goodsService.listGoodsVo();
        modelMap.addAttribute("goodsList", goodsVoList);
        SpringWebContext springWebContext = new SpringWebContext(request, response, request.getServletContext(),
                request.getLocale(), modelMap, applicationContext);
        String html = thymeleafViewResolver.getTemplateEngine().process("goods_list", springWebContext);

        if (!StringUtils.isEmpty(html)) {
            redisService.set(GoodsKey.getGoodsList, "", html);
        }
        return html;
    }
複製代碼

4.從靜態資源角度考慮,咱們進行頁面靜態化、先後端分離、靜態資源優化、CDN節點優化。這裏用靜態資源優化舉例。sql

1.JS/CSS壓縮、減小流量。 2.多個JS/CSS組合,減小鏈接數 3.CDN就近訪問,減小請求時間。 4.將一些界面緩存到用戶的瀏覽器中。數據庫

5.安全優化。密碼兩次加鹽,第一次加鹽是固定的,寫在Java代碼的。第二次加鹽是隨機的,存儲在數據庫中。在商品秒殺頁,添加數學公式驗證碼,分散用戶的請求。對接口加入限流防刷機制。這裏以接口限流防刷機制舉例。

1.定義AccessLimit註解,做用於方法。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface AccessLimit {

    int seconds();

    int maxCount();

    boolean needLogin() default true;
}
複製代碼

2.定義AccessInterceptor攔截器,得到方法中AccessLimit註解中的參數。請求的reqeusturi做爲redis中的key,seconds做爲key的失效時間。每次請求加1,若是在指定時間內訪問該url的次數超過設置的maxCount,那麼返回「訪問太頻繁」。

@Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        if (handler instanceof HandlerMethod) {
            MiaoshaUser user = getUser(request, response);
            UserContext.setUser(user);
            HandlerMethod hm = (HandlerMethod) handler;
            AccessLimit accessLimit = hm.getMethodAnnotation(AccessLimit.class);

            if (Objects.isNull(accessLimit)) {
                return true;
            }
            int seconds = accessLimit.seconds();
            int maxCount = accessLimit.maxCount();
            boolean needLogin = accessLimit.needLogin();
            String key = request.getRequestURI();

            if (needLogin) {
                if (Objects.isNull(user)) {
                    render(response, CodeMsg.SESSION_ERROR);
                    return false;
                }
            }

            AccessKey ak = AccessKey.withExpire(seconds);
            Integer count = redisService.get(ak, key, Integer.class);

            if (Objects.isNull(count)) {
                redisService.set(ak, key, 1);
            } else if (count < maxCount) {
                redisService.incr(ak, key);
            } else {
                render(response, CodeMsg.ACCESS_LIMIT_REACHED);
                return false;
            }
        }
        return true;
    }
複製代碼

6.部署優化。LVS+Keepalived雙機熱備模式+Nginx+Tomcat。


Intelli J IDEA使用技巧

1.全局搜索 Ctrl + Shift + F 2.全局替換 Ctrl +Shift + R


Vim編輯器使用技巧

1.在vim編輯器中進行查找。

1.命令模式輸入「/字符串」,例如"/xiaoma" 2.若是繼續查找下一個,按n便可。


Redis設置密碼

1.由於在application-dev.properties中配置了spring.redis.password=,若是沒有在redis.conf沒有設置requirepass ${password},控制檯會拋出鏈接拒絕異常。

HTTP

Cache Control的用法

no cache : 強制每次請求直接發送給源服務器,而不用通過本地緩存版本的校驗。 max-age > 0 : 直接從瀏覽器緩存中提取。


RabbitMQ

1.AMQP(Advance Message Queuing Protocol)是一個提供統一消息服務的應用層標準高級消息隊列協議,是應用層協議。

2.Exchange在RabbitMQ中充當交換機的角色,也至關於路由。固然也能夠形象的理解成RabbitMQ的過濾器。RabbitMQ有4種模式。

rabbitmq圖解-本圖來自於互聯網

1.Direct:按照Routing Key分到指定的Queue中。 2.Topic:和Direct差很少,可是能夠多關鍵字匹配。 3.Fanout:無Routing Key概念,至關於廣播模式,將消息分發給全部綁定FanoutExchange中的Queue。 4.Header:和上面3個不同,經過添加屬性key-value進行匹配。

3.編寫RabbitMQ代碼

配置RabbitMQ的4種模式

/**
 * @author cmazxiaoma
 * @version V1.0
 * @Description: TODO
 * @date 2018/6/4 11:36
 */
@Configuration
public class MQConfig {

    public static final String MIAOSHA_QUEUE = "miaosha.queue";
    public static final String QUEUE = "queue";
    public static final String TOPIC_QUEUE1 = "topic.queue1";
    public static final String TOPIC_QUEUE2 = "topic.queue2";
    public static final String HEADER_QUEUE = "header.queue";
    public static final String TOPIC_EXCHANGE = "topicExchange";
    public static final String FANOUT_EXCHANGE = "fanoutExchange";
    public static final String HEADERS_EXCHANGE = "headersExchange";

    /**
     * Direct模式
     * @return
     */
    @Bean
    public Queue queue() {
        return new Queue(QUEUE, true);
    }

    @Bean
    public Queue miaoshaoQue() {
        return new Queue(MQConfig.MIAOSHA_QUEUE, true);
    }

    /**
     * Topic模式
     * @return
     */
    @Bean
    public TopicExchange topicExchange() {
        return new TopicExchange(TOPIC_EXCHANGE);
    }

    @Bean
    public Queue topicQueue1() {
        return new Queue(TOPIC_QUEUE1, true);
    }

    @Bean
    public Queue topicQueue2() {
        return new Queue(TOPIC_QUEUE2, true);
    }

    @Bean
    public Binding topicBinding1() {
        return BindingBuilder
                .bind(topicQueue1())
                .to(topicExchange())
                .with("topic.key1");
    }

    @Bean
    public Binding topicBinding2() {
        return BindingBuilder
                .bind(topicQueue2())
                .to(topicExchange())
                .with("topic.#");
    }

    /**
     * Fanout模式
     * @return
     */
    @Bean
    public FanoutExchange fanoutExchange() {
        return new FanoutExchange(FANOUT_EXCHANGE);
    }

    @Bean
    public Binding fanoutBinding1() {
        return BindingBuilder.bind(topicQueue1())
                .to(fanoutExchange());
    }

    @Bean
    public Binding fanoutBinding2() {
        return BindingBuilder.bind(topicQueue2())
                .to(fanoutExchange());
    }

    /**
     * Header模式
     * @return
     */
    @Bean
    public HeadersExchange headersExchange() {
        return new HeadersExchange(HEADERS_EXCHANGE);
    }

    @Bean
    public Queue headerQueue1() {
        return new Queue(HEADER_QUEUE, true);
    }

    @Bean
    public Binding headerBinding() {
        Map<String, Object> map = new HashMap<>();
        map.put("header1", "value1");
        map.put("header2", "value2");
        return BindingBuilder.bind(headerQueue1()).to(headersExchange())
                .whereAll(map).match();
    }
}
複製代碼

配置消息生產者

/**
 * @author cmazxiaoma
 * @version V1.0
 * @Description: TODO
 * @date 2018/6/4 13:05
 */
@Service
@Slf4j
public class MQSender {

    @Autowired
    private AmqpTemplate amqpTemplate;

    public void sendMiaoshaMessageDirect(MiaoshaMessage miaoshaMessage) {
        String msg = RedisService.beanToString(miaoshaMessage);
        log.info("send direct message = {}", msg);
        amqpTemplate.convertAndSend(MQConfig.MIAOSHA_QUEUE, msg);
    }

    public void sendDirect(Object message) {
        String msg = RedisService.beanToString(message);
        log.info("send direct message = {}", msg);
        amqpTemplate.convertAndSend(MQConfig.QUEUE, msg);
    }

    public void sendTopic(Object message) {
        String msg = RedisService.beanToString(message);
        log.info("send topic message = {}", msg);
        amqpTemplate.convertAndSend(MQConfig.TOPIC_EXCHANGE, "topic.key1", msg + "-1");
        amqpTemplate.convertAndSend(MQConfig.TOPIC_EXCHANGE, "topic.key2", msg + "-2");
    }

    public void sendFanout(Object message) {
        String msg = RedisService.beanToString(message);
        log.info("send fanout message = {}", msg);
        amqpTemplate.convertAndSend(MQConfig.FANOUT_EXCHANGE, "", msg);
    }

    public void sendHeader(Object message) {
        String msg = RedisService.beanToString(message);
        log.info("send header message = {}", msg);
        MessageProperties messageProperties = new MessageProperties();
        messageProperties.setHeader("header1", "value1");
        messageProperties.setHeader("header2", "value2");
        Message newMessage = new Message(msg.getBytes(), messageProperties);
        amqpTemplate.convertAndSend(MQConfig.HEADERS_EXCHANGE, "", newMessage);
    }

}
複製代碼

配置消息消費者

/**
 * @author cmazxiaoma
 * @version V1.0
 * @Description: TODO
 * @date 2018/6/4 13:47
 */
@Service
@Slf4j
public class MQReceiver {

    @Autowired
    private RedisService redisService;

    @Autowired
    private GoodsService goodsService;

    @Autowired
    private OrderService orderService;

    @Autowired
    private MiaoshaService miaoshaService;

    @RabbitListener(queues = MQConfig.MIAOSHA_QUEUE)
    public void receiveMiaoshaMessageDirect(String message) {
        log.info("receive direct miaosha message = {}", message);
        MiaoshaMessage miaoshaMessage = RedisService.stringToBean(message, MiaoshaMessage.class);
        MiaoshaUser miaoshaUser = miaoshaMessage.getMiaoshaUser();
        Long goodsId = miaoshaMessage.getGoodsId();
        GoodsVo goodsVo = goodsService.getGoodsVoByGoodsId(goodsId);
        int stock = goodsVo.getStockCount();

        if (stock <= 0) {
            return;
        }
        //判斷是否已經秒殺過
        MiaoshaOrder order = orderService.getMiaoshaOrderByUserIdGoodsId(miaoshaUser.getId(), goodsId);

        if (!Objects.isNull(order)) {
            return;
        }
        //減庫存 下訂單 寫入秒殺訂單
        miaoshaService.miaosha(miaoshaUser, goodsVo);
    }

    @RabbitListener(queues = MQConfig.QUEUE)
    public void receiveDirect(String message) {
        log.info("receive direct message = {}", message);
    }

    @RabbitListener(queues = MQConfig.TOPIC_QUEUE1)
    public void receiveTopic1(String message) {
        log.info("receive topic queue1 message = {}", message);
    }

    @RabbitListener(queues = MQConfig.TOPIC_QUEUE2)
    public void receiveTopic2(String message) {
        log.info("receive topic queue2 message = {}", message);
    }

    @RabbitListener(queues = MQConfig.HEADER_QUEUE)
    public void receiveHeader(byte[] message) {
        log.info("receive header message = {}", new String(message));
    }
}
複製代碼

測試RabbitMQ的Controller

/**
 * @author cmazxiaoma
 * @version V1.0
 * @Description: TODO
 * @date 2018/5/29 16:36
 */
@Controller
@RequestMapping("/rabbitmq")
public class RabbitmqController extends BaseController {

    @Autowired
    private MQSender mqSender;

    @GetMapping("/header")
    @ResponseBody
    public Result<String> header() {
        mqSender.sendHeader("hello, header");
        return Result.success("hello, header");
    }

    @GetMapping("/fanout")
    @ResponseBody
    public Result<String> fanout() {
        mqSender.sendFanout("hello, fanout");
        return Result.success("hello, fanout");
    }

    @GetMapping("/topic")
    @ResponseBody
    public Result<String> topic() {
        mqSender.sendTopic("hello, topic");
        return Result.success("hello, topic");
    }

    @GetMapping("/direct")
    @ResponseBody
    public Result<String> direct() {
        mqSender.sendDirect("hello, direct");
        return Result.success("hello, direct");
    }
}
複製代碼

Nginx

Nginx的命令過一陣子不寫,總是忘記。仍是記在簡書上面吧。 啓動:/usr/local/nginx/sbin/nginx -C /usr/local/nginx/conf/nginx.conf 關閉:/usr/local/nginx/sbin/nginx -s stop

咱們在nginx.conf配置max_fail和fail_timeout參數,當失敗次數超過max_fail,nginx會把接下來的請求交給其餘Real Server去處理。fail_timeout是失敗等待時間,當請求被認定失敗後,等待fail_timeout時間再去請求,判斷是否成功。


Git容易混淆的知識點

工做區:包括實際更改的文件,當前修改還未add進入暫存區的文件變化信息。 暫存區:臨時存儲文件的變化信息

git reset filename:清空add命令向暫存區提交的關於filename文件的修改。 git checkout --filename:撤銷對工做區的修改。


JS基礎知識

衆所周知,Java有三大特性:封裝,繼承,多態。咱們能夠用JS的protoType往java對象中注入這三大特性。

<script type="text/javascript">
var myObject = {
	foo: "bar",
	func: function() {
		var self = this;
		console.log("outer func:this.foo=" + this.foo);
		console.log("outer func:self.foo=" + self.foo);
		
		(function() {
			console.log("inner func:this.foo=" + this.foo);
			console.log("inner func:self.foo=" + self.foo);
		}());
		
	}
};
myObject.func();

Java = function() {};
Java.prototype = {
	oriented: function() {
		console.log("面向對象");
	},
	fengzhuang: function() {
		console.log("封裝");
	},
	extend: function() {
		console.log("繼承");
	}
};
java = new Java();
java.oriented();
java.fengzhuang();
java.extend();
</script>
複製代碼

Spring MVC冷門註解

1.produces="text/html"表示方法將產生「text/html」格式的數據,而且響應條的ContentType。咱們在寫入消息返回響應前,調用addDefaultHeaders()設置響應條中ContentType和ContentLength屬性。

protected void addDefaultHeaders(HttpHeaders headers, T t, MediaType contentType) throws IOException{
		if (headers.getContentType() == null) {
			MediaType contentTypeToUse = contentType;
			if (contentType == null || contentType.isWildcardType() || contentType.isWildcardSubtype()) {
				contentTypeToUse = getDefaultContentType(t);
			}
			else if (MediaType.APPLICATION_OCTET_STREAM.equals(contentType)) {
				MediaType mediaType = getDefaultContentType(t);
				contentTypeToUse = (mediaType != null ? mediaType : contentTypeToUse);
			}
			if (contentTypeToUse != null) {
				if (contentTypeToUse.getCharset() == null) {
					Charset defaultCharset = getDefaultCharset();
					if (defaultCharset != null) {
						contentTypeToUse = new MediaType(contentTypeToUse, defaultCharset);
					}
				}
				headers.setContentType(contentTypeToUse);
			}
		}
		if (headers.getContentLength() < 0 && !headers.containsKey(HttpHeaders.TRANSFER_ENCODING)) {
			Long contentLength = getContentLength(t, headers.getContentType());
			if (contentLength != null) {
				headers.setContentLength(contentLength);
			}
		}
	}
複製代碼

2.@ResponseBody該註解用於將Controller中方法的返回對象,根據HttpRequest中請求頭中Accept的內容,再經過合適的HttpMessageConverter轉換指定格式後,寫入到response對象(HttpOutputMessage)的body數據區中。若指定方法中consume爲「application/json」,那麼方法僅處理請求頭中ContentType屬性值爲"application/json"的請求。

image.png

3.判斷某個方法是否有指定的註解、某個方法所在的類上是否有指定的註解、某個方法的參數上是否有指定的註解。

parameter.hasParameterAnnotation(RequestBody.class)
AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ResponseBody.class)
returnType.hasMethodAnnotation(ResponseBody.class)
複製代碼

4.@ModelAttribute的妙用

1.運用在方法的參數上,會將客戶端傳遞過來的參數按名稱注入到指定對象中,而且會將這個對象自動加入到modelMap中,便於view層調用

2.運用在方法中,會在每個@RequestMapping標註的方法前執行,若是有返回值,則自動將該返回值加入modelMap中。我通常用於封裝BaseController

public abstract class BaseController {

    protected HttpServletRequest request;
    protected HttpServletResponse response;
    protected HttpSession session;
    protected ModelMap modelMap;

    @ModelAttribute
    protected void initSpringMvc(HttpServletRequest request, HttpServletResponse response,
                                         HttpSession session, ModelMap modelMap) {
        this.request = request;
        this.response = response;
        this.session = session;
        this.modelMap = modelMap;
    }
}
複製代碼

5.定時任務,咱們在WebApplication類註解@EnableScheduling,開啓定時任務。cron表達式的參數從左到右分別是秒 、分、 時、 天、 月、 星期、 年。詳細的cron表達式用法請看這個網站http://cron.qqe2.com/

@Component
public class TestTask {

    @Scheduled(cron = "4-40 * * * * ?")
    public void reportCurrentTime() {
        System.out.println("如今時間:" + DateFormatUtils.format(new Date(),
                "yyyy-MM-dd HH:mm:ss"));
    }
}
複製代碼

6.開啓異步任務,咱們在WebApplication類註解@EnableAsync。

咱們能夠寫一個AsyncTask任務類

@Component
public class AsyncTask {

    @Async
    public Future<Boolean> doTask1() throws Exception {
        long start = System.currentTimeMillis();
        Thread.sleep(1000);
        long end = System.currentTimeMillis();
        System.out.println("任務1耗時:" + (end - start));
        return new AsyncResult<>((true));
    }

    @Async
    public Future<Boolean> doTask2() throws Exception {
        long start = System.currentTimeMillis();
        Thread.sleep(2000);
        long end = System.currentTimeMillis();
        System.out.println("任務2耗時:" + (end - start));
        return new AsyncResult<>((true));
    }

    @Async
    public Future<Boolean> doTask3() throws Exception {
        long start = System.currentTimeMillis();
        Thread.sleep(3000);
        long end = System.currentTimeMillis();
        System.out.println("任務3耗時:" + (end - start));
        return new AsyncResult<>((true));
    }
}
複製代碼

而後在寫TaskController

@RestController
@RequestMapping("/tasks")
public class TaskController extends BaseController {

    @Autowired
    private AsyncTask asyncTask;

    @RequestMapping("test")
    public Result test() throws Exception {
        long start = System.currentTimeMillis();

        Future<Boolean> a = asyncTask.doTask1();
        Future<Boolean> b = asyncTask.doTask2();
        Future<Boolean> c = asyncTask.doTask3();

        while (!a.isDone() || !b.isDone() || !c.isDone()) {
            if (a.isDone() && b.isDone() && c.isDone()) {
                break;
            }
        }
        long end = System.currentTimeMillis();
        String times = "任務所有完成,總耗時:" + (end - start) + "毫秒";
        return Result.success(times);
    }
}
複製代碼

咱們能夠看到這3個任務總耗時是3000ms,證實任務是異步執行的。若是去掉@Async,這3個任務執行是同步的,總耗時應該是6000多ms。

{"code":0,"data":"任務所有完成,總耗時:3005毫秒","msg":""}
複製代碼

7.SpringBoot部署到外部Tomcat,配置pom文件。使tomcat做用域設置爲provided,provided代表只在編譯器和測試時候使用,由於咱們部署到外部Tomcat,運行期間有外部Tomcat的支持。

<!--spring boot tomcat
		默承認以不用配置,但當須要把當前web應用佈置到外部servlet容器時就須要配置,
		並將scope配置爲provided
		當須要默認的jar啓動,則去掉provided, provided代表只在編譯期和測試的時候使用-->
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-tomcat</artifactId>
			<scope>provided</scope>
		</dependency>
複製代碼

記得把打包的方式從jar改爲war

<groupId>com.cmazxiaoma</groupId>
	<artifactId>seckillSystem</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<packaging>war</packaging>
複製代碼

重寫SpringApplication這個啓動類,我這裏從新建立了一個類,名爲WebApplication

@SpringBootApplication
//開啓定時任務
@EnableScheduling
//開啓異步調用方法
@EnableAsync
public class WebApplication extends SpringBootServletInitializer {

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) {
        return builder.sources(WebApplication.class);
    }
}
複製代碼

而後Build Artifacts便可。

image.png


OSI

OSI是開放式系統互聯,英文是Open System Interconnection

應用層 表示層 會話層 傳輸層 網絡層 數據鏈路層 物理層

TCP/IP模型

應用層 =》 HTTP(超文本傳輸協議)、TFTP(簡單文件傳輸協議)、SMTP(簡單郵件傳輸協議)、DNS(域名系統)、SNMP(簡單網絡管理協議)、NFS(網絡文件系統)、Telnet(終端登陸) 傳輸層 =》 TCP、IP 網絡層 =》 IP、ICMP(國際控制報文協議)、ARP(地址解析協議)、RARP(反地址解析協議) 數據鏈路層 =》 PPP(點對點協議)


HttpMessageConverter所引起的異常

當我去請求/login/to_login會返回login視圖,login界面會去加載背景圖片。此時咱們沒有去配置資源映射,致使背景圖片會請求後端的Controller。若是沒有找到合適的Controller去處理這個請求,會進入全局異常捕獲器進入異常處理。在RequestResponseBodyMethodProcessor中的writeWithMessageConverters()方法中,咱們會調用getProducibleMediaTypes()方法獲取該請求的全部返回消息格式類型。

HttpServletRequest request = inputMessage.getServletRequest();
		List<MediaType> requestedMediaTypes = getAcceptableMediaTypes(request);
		List<MediaType> producibleMediaTypes = getProducibleMediaTypes(request, valueType, declaredType);

		if (outputValue != null && producibleMediaTypes.isEmpty()) {
			throw new IllegalArgumentException("No converter found for return value of type: " + valueType);
		}

複製代碼

因爲咱們沒有在全局異常捕獲器HandlerMapping中顯式設置produces屬性,咱們只能經過遍歷全部的HttpMessageConverter,經過canWrite()方法找到支持解析Java對象的HttpMessageConverter,而且把其所支持的mediaType加入mediaTypes集合裏面。

protected List<MediaType> getProducibleMediaTypes(HttpServletRequest request, Class<?> valueClass, Type declaredType) {
		Set<MediaType> mediaTypes = (Set<MediaType>) request.getAttribute(HandlerMapping.PRODUCIBLE_MEDIA_TYPES_ATTRIBUTE);
		if (!CollectionUtils.isEmpty(mediaTypes)) {
			return new ArrayList<MediaType>(mediaTypes);
		}
		else if (!this.allSupportedMediaTypes.isEmpty()) {
			List<MediaType> result = new ArrayList<MediaType>();
			for (HttpMessageConverter<?> converter : this.messageConverters) {
				if (converter instanceof GenericHttpMessageConverter && declaredType != null) {
					if (((GenericHttpMessageConverter<?>) converter).canWrite(declaredType, valueClass, null)) {
						result.addAll(converter.getSupportedMediaTypes());
					}
				}
				else if (converter.canWrite(valueClass, null)) {
					result.addAll(converter.getSupportedMediaTypes());
				}
			}
			return result;
		}
		else {
			return Collections.singletonList(MediaType.ALL);
		}
	}
複製代碼

咱們得出producibleMediaTypes都是關於"application/json"的格式,咱們for循環2次,將requestedMediaTypes和producibleMediaTypes一一比較,得出兼容的compatibleMediaTypes。若是請求消息格式和返回消息格式沒有一個匹配的話,則拋出HttpMediaTypeNotAcceptableException異常。

image.png

Set<MediaType> compatibleMediaTypes = new LinkedHashSet<MediaType>();
		for (MediaType requestedType : requestedMediaTypes) {
			for (MediaType producibleType : producibleMediaTypes) {
				if (requestedType.isCompatibleWith(producibleType)) {
					compatibleMediaTypes.add(getMostSpecificMediaType(requestedType, producibleType));
				}
			}
		}
		if (compatibleMediaTypes.isEmpty()) {
			if (outputValue != null) {
				throw new HttpMediaTypeNotAcceptableException(producibleMediaTypes);
			}
			return;
		}
複製代碼
解決辦法

在application-dev.properties文件中配置靜態資源自動映射

spring.resources.add-mappings=true
複製代碼

或者是手動配置資源映射

@Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("/static/**").addResourceLocations("classpath:/static/");
        super.addResourceHandlers(registry);
    }
複製代碼

Java基礎知識

PreparedStatement對象有addBatch()、executeBatch()方法,用於批量插入。

Connection conn = DBUtil.getConn();
        String sql = "insert into miaosha_user(login_count, nickname, register_date, salt, password, id)values(?,?,?,?,?,?)";
        PreparedStatement pstmt = conn.prepareStatement(sql);

        for (int i = 0; i < users.size(); i++) {
            MiaoshaUser user = users.get(i);
            pstmt.setInt(1, user.getLoginCount());
            pstmt.setString(2, user.getNickname());
            pstmt.setTimestamp(3, new Timestamp(user.getRegisterDate().getTime()));
            pstmt.setString(4, user.getSalt());
            pstmt.setString(5, user.getPassword());
            pstmt.setLong(6, user.getId());
            pstmt.addBatch();
        }
        pstmt.executeBatch();
        pstmt.close();
        conn.close();
複製代碼

isAssignableFrom()的用法,判斷Class1和Class2是否相同,判斷Class1是不是Class2的接口或者是其父類。

Class1.isAssignableFrom(Class2)
複製代碼

instance of 容易和isAssignableFrom()混淆,這用cmazxiaoma instance of Object舉例子,判斷一個對象實例是不是一個類、接口的實例,或者是其父類、子接口的實例

###JSR303用法 JSR303是一個數據驗證的規範,這裏用手機號驗證舉例子

定義@IsMobile註解,這個註解要被IsMobileValidator類去實現驗證。

@Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Constraint(validatedBy = {IsMobileValidator.class})
public @interface IsMobile {

    boolean required() default true;

    String message() default "手機號碼格式錯誤";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};

}
複製代碼

定義手機號驗證類,驗證沒經過會拋出BindException

public class IsMobileValidator implements ConstraintValidator<IsMobile, String> {

    private boolean required = false;

    @Override
    public void initialize(IsMobile isMobile) {
        required = isMobile.required();
    }

    @Override
    public boolean isValid(String value, ConstraintValidatorContext constraintValidatorContext) {
        if (required) {
            return ValidatorUtil.isMobile(value);
        } else {
            if (StringUtils.isEmpty(value)) {
                return true;
            } else {
                return ValidatorUtil.isMobile(value);
            }
        }
    }
}
複製代碼

驗證沒經過會拋出BindException,咱們在全局異常捕獲器中捕獲這個異常。

@ControllerAdvice
@ResponseBody
@Slf4j
public class GlobalExceptionHandler {

    @ExceptionHandler(value = Exception.class)
    public Result<String> exceptionHandler(HttpServletRequest request, HttpServletResponse response,
                                           Object handler, Exception e) {
        log.error(e.getMessage());

        if (e instanceof GlobalException) {
            GlobalException ex = (GlobalException) e;
            return Result.error(ex.getCm());
        } else if (e instanceof BindException) {
            BindException ex = (BindException) e;
            List<ObjectError> errors = ex.getAllErrors();
            ObjectError error = errors.get(0);
            String msg = error.getDefaultMessage();
            return Result.error(CodeMsg.BIND_ERROR.fillArgs(msg));
        } else {
            return Result.error(CodeMsg.SERVER_ERROR.fillArgs(e.getMessage()));
        }
    }
}
複製代碼

尾言

每次逛博客的時候,看到不懂的地方,必定要拿小本本記住。而後整理到簡書上面,日積月累,量變引起質變。

相關文章
相關標籤/搜索