新聞中心
前言

你的世界多數(shù)情況下充滿了混沌和單調(diào),你的身體雖然不胖但并不會讓你感覺到那么有力量;你的過往乏善可陳,充斥著很多傷心與自我否定,你過往的未來也沒有驚喜在場。你想要一場新生,想要一次脫胎換骨,沒有行動,一切都是空想,依舊忍受痛苦與弱小。一點一滴的積累,才能感受飛輪轉(zhuǎn)起來時的酣暢淋漓,才會有氣勢如虹的力量 ,一點一滴的積累就是需要一個打卡的功能。
打卡在生活中還是應(yīng)用挺多的,比如上班打卡,健身打卡,學(xué)習(xí)打卡...等等,實際上打卡功能開發(fā)是挺容易的,讓我們來一起實現(xiàn)它吧,讓我們自己用著自己開發(fā)的功能吧!
為什么寫一篇打卡的功能性文章吶?
公司一個系統(tǒng)需要實現(xiàn)的一個小小功能,也是為了方便日后進(jìn)行回顧以及優(yōu)化,特此記錄下來,話不多說我們來實現(xiàn)它吧。
數(shù)據(jù)庫設(shè)計
1. 為什么要設(shè)計數(shù)據(jù)庫
- 節(jié)省數(shù)據(jù)的存儲空間
- 保證數(shù)據(jù)的完整性
- 方便根據(jù)數(shù)據(jù)庫進(jìn)行系統(tǒng)的開發(fā)
2. 根據(jù)需求設(shè)計數(shù)據(jù)庫
- 打卡功能
- 日志表(打卡用戶,打卡項目編號,打卡時間) 這個表中的打卡項目這個字段是非必須的,添加這個字段只是為了方便日后的擴展,其他兩個字段都是必須的
- 項目表(項目編號,項目名稱,創(chuàng)建時間) 這個表是非必須的,只是為了方便日后的擴展,可加可不加
- 統(tǒng)計表(項目編號,總打卡數(shù),連續(xù)打卡數(shù),打卡用戶,打卡時間) 這個表中的字段除了項目編號這個字段不是必須的,其他都是必須要的基本字段,
- 我這樣設(shè)計三張表只是為了方便以后的擴展使用,除了項目表,其他兩張表都是最基礎(chǔ)的必須要的
- 首先創(chuàng)建一個數(shù)據(jù)庫 clockin CHARACTER SET:指定數(shù)據(jù)庫采用的字符集,utf8不能寫成utf-8 COLLATE:指定數(shù)據(jù)庫字符集的排序規(guī)則,utf8的默認(rèn)排序規(guī)則為
- utf8_general_ci(通過show character set查看) drop database if EXISTS clockin ; create database clockin CHARACTER SET utf8 COLLATE utf8_general_ci;
- DROP TABLE IF EXISTS `clockin_count`;
- CREATE TABLE `clockin_count` (
- `id` int(11) NOT NULL AUTO_INCREMENT,
- `pid` int(11) DEFAULT NULL COMMENT '打卡項目',
- `sum` int(11) DEFAULT NULL COMMENT '打卡總次數(shù)',
- `cloop` int(11) DEFAULT NULL COMMENT '打卡連續(xù)次數(shù)',
- `name` varchar(25) COLLATE utf8_bin DEFAULT NULL COMMENT '打卡人',
- `dtime` datetime DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP,
- PRIMARY KEY (`id`)
- ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
- DROP TABLE IF EXISTS `clockin_log`;
- CREATE TABLE `clockin_log` (
- `id` int(11) NOT NULL AUTO_INCREMENT,
- `name` varchar(25) COLLATE utf8_bin DEFAULT NULL COMMENT '打卡人',
- `pid` int(11) DEFAULT NULL COMMENT '打卡項目',
- `dtime` datetime DEFAULT NULL COMMENT '打卡時間',
- PRIMARY KEY (`id`)
- ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
- DROP TABLE IF EXISTS `clockin_project`;
- CREATE TABLE `clockin_project` (
- `id` int(11) NOT NULL AUTO_INCREMENT,
- `pid` int(11) NOT NULL,
- `project` varchar(25) COLLATE utf8_bin NOT NULL,
- `dtime` datetime DEFAULT NULL,
- PRIMARY KEY (`id`)
- ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
3. 理解清楚數(shù)據(jù)表之間的映射關(guān)系
- clockin_log : 日志表,存儲用戶的每天打卡日志信息,表的數(shù)據(jù)更新率比較高,特此提出來,作為單獨的表使用
- clockin_project:項目表,在什么項目上的打卡,不是非必要的,只是為了方便日后的擴展使用,而添加的
- clockin_count:統(tǒng)計表,統(tǒng)計日志表中用戶的打卡信息,數(shù)據(jù)相對比較少,提取出來更加直觀,查詢數(shù)據(jù)用的比較頻繁
實現(xiàn)思路
首先進(jìn)行判斷項目是否存在,存在才能進(jìn)行打卡,接著判斷今天是否已打卡,如果今天沒打卡則插入數(shù)據(jù),然后判斷昨天是否打卡,
昨天如果打卡了則連續(xù)天數(shù),總天數(shù)都加一,若昨天沒打卡,則連續(xù)天數(shù)設(shè)為一,總天數(shù)加一,思維導(dǎo)圖如下:
這里主要就是sql語句的編寫,我這里主要用到了sql中的 LEFT語法 ,對應(yīng)了oracle中的substr(),主要是用來判斷今天昨天是否打卡
語法: ==LEFT (ARG,LENGTH)== ARG源數(shù)據(jù),LENGTH個字符串,ARG可以是CHAR或BINARY STRING
例如:left('12345',2) --> 12
- SELECT * FROM clockin_log where name = #{name} and pid = #{pid}
- and LEFT(dtime,10) = LEFT(NOW(),10)
- SELECT * FROM clockin_log where name = #{name} and pid = #{pid}
- and LEFT(dtime,10)=LEFT(#{dtime},10)
- 既由 2020-10-10T10:30:51 得到 2020-10-10
主要代碼
這里我用到了 MybatisPlus,具體使用可以參照
這里的Result是我定義了一個結(jié)果集包含 code —— 狀態(tài)碼,msg —— 返回消息 ,data —— 數(shù)據(jù)信息
- // name 用戶名 pid 項目編號
- public Result clock(String name,String pid){
- // 返回的數(shù)據(jù)類型
- Result
- // 得到昨天的日期
- Calendar cal= Calendar.getInstance();
- cal.add(Calendar.DATE,-1);
- Date yesterday=cal.getTime();
- // 判斷今天是否打卡
- Boolean isexit = clockinLogService.
- selectClockinIsexit(name, Integer.parseInt(pid));
- LambdaQueryWrapper
wrapper = new LambdaQueryWrapper<>(); - wrapper.eq(ClockinCount::getName,name).eq(ClockinCount::getPid,Integer.parseInt(pid));
- ClockinCount one = clockinCountService.getOne(wrapper);
- LambdaQueryWrapper
wrapper1 = new LambdaQueryWrapper<>(); - wrapper1.eq(ClockinProject::getPid,Integer.parseInt(pid));
- ClockinProject project = clockinProjectService.getOne(wrapper1);
- //打卡項目是否存在
- if (!StringUtils.isEmpty(project)){
- if (isexit){// 判斷今天是否打卡
- result.setRetCode(Result.ERROR);
- result.setRetMsg("今日已打卡");
- result.setRowData(one);
- return result;
- }else {
- // 記錄打卡信息
- ClockinLog clockinLog = new ClockinLog();
- clockinLog.setName(name).setPid(Integer.parseInt(pid)).setDtime(LocalDateTime.now());
- boolean save = clockinLogService.save(clockinLog);
- if(save){//是否插入數(shù)據(jù)成功
- Boolean yesterdayIsexit = clockinLogService.
- selectClockinYesterdayIsexit(name, Integer.parseInt(pid), yesterday);
- System.err.println("yesterdayIsexit->"+yesterdayIsexit);
- if (yesterdayIsexit){ //判斷昨天有沒有打卡
- System.err.println("昨天打卡了");
- // 更新打卡次數(shù)
- ClockinCount cc = new ClockinCount();
- LambdaUpdateWrapper
updateWrapper = new LambdaUpdateWrapper<>(); - updateWrapper.eq(ClockinCount::getPid,Integer.parseInt(pid)).eq(ClockinCount::getName, name);
- cc.setSum(one.getSum()+1).setCloop(one.getCloop()+1);
- clockinCountService.update(cc,updateWrapper);
- }else {
- if (!StringUtils.isEmpty(one)){ // 統(tǒng)計表中用戶存在否
- // 更新打卡次數(shù)
- ClockinCount cc = new ClockinCount();
- LambdaUpdateWrapper
updateWrapper = new LambdaUpdateWrapper<>(); - updateWrapper.eq(ClockinCount::getPid,Integer.parseInt(pid)).eq(ClockinCount::getName, name);
- cc.setSum(one.getSum()+1).setCloop(1);
- clockinCountService.update(cc,updateWrapper);
- }else {
- // 插入數(shù)據(jù)信息
- ClockinCount cc = new ClockinCount();
- cc.setCloop(1).setName(name).setPid(Integer.parseInt(pid)).setSum(1).setDtime(LocalDateTime.now());
- clockinCountService.save(cc);
- }
- }
- one = clockinCountService.getOne(wrapper);
- result.setRetCode(Result.SUCCESS);
- result.setRetMsg("成功");
- result.setRowData(one);
- return result;
- }
- }
- }else {
- result.setRetCode(Result.ERROR);
- result.setRetMsg("失敗");
- result.setRowData("打卡項目不存在");
- }
- return null;
- }
總結(jié)
本文主要介紹了用Java寫一個打卡功能的主要過程,打卡主要是記錄了一個人階段性的努力,如果自己的自我規(guī)劃學(xué)習(xí)能力不高,自我督促能力不強,那么這個打卡功能就可以完美的解決督促自己進(jìn)行學(xué)習(xí)以及其他事情。以上就是本篇文章的主要代碼以及思路。
本文授權(quán)轉(zhuǎn)載自公眾號「良許Linux」。良許,世界500強外企Linux開發(fā)工程師,公眾號里分享大量Linux干貨,歡迎關(guān)注!
當(dāng)前文章:來,你們想要的打卡功能
鏈接地址:http://m.5511xx.com/article/dhphodj.html


咨詢
建站咨詢
