每天一剂开发良药
主要介紹一些小技巧之類,是為備忘也。
TypeScript 導(dǎo)入 *.vue 報(bào)錯(cuò):Cannot find module
如圖
創(chuàng)建一個(gè) shims.d.ts 文件,放置到 src/globalDeclare 中。
typescript-eslint 自作多情提示 xxx is assigned a value but never used
eslintrc.js 加上
"no-unused-vars": "off","@typescript-eslint/no-unused-vars": ["error"],"@typescript-eslint/ban-ts-comment": "off","@typescript-eslint/explicit-function-return-type": "off","@typescript-eslint/no-explicit-any": ["off"]ViewUI Table 單元格 文本將不換行,超出部分顯示為省略號(hào)
組件寫法,比較麻煩
<Table :columns="columns1" :data="list"><template slot-scope="{ row, index }" slot="action"><a href="javascript:void(0);" @click="handleEdit(row, index)">編輯</a> <Divider type="vertical" /><Poptip confirm transfer title="是否要?jiǎng)h除此行?" @on-ok="handleDelete(index)"><a href="javascript:void(0);" style="color:red;">刪除</a></Poptip></template><template slot-scope="{ row }" slot="url"><Ellipsis :text="row.url" :length="50" tooltip :transfer="true"></Ellipsis></template> </Table>其實(shí)可以在列配置中聲明:
{ title: '鏈接地址', minWidth: 190, key: 'url', ellipsis:true, tooltip:true },另外每一列設(shè)置 width/minWidth 就可以保證不受瀏覽器寬度擠壓
Vue 工程里面怎么引入公共的 Less 樣式庫(kù)?
例如 Less 的函數(shù)。
安裝下面插件
- “l(fā)ess”: “^3.0.4”,
- “l(fā)ess-loader”: “^5.0.0”,
- “style-resources-loader”: “^1.4.1”
打開 vue.config.js 配置文件,
module.exports = {pluginOptions: {'style-resources-loader': {preProcessor: 'less',patterns: ['C:\\code\\ajaxjs\\aj-js\\aj-ui\\src\\style\\common-functions.less']}},lintOnSave: true,devServer: {overlay: {warnings: true,error: true}} };路徑寫死,改相對(duì)路徑
var path = require("path");module.exports = {pluginOptions: {'style-resources-loader': {preProcessor: 'less',patterns: [path.resolve(__dirname, './src/style/common-functions.less')]}},lintOnSave: true,devServer: {overlay: {warnings: true,error: true}} };MySQL varchar 文本包含數(shù)字的計(jì)數(shù)器
需求:如果重復(fù)值,則自增 1、2、3……
思路:先查詢是否重復(fù):
如果是,獲取最大值 MaxId,通過(guò)正則查詢、排序,注意參數(shù)拼接了字符串(參數(shù)就是重復(fù)值)。
SELECT urlDir FROM ${tableName} WHERE urlDir REGEXP CONCAT(?, '_[0-9]+$') AND datasourceId = ? ORDER BY urlDir DESC LIMIT 1若無(wú)則 1,有則 MaxId++。
快速 SQL 轉(zhuǎn)換 Java Bean/ POJO
找過(guò)好幾個(gè)的,都不太符合需求,于是自己寫個(gè)腳本,也很快。
<html><head><meta charset="utf-8" /><title>SQL2pojo</title> </head><body><textarea id="sql" rows="20" cols="100"></textarea><br /><br /><button onclick="sql2pojo()">SQL2pojo</button><pre></pre> </body> <script>let tpl = '';function sql2pojo() {let sql = document.querySelector("#sql").value;let arr = sql.match(/CREATE TABLE `(?:\w+|_)` \(((\s|\S)+)(?=PRIMARY KEY)/);let result = arr[1].trim();arr = result.split(',');let output = [];arr.forEach(item => {if (item) {item = item.trim();console.log(item);let _arr = item.match(/^`(\w+)`\s+((?:\w|\(|\))+).*COMMENT '(.*?)'/);let _t = _arr[2], type = 'Object';if (_t.indexOf('VARCHAR') != -1 || _t.indexOf('TEXT') != -1)type = 'String';if (_t.indexOf('TINYINT(1)') != -1)type = 'Boolean';else if (_t.indexOf('TINYINT') != -1)type = 'Integer';else if (_t.indexOf('INT') != -1)type = 'Long';if (_t.indexOf('DATETIME') != -1 || _t.indexOf('DATE') != -1)type = 'Date';tpl = ` /*** ${_arr[3]}*/ private ${type} ${_arr[1]};`;// console.log(tpl);output.push(tpl);}});document.querySelector('pre').innerHTML = output.join('<br />');} </script></html>Vue+TS 工程發(fā)布 npm 組件不能攜帶 *.vue 問題
當(dāng)前 Vue 工程既有網(wǎng)站,也希望發(fā)布為 npm 組件。使用 tsc 編譯結(jié)果到 dist 目錄,注意下面問題:
- 配置 main 文件,不然 import 包時(shí)候會(huì) undefined。具體就是在 package.json 配置結(jié)果目錄的 index.js 和 index.d.ts。
- 編譯依靠不能使用 vue-cli-service build,那是編譯網(wǎng)站的。我們目的是打包組件,使用 tsc 即可。其實(shí)就是生成 js 和 map,我的tsconfig.json 配置如下。
其中的 include "src/**/*.vue", 其實(shí)沒作用,因?yàn)?tsc 只管 js/ts/json 的編譯,其他文件它不處理的。那么問題來(lái)了——打包就需要 *.vue 文件,——我搜索了很久終于找到一個(gè)比較簡(jiǎn)單的方法,就是直接復(fù)制過(guò)去。package.json 增加一個(gè) scripts 命令:
"release": "tsc && xcopy src\\components dist\\components /s /y /d && npm publish --access public",tsc 編譯后通過(guò) DOS 命令 xcopy 復(fù)制目錄,/s 表示包含所有子目錄和文件, /y 表示不確認(rèn)并進(jìn)行覆蓋,/d 表示文件日期對(duì)比,日期較新的不覆蓋。
另外,tsc 不會(huì)覆蓋現(xiàn)有文件,所以最好先刪除一下 dist 目錄再 npm run release。
Vue 中單頁(yè)面組件中 render 函數(shù)不運(yùn)行?
要用 render 函數(shù),把 <template> 給去掉。https://segmentfault.com/q/1010000016677825
簡(jiǎn)單使得元素可拖動(dòng)
/*** 使得面板浮動(dòng),可拖放*/ export default function float() {setTimeout(() => {let el: HTMLElement = this.$el;let rect: DOMRect = el.getBoundingClientRect();let controls: HTMLElement = (<HTMLElement>el.querySelector('.controls'));let top: number = rect.top - el.offsetTop;let left: number = rect.left - el.offsetLeft - controls.offsetWidth - 10;let style: CSSStyleDeclaration = controls.style;style.top = top + 'px';style.left = left + 'px';makeDD(controls, <HTMLElement>controls.querySelector('.movable'));let btns: HTMLElement = (<HTMLElement>el.querySelector('.btns'));top = rect.top - el.offsetTop - btns.offsetHeight - 10;left = rect.left - el.offsetLeft;style = btns.style;style.top = top + 'px';style.left = left + 'px';makeDD(btns, <HTMLElement>btns.querySelector('.movable'));}, 10); }/*** 拖放* * @param box 被拖放的區(qū)域* @param dragBar 拖放的按鈕*/ function makeDD(box: HTMLElement, dragBar: HTMLElement): void {// 鼠標(biāo)按下的函數(shù)dragBar.onmousedown = function (oEvent: MouseEvent) {// 求出鼠標(biāo)和box的位置差值let x: number = oEvent.clientX - box.offsetLeft, y: number = oEvent.clientY - box.offsetTop;// 鼠標(biāo)移動(dòng)的函數(shù)// 把事件加在document上,解決因?yàn)槭髽?biāo)移動(dòng)太快時(shí),鼠標(biāo)超過(guò)box后就沒有了拖拽的效果的問題document.onmousemove = function (oEvent: MouseEvent) {// 只能拖動(dòng)窗口標(biāo)題才能移動(dòng)if (oEvent.target != dragBar) {// return;}// 保證拖拽框一直保持在瀏覽器窗口內(nèi)部,不能被拖出的瀏覽器窗口的范圍let l: number = oEvent.clientX - x, t = oEvent.clientY - y;let doc: HTMLElement = document.documentElement;if (l < 0)l = 0;else if (l > doc.clientWidth - box.offsetWidth)l = doc.clientWidth - box.offsetWidth;if (t < 0)t = 0;else if (t > doc.clientHeight - box.offsetHeight)t = doc.clientHeight - box.offsetHeight;box.style.left = l + "px";box.style.top = t + "px";}// 鼠標(biāo)抬起的函數(shù)document.onmouseup = function () {document.onmousemove = document.onmouseup = null;}// 火狐瀏覽器在拖拽空div時(shí)會(huì)出現(xiàn) bug return false阻止默認(rèn)事件,解決火狐的bugreturn false;} }Spring MVC 加入 JSP 支持
/WEB-INF/jsp/ .jsp解決煩人的 sockjs-node/info 跨域問題
打開 build/webpack.conf.js
const config = {resolve: {alias: {}},devServer: {// host: 'localhost',disableHostCheck: true,public: '0.0.0.0'}, };module.exports = config;JSP 頁(yè)面異常
JSP 需要加上下面代碼,運(yùn)行時(shí)才不會(huì)出現(xiàn) java.lang.IllegalStateException: getOutputStream() has already been called …等異常。
/**** @param ctx 頁(yè)面上下文*/ public static void fix(PageContext ctx) {HttpServletResponse response = (HttpServletResponse) ctx.getResponse();try {OutputStream out = response.getOutputStream();out.flush();out.close();response.flushBuffer();ctx.getOut().clear();ctx.pushBody();// out = pageContext.pushBody();} catch (IOException e) {LOGGER.warning(e);} }參考 JSP 內(nèi)置對(duì)象 out 和 response.getWrite() 的區(qū)別
- http://blog.sina.com.cn/s/blog_7217e4320101l8gq.html
- http://www.2cto.com/kf/201109/103284.html
響應(yīng)禁止緩存
/*** 新的輸出,不要緩存** @return 當(dāng)前對(duì)象*/ public MvcOutput noCache() {setHeader("Pragma", "No-cache");setHeader("Cache-Control", "no-cache");setDateHeader("Expires", 0);return this; }返回到前一頁(yè)并刷新
window.location = document.referrer;靜態(tài)的錯(cuò)誤提示頁(yè)
<title>操作錯(cuò)誤</title> <meta charset="utf-8" /> <div style="height: 100%%; display: flex; justify-content: center; align-items: center;"><table><tr><td align="center"> <svg width="150px" viewBox="0 0 1000 1000"><g><path fill="#ea8010" d="M500,10c-46.7,0-84.5,38-84.5,84.9v573.7c0,46.9,37.8,84.9,84.5,84.9c46.7,0,84.5-38,84.5-84.9V94.9C584.5,48,546.7,10,500,10z M500,821c-46.7,0-84.5,37.8-84.5,84.5c0,46.7,37.8,84.5,84.5,84.5c46.7,0,84.5-37.8,84.5-84.5C584.4,858.9,546.6,821,500,821z" /></g></svg></td></tr><tr><td align="center"><br />%s<br /><a href="javascript:history.go(-1);">返回</a></td></tr></table> </div>隨時(shí)隨地獲取 Request/Response
為獲取請(qǐng)求的上下文,能夠在控制器中拿到最常用的對(duì)象,例如 HttpServletRequest 和 HttpServletResponse 等的對(duì)象(甚至 Web App 的啟動(dòng)上下文( 在 web.xml 中配置的參數(shù))),因此還需要設(shè)計(jì)一個(gè) RequestHelper 類,通過(guò) ThreadLocal 讓控制器能輕易地訪問到這些對(duì)象。
一個(gè)容器,向這個(gè)容器存儲(chǔ)的對(duì)象,在當(dāng)前線程范圍內(nèi)都可以取得出來(lái),向 ThreadLocal 里面存東西就是向它里面的 Map 存東西的,然后 ThreadLocal 把這個(gè) Map 掛到當(dāng)前的線程底下,這樣 Map 就只屬于這個(gè)線程了。
private static ThreadLocal<HttpServletRequest> threadLocalRequest = new ThreadLocal<>();private static ThreadLocal<HttpServletResponse> threadLocalResponse = new ThreadLocal<>();/*** 保存一個(gè) request 對(duì)象** @param req 請(qǐng)求對(duì)象*/ public static void setHttpServletRequest(HttpServletRequest req) {threadLocalRequest.set(req); }/*** 獲取請(qǐng)求對(duì)象** @return 請(qǐng)求對(duì)象*/ public static HttpServletRequest getHttpServletRequest() {return threadLocalRequest.get(); }/*** 保存一個(gè) response 對(duì)象** @param resp 響應(yīng)對(duì)象*/ public static void setHttpServletResponse(HttpServletResponse resp) {threadLocalResponse.set(resp); }/*** 獲取上下文中 response 對(duì)象** @return response 響應(yīng)對(duì)象*/ public static HttpServletResponse getHttpServletResponse() {HttpServletResponse resp = threadLocalResponse.get();if (resp == null)throw new RuntimeException("響應(yīng)對(duì)象未初始化");return resp; }/*** 清空 request 和 response*/ public static void clean() {threadLocalRequest.set(null);threadLocalResponse.set(null); }javax.servlet.forward.request_uri 之作用
綁定 JSP 時(shí)候,獲取原請(qǐng)求的 uri,而非模版所在的 uri。
生成模擬數(shù)據(jù)的 SQL
隨機(jī) SET
UPDATE enterprise_tract_meeting SET TYPE = ELT(FLOOR(RAND() * 3 + 1), 1, 2, 3);IDEA 設(shè)置 Javac 編譯參數(shù)對(duì)于 Maven 無(wú)效
Eclipse 不會(huì)那樣,Idea 2018 無(wú)效。你 re-build 然后 deploy 那樣就可以。下面的 pom.xml 永久解決。
<build><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId><version>3.9.0</version><configuration><source>1.8</source><target>1.8</target><compilerArgs><arg>-parameters</arg><!-- IDEA 設(shè)置 Javac 編譯參數(shù)對(duì)于 Maven 無(wú)效 --></compilerArgs></configuration></plugin></plugins> </build>Mysql 創(chuàng)建流水號(hào)
方法一,觸發(fā)器:
CREATE TRIGGER saledetail_id BEFORE INSERT ON saledetail FOR EACH ROW BEGINdeclare n int;select IFNULL(max(right(ItemID,4)),0) into n from saledetail where mid(ItemID,1,8)=DATE_FORMAT(CURDATE(),'%Y%m%d');set NEW.ItemID=concat(DATE_FORMAT(CURDATE(),'%Y%m%d'),right(10001+n,4)); END;注意在插入的時(shí)候主鍵要設(shè)置一個(gè)默認(rèn)值才能插入進(jìn)去,這里我設(shè)置的是空字符串 ""。
方法二:
SELECTsubstr(CONCAT('0000', (IFNULL(MAX(substr(fund_code, -3)),0) + 1)), -3) FROM enterprise_trace_fund WHERE fund_code LIKE 'DF20211223%'fund_code 字段值就是傳過(guò)來(lái)的字符串,mysql 數(shù)據(jù)庫(kù)會(huì)自動(dòng)進(jìn)行匹配,然后自行自增。fund_code 字段值如果加入日期值,三位的流水號(hào)一般是夠用的。
但這做法并不能在并發(fā)下保證流水號(hào)的唯一性。可以用 MySQL 寫鎖(select...for update,也叫 X 鎖,排它鎖)。
方法三:
CREATE DEFINER=`root`@`localhost` PROCEDURE `GetSerialNo`(IN tsCode VARCHAR(50),OUT result VARCHAR(200) ) BEGIN DECLARE tsValue VARCHAR(50); DECLARE tdToday VARCHAR(20); DECLARE nowdate VARCHAR(20); DECLARE tsQZ VARCHAR(50); DECLARE t_error INTEGER DEFAULT 0; DECLARE CONTINUE HANDLER FOR SQLEXCEPTION SET t_error=1; START TRANSACTION; /* UPDATE sys_sno SET sValue=sValue WHERE sCode=tsCode; */SELECT sValue INTO tsValue FROM sys_sno WHERE sCode=tsCode for UPDATE; SELECT sQz INTO tsQZ FROM sys_sno WHERE sCode=tsCode ; -- 因子表中沒有記錄,插入初始值 IF tsValue IS NULL THEN SELECT CONCAT(DATE_FORMAT(NOW(),'%y%m'),'0001') INTO tsValue; UPDATE sys_sno SET sValue=tsValue WHERE sCode=tsCode ; SELECT CONCAT(tsQZ,tsValue) INTO result; ELSE SELECT SUBSTRING(tsValue,1,4) INTO tdToday; SELECT CONVERT(DATE_FORMAT(NOW(),'%y%m'),SIGNED) INTO nowdate;-- 判斷年月是否需要更新IF tdToday = nowdate THEN SET tsValue=CONVERT(tsValue,SIGNED) + 1; ELSE SELECT CONCAT(DATE_FORMAT(NOW(),'%y%m') ,'0001') INTO tsValue ; END IF; UPDATE sys_sno SET sValue =tsValue WHERE sCode=tsCode; SELECT CONCAT(tsQZ,tsValue) INTO result; END IF; IF t_error =1 THEN ROLLBACK; SET result = 'Error'; ELSE COMMIT; END IF; SELECT result ; END;出處:https://www.jianshu.com/p/d7570564f104
Eclipse Maven Dependencies下引入本地工程的 jar 包卻變成源碼文件夾
How to tell Maven to include the jar dependency, not the subproject source directory in Eclipse?
開始找到這個(gè)方法 https://blog.csdn.net/Harbourside1/article/details/111871122 是不對(duì)的,后來(lái)找到這個(gè) https://blog.csdn.net/oh_maxy/article/details/48347897,正確!就是沒有選 Utility Module 這個(gè)導(dǎo)致的:
Spring MVC Java 代替 xml
web.xml
原來(lái) web.xml:
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" id="WebApp_ID" version="3.1"><!-- 啟用 Spring --><listener><listener-class>org.springframework.web.context.ContextLoaderListener</listener-class></listener><!-- // --><!-- 對(duì) Request、Response 的擴(kuò)展 --><filter><filter-name>InitMvcRequest</filter-name><filter-class>com.ajaxjs.util.spring.InitMvcRequest</filter-class></filter><filter-mapping><filter-name>InitMvcRequest</filter-name><url-pattern>/*</url-pattern></filter-mapping><!-- 全部允許跨域 --><filter><filter-name>Cors</filter-name><filter-class>com.ajaxjs.util.spring.CorsFilter</filter-class></filter><filter-mapping><filter-name>Cors</filter-name><url-pattern>/*</url-pattern></filter-mapping><!-- // --> </web-app> <!-- // -->采用 Java:
public abstract void initWeb(ServletContext servletContext);@Override public void onStartup(ServletContext servletCxt) {LOGGER.info("WEB 程序啟動(dòng)中……");servletCxt.setInitParameter("contextConfigLocation", "classpath:applicationContext.xml");servletCxt.addListener(new ContextLoaderListener()); // 監(jiān)聽器FilterRegistration.Dynamic filterReg = servletCxt.addFilter("InitMvcRequest", new InitMvcRequest());filterReg.addMappingForUrlPatterns(null, true, "/*");initWeb(servletCxt);…… }數(shù)據(jù)庫(kù)連接池失效
注意 Connection 一定要關(guān)閉!
<!-- 配置數(shù)據(jù)源 https://blog.csdn.net/syslbjjly/article/details/97108560 --> <bean id="dataSource" class="org.apache.tomcat.jdbc.pool.DataSource" destroy-method="close"><property name="driverClassName" value="com.mysql.cj.jdbc.Driver" /><!--內(nèi)網(wǎng)數(shù)據(jù)庫(kù) --><property name="url" value="jdbc:mysql://10.201.xxx.xxx:3306/bdp?useUnicode=true&characterEncoding=UTF-8&useSSL=false"></property> <property name="username" value="root"></property> <property name="password" value="xxx"></property> <!-- 驗(yàn)證連接是否有效,(String) SQL 查詢,用來(lái)驗(yàn)證從連接池取出的連接,在將連接返回給調(diào)用者之前。 如果指定,則查詢必須是一個(gè) SQL SELECT 并且必須返回至少一行記錄查詢不必返回記錄,但這樣將不能拋出 SQL 異常 --><property name="validationQuery" value="SELECT 1" /><!-- (long) 避免過(guò)度驗(yàn)證,保證驗(yàn)證不超過(guò)這個(gè)頻率——以毫秒為單位。如果一個(gè)連接應(yīng)該被驗(yàn)證, 但上次驗(yàn)證未達(dá)到指定間隔,將不再次驗(yàn)證。 30000(30秒) --><property name="validationInterval" value="18800" /><!-- 驗(yàn)證失敗時(shí),是否將連接從池中丟棄 --><property name="testWhileIdle" value="true" /><property name="testOnBorrow" value="true" /><property name="testOnReturn" value="true" /> </bean>MVC 框架中靜態(tài)資源的劃分
一般都是 MVC 框架接收所有的請(qǐng)求然后分別處理,去控制器的,還是靜態(tài)資源的,
import java.io.IOException; import java.io.UnsupportedEncodingException; import java.nio.charset.StandardCharsets; import java.util.regex.Pattern;import javax.servlet.Filter; import javax.servlet.FilterChain; import javax.servlet.FilterConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import javax.servlet.http.HttpServletRequest;/*** 自定義請(qǐng)求對(duì)象,對(duì)請(qǐng)求對(duì)象和響應(yīng)對(duì)象有很多需要擴(kuò)展的地方* * @author Frank Cheung<sp42@qq.com>**/ //@Component public class InitMvcRequest implements Filter {/*** 字符串判斷是否靜態(tài)文件*/private static final Pattern IS_STATIC = Pattern.compile("\\.jpg|\\.png|\\.gif|\\.js|\\.css|\\.less|\\.ico|\\.jpeg|\\.htm|\\.swf|\\.txt|\\.mp4|\\.flv");@Overridepublic void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws IOException, ServletException {HttpServletRequest _req = (HttpServletRequest) req;try {// 為防止中文亂碼,統(tǒng)一設(shè)置 UTF-8,設(shè)置請(qǐng)求編碼方式_req.setCharacterEncoding(StandardCharsets.UTF_8.toString());} catch (UnsupportedEncodingException e) {}if (!IS_STATIC.matcher(_req.getRequestURI()).find())chain.doFilter(req, resp);elsechain.doFilter(req, resp);}@Overridepublic void init(FilterConfig arg0) {}@Overridepublic void destroy() {} }關(guān)于 Tomcat 的一些冷知識(shí)
Tomcat 自帶許多有用的組件,直接可用。
- Tomcat 也有自己的數(shù)據(jù)庫(kù)連接池 jdbc-pool
- 自帶管理監(jiān)控工具 Manager,若不滿可以參考 PSI Probe
- 想要一個(gè)模板系統(tǒng)?用 Tomcat 自帶的 EL表達(dá)式解析器 吧,可惜的是我找不到相關(guān)的教程……只能用 Spring 的
- 可插拔以及 SCI 的實(shí)現(xiàn)原理,以及 Wrapper
- 對(duì)于特定資源的保護(hù),Tomcat 提供了安全域的功能實(shí)現(xiàn)
- Tomcat 也可以做 SSO……
- 一堆過(guò)濾器
我們知道 Spring 本身自帶一堆過(guò)濾器,Tomcat 也有呀。
- SetCharacterEncodingFilter 解決亂碼問題
- CorsFilter 跨域問題
- CsrfPreventionFilter 防止跨站請(qǐng)求偽造(CSRF)
- RemoteIpFilter/RemoteIpValve 獲取客戶端真實(shí) ip
- RemoteHostFilter、RemoteAddrFilter 獲取客戶端Host、Ip
觀察代碼執(zhí)行時(shí)間
分析效率用,用 Spring 的 StopWatch。
StopWatch sw = new StopWatch();sw.start("起床"); Thread.sleep(1000); sw.stop();sw.start("洗漱"); Thread.sleep(2000); sw.stop();System.out.println(sw.prettyPrint()); System.out.println(sw.getTotalTimeMillis()); System.out.println(sw.getLastTaskName()); System.out.println(sw.getLastTaskInfo()); System.out.println(sw.getTaskCount());如何對(duì)Spring MVC中的 Controller 進(jìn)行單元測(cè)試
例子如下,參見。
@ContextConfiguration(locations = { "classpath*:applicationContext.xml" }) @RunWith(SpringJUnit4ClassRunner.class) @WebAppConfiguration public class TestAlipay {MockMvc mockMvc;@AutowiredWebApplicationContext wac;@Beforepublic void init() {mockMvc = MockMvcBuilders.webAppContextSetup(wac).build();}// @Testpublic void testCommonUpload() throws Exception {File file = new File("C:\\Users\\frank\\Desktop\\abldj75zav.png");byte[] bytes = FileHelper.openAsByte(file);String filenmae = "abldj75zav.png";MockMultipartFile mockMultipartFile = new MockMultipartFile("file", filenmae, MediaType.MULTIPART_FORM_DATA_VALUE, bytes);ResultActions andDo = mockMvc.perform(multipart("/upload").file(mockMultipartFile)).andExpect(status().isOk()).andExpect(content().string(filenmae)).andDo(print());System.out.println(andDo);assertNotNull(andDo);}@Testpublic void testNso() {assertTrue(true);}}MySQL 于 Tomcat 的沖突
出現(xiàn)異常:checkStateForResourceLoading Illegal access,Eclipse 提示異常,生產(chǎn)環(huán)境應(yīng)該不會(huì)。寫一個(gè) Listener 解決。出處:1、2。
import java.sql.Driver; import java.sql.DriverManager; import java.sql.SQLException; import java.util.Enumeration;import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener;import org.springframework.stereotype.Component;import com.ajaxjs.Version; import com.mysql.cj.jdbc.AbandonedConnectionCleanupThread;/*** Eclipse 提示異常,生產(chǎn)環(huán)境應(yīng)該不會(huì)* * @author Frank Cheung<sp42@qq.com>**/ @WebListener @Component public class ContainerContextClosedHandler implements ServletContextListener {@SuppressWarnings("deprecation")@Overridepublic void contextDestroyed(ServletContextEvent arg0) {if (Version.isDebug) {Enumeration<Driver> drivers = DriverManager.getDrivers();Driver driver = null;// clear driverswhile (drivers.hasMoreElements()) {try {driver = drivers.nextElement();DriverManager.deregisterDriver(driver);} catch (SQLException ex) {// deregistration failed, might want to do something, log at the very least}}// MySQL driver leaves around a thread. This static method cleans it up.try {AbandonedConnectionCleanupThread.shutdown();} catch (Exception e) {// again failure, not much you can do}}}@Overridepublic void contextInitialized(ServletContextEvent arg0) {System.out.println("------------------------------------------");}}錯(cuò)誤信息 HTML
急用一個(gè)錯(cuò)誤提示頁(yè)面:
<title>操作錯(cuò)誤</title> <meta charset="utf-8" /> <div style="height: 100%%; display: flex; justify-content: center; align-items: center;"><table><tr><td align="center"> <svg width="150px" viewBox="0 0 1000 1000"><g><path fill="#ea8010" d="M500,10c-46.7,0-84.5,38-84.5,84.9v573.7c0,46.9,37.8,84.9,84.5,84.9c46.7,0,84.5-38,84.5-84.9V94.9C584.5,48,546.7,10,500,10z M500,821c-46.7,0-84.5,37.8-84.5,84.5c0,46.7,37.8,84.5,84.5,84.5c46.7,0,84.5-37.8,84.5-84.5C584.4,858.9,546.6,821,500,821z" /></g></svg></td></tr><tr><td align="center"><br />錯(cuò)誤XXXX<br /><a href="javascript:history.go(-1);">返回</a></td></tr></table> </div>效果如圖
Eclipse 重啟 Tomcat 提示 May be locked by another process
每次重啟都會(huì),很煩。原因是有打開文件的文件未關(guān)閉,Files.lines() 打開返回的 Stream<String> 也要關(guān)閉!
StringBuilder sb = new StringBuilder();try (Stream<String> lines = Files.lines(path, encode);) {lines.forEach(str -> sb.append(str));return sb.toString(); } catch (IOException e) {LOGGER.warning(e); }典型的 Spring MVC 控制器單測(cè)
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.web.WebAppConfiguration; import org.springframework.test.web.servlet.MockMvc; import org.springframework.test.web.servlet.request.MockHttpServletRequestBuilder; import org.springframework.test.web.servlet.setup.MockMvcBuilders; import org.springframework.web.context.WebApplicationContext;import com.ajaxjs.data_service.api.ApiController;@ContextConfiguration(locations = { "classpath*:applicationContext.xml" }) @RunWith(SpringJUnit4ClassRunner.class) @WebAppConfiguration public class TestSsoAccessTokenInterceptor {MockMvc mockMvc;@AutowiredWebApplicationContext wac;@AutowiredApiController apiController;@Beforepublic void init() {apiController.initCache();mockMvc = MockMvcBuilders.webAppContextSetup(wac).build();}@Testpublic void test() throws Exception {MockHttpServletRequestBuilder req = get("/user_api").param("redirect_uri", "https://www.qq.com").param("client_id", "dss23s");mockMvc.perform(req).andExpect(status().is2xxSuccessful()).andDo(print()).andReturn().getResponse();} }規(guī)避使用 PowerDesigner
老版本 12 沒有上傳數(shù)據(jù)的功能,可以規(guī)避版權(quán)檢測(cè)。通用破解方法:修改安裝目錄下的 pdflm12.dll 文件,使用二進(jìn)制編輯器打開此文件,查找:83 C4 14 8B 85 E4 FE FF FF將此字符串改為 83 C4 14 33 C0 90 90 90 90
推薦 Hex Editor: wxMEdit。 Frhed 也小巧,但找不到搜索 Hex 的方法,郁悶。
使用正則提取網(wǎng)頁(yè)中a標(biāo)簽的鏈接和標(biāo)題
import java.util.regex.Matcher; import java.util.regex.Pattern;public class Test1 {public static void main(String[] args) {String str1 = "<a href=\"https://www.zifangsky.cn/2015/10/hello-world/\" title=\"\" data-original-title=\"Hello World\">Hello World</a>";String str2 = "<a href=\"http://banzhuanboy.com/363.html\" class=\"post-feature\" \">123</a>";String str3 = " <a class=\"article-title\" href=\"/2015/12/17/Webstorm-Hotkeys-For-Mac/\">c</a>";String str4 = " <a rel=\"bookmark\" title=\"Permanent Link to 黑客組織‘SkidNP’涂改了Phantom Squad的網(wǎng)站首頁(yè)\" href='12/hack-30127.htm'>黑</a>";String str5 = "<a href=\"http://www.imorlin.com/2015/12/24/1-3/\" title=\"\" data-original-title=\"2015圣誕節(jié)雪花代碼[天貓+C店]\"> 2015圣誕節(jié)雪花代碼[天貓+C店] <span class=\"label label-new entry-tag\">New</span> </a>";Pattern pattern = Pattern.compile("<a.*?href=[\"']?((https?://)?/?[^\"']+)[\"']?.*?>(.+)</a>"); Matcher matcher = pattern.matcher(str1);if(matcher.find()){String link = matcher.group(1).trim();String title = matcher.group(3).trim();if(!link.startsWith("http")){if(link.startsWith("/"))link = "https://www.zifangsky.cn" + link;else link = "https://www.zifangsky.cn" + link; }System.out.println("link: " + link);System.out.println("title: " + title);}} }解釋:
1 選取了幾個(gè)有代表性的 a 標(biāo)簽樣式進(jìn)行測(cè)試
2 關(guān)于正則匹配模式”<a.*?href=[\”‘]?((https?://)?/?[^\”‘]+)[\”‘]?.*?>(.+)</a>“的說(shuō)明:
i)<a.*?href= <a 開頭,中間緊跟著有0個(gè)或者多個(gè)字符,然后再跟著 href=
ii)[\”‘]?((https?://)?/? 一個(gè)或者0個(gè)的” 或者 ‘ ,然后再跟著0個(gè)或者一個(gè)的http://或者h(yuǎn)ttps:// ,再跟著0個(gè)或者1個(gè)的 /
iii)[^\”‘]+ 表示1個(gè)以上的不包括’或者” 的任意字符
iv)[\”‘]?表示鏈接后面的’或者” 當(dāng)然也可能沒有
后面的可以根據(jù)前面的自己推理,就不解釋了
3 matcher.group(1)表示取出鏈接,也就是第二個(gè)()的內(nèi)容(PS:第一個(gè)()表示的是整個(gè)正則表達(dá)式,默認(rèn)省略了),在正則中是這一段規(guī)則:((https?://)?/?[^\”‘]+)
4 matcher.group(3) 同理可知,對(duì)應(yīng)的是這一段規(guī)則:(.+)
5 對(duì)于代碼中的 https://www.zifangsky.cn ,這是由于部分鏈接使用了相對(duì)路徑,比如說(shuō):href=’12/hack-30127.htm’ 。這時(shí)我們就需要加上它的域名,當(dāng)然需要根據(jù)實(shí)際情況來(lái)加。這里我就隨便亂加了
構(gòu)建可重復(fù)讀取 inputStream 的 request
我們知道,request 的 inputStream 只能被讀取一次,多次讀取將報(bào)錯(cuò),那么如何才能重復(fù)讀取呢?答案之一是:增加緩沖,記錄已讀取的內(nèi)容。
import org.springframework.mock.web.DelegatingServletInputStream;import javax.servlet.ServletInputStream; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequestWrapper; import java.io.*;/*** request wrapper: 可重復(fù)讀取request.getInputStream*/ public class RepeatedlyReadRequestWrapper extends HttpServletRequestWrapper {private static final int BUFFER_START_POSITION = 0;private static final int CHAR_BUFFER_LENGTH = 1024;/*** input stream 的buffer*/private final String body;/*** @param request {@link javax.servlet.http.HttpServletRequest} object.*/public RepeatedlyReadRequestWrapper(HttpServletRequest request) {super(request);StringBuilder stringBuilder = new StringBuilder();InputStream inputStream = null;try {inputStream = request.getInputStream();} catch (IOException e) {log.error("Error reading the request body…", e);}if (inputStream != null) {try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream))) {char[] charBuffer = new char[CHAR_BUFFER_LENGTH];int bytesRead;while ((bytesRead = bufferedReader.read(charBuffer)) > 0) {stringBuilder.append(charBuffer, BUFFER_START_POSITION, bytesRead);}} catch (IOException e) {log.error("Fail to read input stream",e);}} else {stringBuilder.append("");}body = stringBuilder.toString();}@Overridepublic ServletInputStream getInputStream() throws IOException {final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(body.getBytes());return new DelegatingServletInputStream(byteArrayInputStream);} }接下來(lái),需要一個(gè)對(duì)應(yīng)的 Filter
import javax.servlet.*; import javax.servlet.http.HttpServletRequest; import java.io.IOException;public class RepeatlyReadFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {if (request instanceof HttpServletRequest) {request = new RepeatedlyReadRequestWrapper((HttpServletRequest) request);}chain.doFilter(request, response);}@Overridepublic void destroy() { } }出處。
使用 Javassist 在 tomcat 容器中實(shí)現(xiàn)動(dòng)態(tài) Mock
在某些復(fù)雜場(chǎng)景下,我們需要對(duì)運(yùn)行在 tomcat 容器中部分功能進(jìn)行 mock(替換其實(shí)現(xiàn)),但該部分功能散落在各處,我們希望不修改源代碼以非侵入的方式來(lái)實(shí)現(xiàn) Mock,在這種情況下,我們可以應(yīng)用 Javassist 來(lái)實(shí)現(xiàn)。
使用Javassist在tomcat容器中動(dòng)態(tài)替換源碼來(lái)實(shí)現(xiàn)動(dòng)態(tài) Mock
我們可以定義一個(gè) ContextListener 的實(shí)例,在 tomcat 啟動(dòng)時(shí)通過(guò) Javassis t對(duì)源代碼進(jìn)行動(dòng)態(tài)替換,來(lái)實(shí)現(xiàn) mock 的功能。
Javassist 相關(guān)代碼,需要使用的工具類。
private static ClassPool classPool;static {classPool = ClassPool.getDefault();classPool.insertClassPath(new ClassClassPath(JavassitUtil.class)); //主要用于web環(huán)境}/*** 替換方法體** @param className 類名,如:foo.Student* @param methodName 方法名* @param newMethodBody 新的方法體,如:"System.out.println(\"this method is changed dynamically!\");"*/public static void replaceMethodBody(String className, String methodName, String newMethodBody) {try {CtClass clazz =classPool.get(className);CtMethod method = clazz.getDeclaredMethod(methodName);method.setBody(newMethodBody);clazz.toClass();} catch (NotFoundException | CannotCompileException e) {throw new RuntimeException(e);}}出處。
Spring 單元測(cè)試干掉 xml
Spring 全注解化了,但單測(cè)還是有個(gè) xml 的小尾巴。
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd"><!-- 掃描的包 --><context:component-scanbase-package="com.ajaxjs.data_service, com.ajaxjs.entity, com.ajaxjs.rpc" /> </beans>二貨 Eclipse 整天校驗(yàn)這個(gè) xml,還卡住。——其實(shí)可以創(chuàng)建一個(gè) Java 類來(lái)代替 XML 文件:
import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration;@Configuration @ComponentScan({ "com.ajaxjs.data_service", "com.ajaxjs.entity", "com.ajaxjs.rpc" }) public class TestConfig {}單測(cè):
import static org.junit.Assert.assertNotNull;import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.web.WebAppConfiguration;import com.ajaxjs.entity.datadict.DataDictService;@ContextConfiguration(classes = TestConfig.class) @RunWith(SpringJUnit4ClassRunner.class) @WebAppConfiguration public class TestDataDict {@AutowiredDataDictService dataDictService;@Testpublic void test() {assertNotNull(dataDictService);} }Java 枚舉技巧
枚舉除了名稱還有常量值,可以用 int 保存。
public static enum Lock {ADD_LOCK(0), UNLOCK(1);private int value;Lock(int value) {this.value = value;}public int getValue() {return value;} }如果常量值是 0、1、2 順序的,可以不設(shè)置 int,直接:
public static enum Lock {ADD_LOCK, UNLOCK; }lock.ordinal() 即可返回順序的 int。
Ubuntu sudo 不用每次都輸入密碼的解決辦法
雖然 sudo -i 可以避免輸入密碼,但 sftp 不行啊,怎么辦!?
網(wǎng)上說(shuō)的辦法都不行,直接修改/etc/sudoers文件的最后一行:
%sudo ALL=(ALL:ALL) ALL 修改為 %sudo ALL=(ALL:ALL) NOPASSWD:ALL我還改崩了,無(wú)法使用 sudo,解決辦法參見《Ubuntu改壞sudoers后無(wú)法使用sudo的解決辦法》
實(shí)際上 如果你對(duì)那文件有用戶權(quán)限,是不用輸入密碼的,使用 chown -R 用戶名 修改就行。
前端:左菜單,右主區(qū)域,左絕對(duì)值,右自動(dòng)填滿的布局
用 Flex 布局,如下:
.container {display: flex;height: 100%;.left {height: 100%;flex: 0 0 300px;border-right: 1px solid lightgray;}.right {height: 100%;flex: 1;/*div占據(jù)所有剩余寬度 */} }JS 快速壓縮 CSS 代碼
很簡(jiǎn)單的:
/* 壓縮 css 并保存 */ function compress(code) { code = code.replace(/\n/ig, ''); // 去掉換行 code = code.replace(/(\s){2,}/ig, '$1'); // 多空間(兩個(gè)以上) 變 一個(gè)空格 code = code.replace(/\t/ig, ''); // 去掉tab code = code.replace(/\n\}/ig, '\}'); // 換行+} 變 不換行 code = code.replace(/\n\{\s*/ig, '\{'); // {+換行 變 不換行 code = code.replace(/(\S)\s*\}/ig, '$1\}'); // 去掉 內(nèi)容 與 } 之間的空格 code = code.replace(/(\S)\s*\{/ig, '$1\{'); // 去掉 內(nèi)容 與 { 之間的空格 code = code.replace(/\{\s*(\S)/ig, '\{$1'); // 去掉 { 與 內(nèi)容之間空格 return code; }MySQL 調(diào)整時(shí)區(qū)
SELECT TIMEDIFF(NOW(), UTC_TIMESTAMP);如果是中國(guó)標(biāo)準(zhǔn)時(shí)間, 會(huì)輸出08:00
修改時(shí)區(qū)
set global time_zone = '+8:00'; ##修改mysql全局時(shí)區(qū)為北京時(shí)間,即我們所在的東8區(qū) set time_zone = '+8:00'; ##修改當(dāng)前會(huì)話時(shí)區(qū) flush privileges; #立即生效IDEA 社區(qū)版新建 SpringBoot 項(xiàng)目
社區(qū)版下,Spring BootHelper 居然收費(fèi)。怎么破?利用官方的 Spring Initializr 生成項(xiàng)目,導(dǎo)入即可。
Mybatis insert 的入?yún)閙ap時(shí),insert 語(yǔ)句中獲取key和value的寫法
https://blog.csdn.net/qq_40580023/article/details/84992429
MyBatis更新數(shù)據(jù)(輸入?yún)?shù)類型為Map)
https://blog.csdn.net/Dr_Guo/article/details/79057153
Servlet 3 原生文件上傳
package com.ajaxjs.image;import java.io.BufferedInputStream; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.io.PrintWriter; import java.util.Collection;import javax.servlet.ServletException; import javax.servlet.annotation.MultipartConfig; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.Part;import com.ajaxjs.util.io.StreamHelper;/*** Servlet implementation class Api*/ @WebServlet("/img_api/*") @MultipartConfig public class Api extends HttpServlet {private static final long serialVersionUID = 1L;/*** @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse* response)*/protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {new Target(request);String filename = "c:\\temp\\11.jpg";try (InputStream bin = new BufferedInputStream(new FileInputStream(filename));) {StreamHelper.write(bin, response.getOutputStream(), true);}}String savePath = "c:\\temp\\";/*** @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse* response)*/protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {Collection<Part> parts = request.getParts();// 獲取上傳的文件集合if (parts.size() == 1) {// 上傳單個(gè)文件// Servlet3.0 將 multipart/form-data 的 POST 請(qǐng)求封裝成 Part,通過(guò) Part 對(duì)上傳的文件進(jìn)行操作。// Part part = parts[0];//從上傳的文件集合中獲取 Part 對(duì)象Part part = request.getPart("file");// 通過(guò)表單 file 控件(<input type="file" name="file">)的名字直接獲取 Part 對(duì)象uplaod(part, savePath);} else {for (Part part : parts)// 一次性上傳多個(gè)文件uplaod(part, savePath);}// response.getWriter().append("Served at: ").append(request.getContextPath());try (PrintWriter out = response.getWriter();) {out.println("上傳成功");}}public static void uplaod(Part part, String savePath) {String header = part.getHeader("content-disposition");// 獲取請(qǐng)求頭,請(qǐng)求頭的格式:form-data; name="file"; filename="snmp4j--api.zip"try {part.write(savePath + File.separator + getFileName(header));// 把文件寫到指定路徑} catch (IOException e) {e.printStackTrace();}}/*** Servlet3 沒有提供直接獲取文件名的方法,需要從請(qǐng)求頭中解析出來(lái) 根據(jù)請(qǐng)求頭解析出文件名* 請(qǐng)求頭的格式:火狐和google瀏覽器下:form-data; name="file"; filename="snmp4j--api.zip"* IE瀏覽器下:form-data; name="file"; filename="E:\snmp4j--api.zip"* * @param header 請(qǐng)求頭* @return 文件名*/public static String getFileName(String header) {/** String[] tempArr1 = header.split(";");代碼執(zhí)行完之后,在不同的瀏覽器下,tempArr1數(shù)組里面的內(nèi)容稍有區(qū)別* 火狐或者google瀏覽器下:tempArr1={form-data,name="file",filename="snmp4j--api.zip"}* IE瀏覽器下:tempArr1={form-data,name="file",filename="E:\snmp4j--api.zip"}*/String[] tempArr1 = header.split(";");/** 火狐或者google瀏覽器下:tempArr2={filename,"snmp4j--api.zip"}* IE瀏覽器下:tempArr2={filename,"E:\snmp4j--api.zip"}*/String[] tempArr2 = tempArr1[2].split("=");// 獲取文件名,兼容各種瀏覽器的寫法String fileName = tempArr2[1].substring(tempArr2[1].lastIndexOf("\\") + 1).replaceAll("\"", "");return fileName;}}這樣理解 java 中的 volatile 更簡(jiǎn)單些
程序運(yùn)行時(shí),有2大塊內(nèi)存,主內(nèi)存和本地內(nèi)存,當(dāng)某線程讀或?qū)懸粋€(gè)變量時(shí),先操作本地內(nèi)存,再選擇合適的時(shí)機(jī)同步到主內(nèi)存中。
并發(fā)三個(gè)重要的概念:原子性,可見性,有序性。
關(guān)于原子性:
1,synchronized{}修飾的代碼塊,可保證原子性
2,對(duì)于volatile int i = 0;
i = 2;是原子操作
i++;不是原子操作,因?yàn)橐茸x取i的當(dāng)前值,再進(jìn)行自增,再進(jìn)行賦值操作
i = i;不是原子操作,因?yàn)橐茸x取i的當(dāng)前值,再進(jìn)行賦值操作
int j = i;不是原子操作,因?yàn)橐茸x取i的當(dāng)前值,再進(jìn)行賦值操作
- volatile 只具備可見性和有序性。
- volatile 可見性,當(dāng)線程給該變量賦值時(shí),該新值會(huì)先修改到本地內(nèi)存,再直接同步到主內(nèi)存。
- volatile 有序性,當(dāng)純種讀取該變量值時(shí),必須先從主內(nèi)存讀取最新的值,再同步到本地內(nèi)存中,再?gòu)谋镜貎?nèi)存中讀取最新值。
Linux 啟動(dòng) JAR 包 Shell 腳本
實(shí)用呀,可以 kill 掉已有進(jìn)程然后啟動(dòng)
port=8083 pid=$(netstat -nlp | grep :$port | awk '{print $7}' | awk -F"/" '{ print $1 }') kill -9 ${pid} echo "killed ${pid}"nohup java -jar auth.jar > /dev/null & tail -f /data/logs/uam/auth/log_debug.log最后一行 tail -f /data/logs/uam/auth/log_debug.log 觀察日志的可以不執(zhí)行或者修改 文檔地址。
如何測(cè)試高并發(fā)的線程安全
結(jié)合 CountDownLatch類和Semaphore類測(cè)試(出處),例子如下。
import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Semaphore;/*** @author binghe* @version 1.0.0* @description 測(cè)試SimpleDateFormat的線程不安全問題*/ public class SimpleDateFormatTest01 {//執(zhí)行總次數(shù)private static final int EXECUTE_COUNT = 1000;//同時(shí)運(yùn)行的線程數(shù)量private static final int THREAD_COUNT = 20;//SimpleDateFormat對(duì)象private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");public static void main(String[] args) throws InterruptedException {final Semaphore semaphore = new Semaphore(THREAD_COUNT);final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);ExecutorService executorService = Executors.newCachedThreadPool();for (int i = 0; i < EXECUTE_COUNT; i++){executorService.execute(() -> {try {semaphore.acquire();try {simpleDateFormat.parse("2020-01-01");} catch (ParseException e) {System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");e.printStackTrace();System.exit(1);}catch (NumberFormatException e){System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");e.printStackTrace();System.exit(1);}semaphore.release();} catch (InterruptedException e) {System.out.println("信號(hào)量發(fā)生錯(cuò)誤");e.printStackTrace();System.exit(1);}countDownLatch.countDown();});}countDownLatch.await();executorService.shutdown();System.out.println("所有線程格式化日期成功");} }maven中 Failed to read schema document 錯(cuò)誤
https://blog.csdn.net/qq_39741730/article/details/104663761
自動(dòng) kill 進(jìn)程再啟動(dòng),并輸出日志文件
# 獲取進(jìn)程名 process_name=new-fleet-market-business-1.0-SNAPSHOT.jar# 查找進(jìn)程 ID pid=$(jps -l | grep $process_name | awk '{print $1}')# 打印進(jìn)程 ID echo "進(jìn)程 ID 為:$pid"# 判斷進(jìn)程 ID 是否為空 if [ -n "$pid" ]; then# 終止進(jìn)程kill -9 $pidecho "停止進(jìn)程 $pid" elseecho "沒有找到進(jìn)程 $process_name" fiecho "啟動(dòng)程序" nohup java -Xms512m -Xmx512m -jar ./$process_name >message.log 2>&1 &總結(jié)
- 上一篇: 1.13正版服务器,我的世界Minecr
- 下一篇: 计算机导论alu的全名,计算机导论试题1