Browse Source
* commit 'fb7c78b89f87c954cb1d14e7ba5668972a4b2962': REPORT-15314 103模板信息收集=>调整代码 REPORT-15314 103模板信息收集=>考虑放假的情况 REPORT-15314 103模板信息收集=>精确度提升research/10.0
plough
6 years ago
9 changed files with 427 additions and 52 deletions
@ -0,0 +1,117 @@ |
|||||||
|
package com.fr.design.mainframe.template.info; |
||||||
|
|
||||||
|
import com.fr.general.ComparatorUtils; |
||||||
|
import com.fr.log.FineLoggerFactory; |
||||||
|
import com.fr.stable.StringUtils; |
||||||
|
import com.fr.stable.xml.XMLPrintWriter; |
||||||
|
import com.fr.stable.xml.XMLReadable; |
||||||
|
import com.fr.stable.xml.XMLWriter; |
||||||
|
import com.fr.stable.xml.XMLableReader; |
||||||
|
|
||||||
|
import java.text.ParseException; |
||||||
|
import java.text.SimpleDateFormat; |
||||||
|
import java.util.Calendar; |
||||||
|
import java.util.Date; |
||||||
|
import java.util.concurrent.TimeUnit; |
||||||
|
|
||||||
|
/** |
||||||
|
* 管理打开设计器的日期记录 |
||||||
|
* Created by plough on 2019/4/19. |
||||||
|
*/ |
||||||
|
class DesignerOpenHistory implements XMLReadable, XMLWriter { |
||||||
|
static final String XML_TAG = "DesignerOpenHistory"; |
||||||
|
private static DesignerOpenHistory singleton; |
||||||
|
private static final String SPLITTER = ","; |
||||||
|
private static final int LENGTH = 3; // 保留最近 3 次的记录
|
||||||
|
// 最近的日期是 history[0],最早的日期是 history[LENGTH-1]
|
||||||
|
private String[] history = new String[LENGTH]; |
||||||
|
|
||||||
|
private DesignerOpenHistory() { |
||||||
|
for (int i = 0; i < LENGTH; i++) { |
||||||
|
history[i] = StringUtils.EMPTY; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
static DesignerOpenHistory getInstance() { |
||||||
|
if (singleton == null) { |
||||||
|
singleton = new DesignerOpenHistory(); |
||||||
|
} |
||||||
|
return singleton; |
||||||
|
} |
||||||
|
|
||||||
|
void update() { |
||||||
|
String today = getToday(); |
||||||
|
if (ComparatorUtils.equals(history[0], today)) { |
||||||
|
return; |
||||||
|
} |
||||||
|
shiftByOne(); |
||||||
|
history[0] = today; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获取历史记录中囊括的日子数。即最早的历史记录 history[LENGTH - 1],到最晚的记录 history[0] 之间的时间跨度 |
||||||
|
*/ |
||||||
|
int getHistorySpanDayCount() { |
||||||
|
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); |
||||||
|
try { |
||||||
|
Date earliestDate = sdf.parse(getEarliestDate()); |
||||||
|
Date latestDate = sdf.parse(history[0]); |
||||||
|
long diffInMillies = latestDate.getTime() - earliestDate.getTime(); |
||||||
|
return (int)TimeUnit.DAYS.convert(diffInMillies,TimeUnit.MILLISECONDS) + 1; |
||||||
|
} catch (ParseException e) { |
||||||
|
FineLoggerFactory.getLogger().error(e.getMessage(), e); |
||||||
|
} |
||||||
|
return 1; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 今天是否已经打开过设计器 |
||||||
|
*/ |
||||||
|
boolean hasOpenedToday() { |
||||||
|
String today = getToday(); |
||||||
|
return ComparatorUtils.equals(today, history[0]); |
||||||
|
} |
||||||
|
|
||||||
|
private String getEarliestDate() { |
||||||
|
for (int i = LENGTH - 1; i >= 0; i--) { |
||||||
|
if (StringUtils.isNotEmpty(history[i])) { |
||||||
|
return history[i]; |
||||||
|
} |
||||||
|
} |
||||||
|
throw new AssertionError("Designer open history is empty!"); |
||||||
|
} |
||||||
|
|
||||||
|
private void shiftByOne() { |
||||||
|
for (int i = LENGTH - 1; i > 0; i--) { |
||||||
|
history[i] = history[i-1]; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private String getToday() { |
||||||
|
return new SimpleDateFormat("yyyy-MM-dd").format(Calendar.getInstance().getTime()); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public String toString() { |
||||||
|
return StringUtils.join(SPLITTER, history); |
||||||
|
} |
||||||
|
|
||||||
|
private void parseString(String s) { |
||||||
|
String[] arr = s.split(SPLITTER); |
||||||
|
System.arraycopy(arr, 0, history, 0, arr.length); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void readXML(XMLableReader reader) { |
||||||
|
if (XML_TAG.equals(reader.getTagName())) { |
||||||
|
parseString(reader.getElementValue()); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void writeXML(XMLPrintWriter writer) { |
||||||
|
writer.startTAG(XML_TAG); |
||||||
|
writer.textNode(toString()); |
||||||
|
writer.end(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,69 @@ |
|||||||
|
package com.fr.design.mainframe.template.info; |
||||||
|
|
||||||
|
/** |
||||||
|
* Created by plough on 2019/4/19. |
||||||
|
*/ |
||||||
|
public class TimeConsumeTimer { |
||||||
|
private static final int ONE_THOUSAND = 1000; |
||||||
|
private enum State { |
||||||
|
RUNNING, STOPPED |
||||||
|
} |
||||||
|
private int timeConsume; // 单位 s
|
||||||
|
private long startMS; // 单位 ms
|
||||||
|
private long stopMS; |
||||||
|
private State state; |
||||||
|
private boolean enabled; |
||||||
|
|
||||||
|
public TimeConsumeTimer() { |
||||||
|
reset(); |
||||||
|
} |
||||||
|
|
||||||
|
public boolean isEnabled() { |
||||||
|
return enabled; |
||||||
|
} |
||||||
|
|
||||||
|
public void setEnabled(boolean enabled) { |
||||||
|
this.enabled = enabled; |
||||||
|
} |
||||||
|
|
||||||
|
public void start() { |
||||||
|
if (!isEnabled() || isRunning()) { |
||||||
|
return; |
||||||
|
} |
||||||
|
startMS = System.currentTimeMillis(); |
||||||
|
state = State.RUNNING; |
||||||
|
} |
||||||
|
|
||||||
|
public void stop() { |
||||||
|
if (!isEnabled() || !isRunning()) { |
||||||
|
return; |
||||||
|
} |
||||||
|
stopMS = System.currentTimeMillis(); |
||||||
|
|
||||||
|
timeConsume += ((stopMS - startMS) / ONE_THOUSAND); |
||||||
|
startMS = 0; |
||||||
|
stopMS = 0; |
||||||
|
state = State.STOPPED; |
||||||
|
} |
||||||
|
|
||||||
|
public int popTime() { |
||||||
|
if (!isEnabled()) { |
||||||
|
return 0; |
||||||
|
} |
||||||
|
stop(); |
||||||
|
int result = timeConsume; |
||||||
|
reset(); |
||||||
|
return result; |
||||||
|
} |
||||||
|
|
||||||
|
private boolean isRunning() { |
||||||
|
return state == State.RUNNING; |
||||||
|
} |
||||||
|
|
||||||
|
private void reset() { |
||||||
|
timeConsume = 0; |
||||||
|
startMS = 0; |
||||||
|
stopMS = 0; |
||||||
|
state = State.STOPPED; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,108 @@ |
|||||||
|
package com.fr.design.mainframe.template.info; |
||||||
|
|
||||||
|
import com.fr.invoke.Reflect; |
||||||
|
import com.fr.stable.StringUtils; |
||||||
|
import com.fr.stable.xml.XMLPrintWriter; |
||||||
|
import com.fr.stable.xml.XMLableReader; |
||||||
|
import com.fr.third.javax.xml.stream.XMLStreamException; |
||||||
|
import org.junit.Before; |
||||||
|
import org.junit.Test; |
||||||
|
|
||||||
|
import java.io.PrintWriter; |
||||||
|
import java.io.StringReader; |
||||||
|
import java.io.StringWriter; |
||||||
|
import java.text.SimpleDateFormat; |
||||||
|
import java.util.Calendar; |
||||||
|
|
||||||
|
import static org.junit.Assert.assertArrayEquals; |
||||||
|
import static org.junit.Assert.assertEquals; |
||||||
|
import static org.junit.Assert.assertFalse; |
||||||
|
import static org.junit.Assert.assertTrue; |
||||||
|
import static org.junit.Assert.fail; |
||||||
|
|
||||||
|
/** |
||||||
|
* Created by plough on 2019/4/21. |
||||||
|
*/ |
||||||
|
public class DesignerOpenHistoryTest { |
||||||
|
private DesignerOpenHistory openHistory; |
||||||
|
private String[] mockHistory = new String[] { |
||||||
|
"2019-04-08", "2019-04-03", "2019-03-29" |
||||||
|
}; |
||||||
|
|
||||||
|
@Before |
||||||
|
public void setUp() { |
||||||
|
openHistory = DesignerOpenHistory.getInstance(); |
||||||
|
Reflect.on(openHistory).set("history", mockHistory); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testReadAndWrite() throws XMLStreamException { |
||||||
|
// 写入 xml
|
||||||
|
StringWriter sw = new StringWriter(); |
||||||
|
XMLPrintWriter writer = XMLPrintWriter.create(new PrintWriter(sw)); |
||||||
|
openHistory.writeXML(writer); |
||||||
|
writer.flush(); |
||||||
|
writer.close(); |
||||||
|
|
||||||
|
String xmlText = sw.getBuffer().toString(); |
||||||
|
|
||||||
|
// 临时修改配置
|
||||||
|
Reflect.on(openHistory).set("history", new String[] {"", "", ""}); |
||||||
|
|
||||||
|
// 从 xml 中读取
|
||||||
|
StringReader sr = new StringReader(xmlText); |
||||||
|
XMLableReader xmlReader = XMLableReader.createXMLableReader(sr); |
||||||
|
xmlReader.readXMLObject(openHistory); |
||||||
|
|
||||||
|
// 验证:与写入时的配置一致
|
||||||
|
assertArrayEquals(mockHistory, (String[])Reflect.on(openHistory).field("history").get()); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testToString() { |
||||||
|
assertEquals("2019-04-08,2019-04-03,2019-03-29", openHistory.toString()); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testParseString() { |
||||||
|
String[] mockDates = {"2020-04-08", "2019-04-03", "2016-03-29"}; |
||||||
|
Reflect.on(openHistory).call("parseString", StringUtils.join(",", mockDates)); |
||||||
|
assertArrayEquals(mockDates, (String[])Reflect.on(openHistory).field("history").get()); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testGetHistorySpanDayCount() { |
||||||
|
assertEquals(11, openHistory.getHistorySpanDayCount()); |
||||||
|
|
||||||
|
Reflect.on(openHistory).set("history", new String[] {"2019-05-03", "2019-05-02", ""}); |
||||||
|
assertEquals(2, openHistory.getHistorySpanDayCount()); |
||||||
|
|
||||||
|
Reflect.on(openHistory).set("history", new String[] {"2019-05-03", "", ""}); |
||||||
|
assertEquals(1, openHistory.getHistorySpanDayCount()); |
||||||
|
|
||||||
|
try { |
||||||
|
Reflect.on(openHistory).set("history", new String[] {"", "", ""}); |
||||||
|
fail("should not be here"); |
||||||
|
} catch (AssertionError ignore) { |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testHasOpenedToday() { |
||||||
|
assertFalse(openHistory.hasOpenedToday()); |
||||||
|
|
||||||
|
Reflect.on(openHistory).set("history", new String[] {getToday(), "2019-02-02", ""}); |
||||||
|
assertTrue(openHistory.hasOpenedToday()); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testUpdate() { |
||||||
|
openHistory.update(); |
||||||
|
String[] arr = { getToday(), "2019-04-08", "2019-04-03" }; |
||||||
|
assertArrayEquals(arr, (String[])Reflect.on(openHistory).field("history").get()); |
||||||
|
} |
||||||
|
|
||||||
|
private String getToday() { |
||||||
|
return new SimpleDateFormat("yyyy-MM-dd").format(Calendar.getInstance().getTime()); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,62 @@ |
|||||||
|
package com.fr.design.mainframe.template.info; |
||||||
|
|
||||||
|
import org.junit.Test; |
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals; |
||||||
|
|
||||||
|
/** |
||||||
|
* Created by plough on 2019/4/19. |
||||||
|
*/ |
||||||
|
public class TimeConsumeTimerTest { |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testNotEnabled() throws InterruptedException { |
||||||
|
TimeConsumeTimer consumeTimer = new TimeConsumeTimer(); |
||||||
|
consumeTimer.start(); |
||||||
|
Thread.sleep(1100); |
||||||
|
consumeTimer.stop(); |
||||||
|
assertEquals(0, consumeTimer.popTime()); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testEnabled() throws InterruptedException { |
||||||
|
TimeConsumeTimer consumeTimer = new TimeConsumeTimer(); |
||||||
|
consumeTimer.setEnabled(true); |
||||||
|
consumeTimer.start(); |
||||||
|
Thread.sleep(1100); |
||||||
|
consumeTimer.stop(); |
||||||
|
assertEquals(1, consumeTimer.popTime()); |
||||||
|
assertEquals(0, consumeTimer.popTime()); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testMultiTimes() throws InterruptedException { |
||||||
|
TimeConsumeTimer consumeTimer = new TimeConsumeTimer(); |
||||||
|
consumeTimer.setEnabled(true); |
||||||
|
|
||||||
|
consumeTimer.start(); |
||||||
|
Thread.sleep(1010); |
||||||
|
consumeTimer.stop(); |
||||||
|
|
||||||
|
Thread.sleep(2000); |
||||||
|
|
||||||
|
consumeTimer.start(); |
||||||
|
Thread.sleep(1010); |
||||||
|
assertEquals(2, consumeTimer.popTime()); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testStartMultiTime() throws InterruptedException { |
||||||
|
TimeConsumeTimer consumeTimer = new TimeConsumeTimer(); |
||||||
|
consumeTimer.setEnabled(true); |
||||||
|
|
||||||
|
consumeTimer.start(); |
||||||
|
Thread.sleep(1010); |
||||||
|
consumeTimer.start(); |
||||||
|
Thread.sleep(1010); |
||||||
|
consumeTimer.start(); |
||||||
|
Thread.sleep(1010); |
||||||
|
|
||||||
|
assertEquals(3, consumeTimer.popTime()); |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue