how2j.cn

-->
下载区
文件名 文件大小
请先登录 11m
增值内容 11m
11m

解压rar如果失败,请用5.21版本或者更高版本的winrar

点击下载 winrar5.21

2分
本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器。 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)。 chrome 的 视频下载插件会影响播放,如 IDM 等,请关闭或者切换其他浏览器

步骤 1 : 拓扑图点亮   
步骤 2 : Request 的方法   
步骤 3 : 常见方法   
步骤 4 : TestTomcat   
步骤 5 : 比较可运行项目,快速定位问题   

增值内容,请先登录
自己写一个Tomcat, 几乎使用到了除开框架外的所有Java 技术,如多线程,Socket, J2EE, 反射,Log4j, JSoup, JUnit, Html 等一整套技术栈, 从无到有,循序渐进涵盖全部74个知识点,549个开发步骤, 为竞争高薪资职位加上一个有吸引力的砝码.
增值内容,点击购买
使用爬虫已经被系统记录,请勿使用爬虫,增大封号风险。 如果是误封 ,请联系站长,谢谢
拓扑图点亮
步骤 2 :

Request 的方法

edit
增值内容,请先登录
自己写一个Tomcat, 几乎使用到了除开框架外的所有Java 技术,如多线程,Socket, J2EE, 反射,Log4j, JSoup, JUnit, Html 等一整套技术栈, 从无到有,循序渐进涵盖全部74个知识点,549个开发步骤, 为竞争高薪资职位加上一个有吸引力的砝码.
增值内容,点击购买
使用爬虫已经被系统记录,请勿使用爬虫,增大封号风险。 如果是误封 ,请联系站长,谢谢
增值内容,请先登录
自己写一个Tomcat, 几乎使用到了除开框架外的所有Java 技术,如多线程,Socket, J2EE, 反射,Log4j, JSoup, JUnit, Html 等一整套技术栈, 从无到有,循序渐进涵盖全部74个知识点,549个开发步骤, 为竞争高薪资职位加上一个有吸引力的砝码.
增值内容,点击购买
使用爬虫已经被系统记录,请勿使用爬虫,增大封号风险。 如果是误封 ,请联系站长,谢谢
package cn.how2j.diytomcat.http; import cn.how2j.diytomcat.catalina.Context; import cn.how2j.diytomcat.catalina.Engine; import cn.how2j.diytomcat.catalina.Service; import cn.how2j.diytomcat.util.MiniBrowser; import cn.hutool.core.convert.Convert; import cn.hutool.core.convert.Converter; import cn.hutool.core.io.IoUtil; import cn.hutool.core.util.ArrayUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.core.util.URLUtil; import javax.servlet.ServletContext; import java.io.IOException; import java.io.InputStream; import java.io.StringReader; import java.net.InetSocketAddress; import java.net.Socket; import java.util.*; public class Request extends BaseRequest{ private String requestString; private String uri; private Socket socket; private Context context; private Service service; private String method; private String queryString; private Map<String, String[]> parameterMap; private Map<String, String> headerMap; public Request(Socket socket, Service service) throws IOException { this.parameterMap = new HashMap(); this.headerMap = new HashMap<>(); this.socket = socket; this.service = service; parseHttpRequest(); if(StrUtil.isEmpty(requestString)) return; parseUri(); parseContext(); parseMethod(); if(!"/".equals(context.getPath())){ uri = StrUtil.removePrefix(uri, context.getPath()); if(StrUtil.isEmpty(uri)) uri = "/"; } parseParameters(); parseHeaders(); } private void parseMethod() { method = StrUtil.subBefore(requestString, " ", false); } private void parseContext() { Engine engine = service.getEngine(); context = engine.getDefaultHost().getContext(uri); if(null!=context) return; String path = StrUtil.subBetween(uri, "/", "/"); if (null == path) path = "/"; else path = "/" + path; context = engine.getDefaultHost().getContext(path); if (null == context) context = engine.getDefaultHost().getContext("/"); } private void parseHttpRequest() throws IOException { InputStream is = this.socket.getInputStream(); byte[] bytes = MiniBrowser.readBytes(is,false); requestString = new String(bytes, "utf-8"); } private void parseUri() { String temp; temp = StrUtil.subBetween(requestString, " ", " "); if (!StrUtil.contains(temp, '?')) { uri = temp; return; } temp = StrUtil.subBefore(temp, '?', false); uri = temp; } public Context getContext() { return context; } public String getUri() { return uri; } public String getRequestString(){ return requestString; } @Override public String getMethod() { return method; } public ServletContext getServletContext() { return context.getServletContext(); } public String getRealPath(String path) { return getServletContext().getRealPath(path); } private void parseParameters() { if ("GET".equals(this.getMethod())) { String url = StrUtil.subBetween(requestString, " ", " "); if (StrUtil.contains(url, '?')) { queryString = StrUtil.subAfter(url, '?', false); } } if ("POST".equals(this.getMethod())) { queryString = StrUtil.subAfter(requestString, "\r\n\r\n", false); } if (null == queryString || 0==queryString.length()) return; queryString = URLUtil.decode(queryString); String[] parameterValues = queryString.split("&"); if (null != parameterValues) { for (String parameterValue : parameterValues) { String[] nameValues = parameterValue.split("="); String name = nameValues[0]; String value = nameValues[1]; String values[] = parameterMap.get(name); if (null == values) { values = new String[] { value }; parameterMap.put(name, values); } else { values = ArrayUtil.append(values, value); parameterMap.put(name, values); } } } } public String getParameter(String name) { String values[] = parameterMap.get(name); if (null != values && 0 != values.length) return values[0]; return null; } public Map getParameterMap() { return parameterMap; } public Enumeration getParameterNames() { return Collections.enumeration(parameterMap.keySet()); } public String[] getParameterValues(String name) { return parameterMap.get(name); } public String getHeader(String name) { if(null==name) return null; name = name.toLowerCase(); return headerMap.get(name); } public Enumeration getHeaderNames() { Set keys = headerMap.keySet(); return Collections.enumeration(keys); } public int getIntHeader(String name) { String value = headerMap.get(name); return Convert.toInt(value, 0); } public void parseHeaders() { StringReader stringReader = new StringReader(requestString); List<String> lines = new ArrayList<>(); IoUtil.readLines(stringReader, lines); for (int i = 1; i < lines.size(); i++) { String line = lines.get(i); if (0 == line.length()) break; String[] segs = line.split(":"); String headerName = segs[0].toLowerCase(); String headerValue = segs[1]; headerMap.put(headerName, headerValue); } } public String getLocalAddr() { return socket.getLocalAddress().getHostAddress(); } public String getLocalName() { return socket.getLocalAddress().getHostName(); } public int getLocalPort() { return socket.getLocalPort(); } public String getProtocol() { return "HTTP:/1.1"; } public String getRemoteAddr() { InetSocketAddress isa = (InetSocketAddress) socket.getRemoteSocketAddress(); String temp = isa.getAddress().toString(); return StrUtil.subAfter(temp, "/", false); } public String getRemoteHost() { InetSocketAddress isa = (InetSocketAddress) socket.getRemoteSocketAddress(); return isa.getHostName(); } public int getRemotePort() { return socket.getPort(); } public String getScheme() { return "http"; } public String getServerName() { return getHeader("host").trim(); } public int getServerPort() { return getLocalPort(); } public String getContextPath() { String result = this.context.getPath(); if ("/".equals(result)) return ""; return result; } public String getRequestURI() { return uri; } public StringBuffer getRequestURL() { StringBuffer url = new StringBuffer(); String scheme = getScheme(); int port = getServerPort(); if (port < 0) { port = 80; // Work around java.net.URL bug } url.append(scheme); url.append("://"); url.append(getServerName()); if ((scheme.equals("http") && (port != 80)) || (scheme.equals("https") && (port != 443))) { url.append(':'); url.append(port); } url.append(getRequestURI()); return url; } public String getServletPath() { return uri; } }
增值内容,请先登录
自己写一个Tomcat, 几乎使用到了除开框架外的所有Java 技术,如多线程,Socket, J2EE, 反射,Log4j, JSoup, JUnit, Html 等一整套技术栈, 从无到有,循序渐进涵盖全部74个知识点,549个开发步骤, 为竞争高薪资职位加上一个有吸引力的砝码.
增值内容,点击购买
使用爬虫已经被系统记录,请勿使用爬虫,增大封号风险。 如果是误封 ,请联系站长,谢谢
TestTomcat
步骤 5 :

比较可运行项目,快速定位问题

edit
增值内容,请先登录
自己写一个Tomcat, 几乎使用到了除开框架外的所有Java 技术,如多线程,Socket, J2EE, 反射,Log4j, JSoup, JUnit, Html 等一整套技术栈, 从无到有,循序渐进涵盖全部74个知识点,549个开发步骤, 为竞争高薪资职位加上一个有吸引力的砝码.
增值内容,点击购买
使用爬虫已经被系统记录,请勿使用爬虫,增大封号风险。 如果是误封 ,请联系站长,谢谢


HOW2J公众号,关注后实时获知最新的教程和优惠活动,谢谢。


问答区域    
2020-11-18 getRemotePort()
ilqjx




站长,这个套接字所连接的端点是什么啊,是Tomcat服务器吗?那个端口不是Tomcat服务器的端口 18080,而且每次属性都会有一个新的端口
public int getRemotePort() {
      return socket.getPort();
}

							


2 个答案

how2j
答案时间:2020-11-19
因为每次连接客户端用的端口号都不一样嘛。 Connection: keep alive 是长连接,站长觉得这个太复杂了,所以没有纳入 diytomcat 开发来,觉得同学们会在这里卡住不动,影响学习积极性。。。。

ilqjx
答案时间:2020-11-18
getRemotePort() 是获取远程的端口,那么这个远程是谁呢?因为是服务器在问,所以远程就是客户端啦。所以 getRemotePort() 获取到的就是客户端的端口号了呗。 那为什么每次会变呢? 我猜一下,这个具体还需要站长来解答才行。 因为我们目前没有对Connection: keep alive这个头信息做任何的处理,这个头信息就是告诉服务器传完数据后,不要立即关闭连接,要等待一小段时间。目前我们的做法是传完数据后,直接把连接给关闭了,所以下一次访问就是一个新的连接了,所以每次刷新端口号会不一样。



回答已经提交成功,正在审核。 请于 我的回答 处查看回答记录,谢谢
答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到





2020-10-11 哈哈哈,图片
HOW2JZkk817

《懒得写》是认真的么?站长大大




1 个答案

how2j
答案时间:2020-10-13
是的啊,哈哈哈~



回答已经提交成功,正在审核。 请于 我的回答 处查看回答记录,谢谢
答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到








提问之前请登陆
提问已经提交成功,正在审核。 请于 我的提问 处查看提问记录,谢谢
关于 实践项目-DiyTomcat-Request 常见方法 的提问

尽量提供截图代码异常信息,有助于分析和解决问题。 也可进本站QQ群交流: 578362961
提问尽量提供完整的代码,环境描述,越是有利于问题的重现,您的问题越能更快得到解答。
对教程中代码有疑问,请提供是哪个步骤,哪一行有疑问,这样便于快速定位问题,提高问题得到解答的速度
在已经存在的几千个提问里,有相当大的比例,是因为使用了和站长不同版本的开发环境导致的,比如 jdk, eclpise, idea, mysql,tomcat 等等软件的版本不一致。
请使用和站长一样的版本,可以节约自己大量的学习时间。 站长把教学中用的软件版本整理了,都统一放在了这里, 方便大家下载: https://how2j.cn/k/helloworld/helloworld-version/1718.html

上传截图