会员可以在此提问,百战程序员老师有问必答
对大家有帮助的问答会被标记为“推荐”
看完课程过来浏览一下别人提的问题,会帮你学得更全面
截止目前,同学们一共提了 133059个问题
JAVA 全系列/第二阶段:JAVA 基础深化和提高/容器 3769楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/手写服务器项目(旧) 3770楼

老师好,

请看一下,这个细节问题,找了很久,请看一下,结果不一样,多了点东西,找不到

package com.bjsxt.server;

import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class Request {//请求
	private InputStream in;//输入流
	private String requestInfo;//请求字符串,请求方式,请求请求路径 请求参数,请求正文 请求协议。。。。
	private String method;//情趣方式
	private String url;//请求的url路径
	/*输入框的name为key,值为:value
	 * String  list集合
	 * key: name   value: bjsxt
	 * key: pwd    value: 123
	 * key: hobby  value: read,ball   对应多个值的情况
	 */
	private Map<String, List<String>> parameterMapValues;//请求参数
	private static final String CRLF="\r\n";//换行
	private static final String BLANK=" ";//空格
	public Request() {
		super();
		// TODO Auto-generated constructor stub
		method = "";
		url = "";
		requestInfo = "";
		parameterMapValues = new HashMap<String, List<String>>();
	}
	public Request(InputStream in) {
		this();//调用本类中的无参构造法方法
		this.in = in;
		byte[] buf = new byte[20480];//一次性读过来
		try {
			int len = this.in.read(buf);
			requestInfo = new String(buf,0,len);
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
			return;
		}
		//调用分解请求信息的方法
		parseRequestInfo();
	}
	//分解请求信息的方法
	private void parseRequestInfo() {
		/*
		 * 请求方式
		 * 请求路径
		 * 请求参数
		 */
		//获取请求参数的第一行
		String firstLine = requestInfo.substring(0,requestInfo.indexOf(CRLF)).trim();
		//分解出请求方式
		int index = firstLine.indexOf("/");
		this.method = firstLine.substring(0,index).trim();
		String paraString = "";//用于存储请求参数
		//分解url 可能包含参数get  也可能不包含参数post
		String urlString = firstLine.substring(index,firstLine.indexOf("HTTP/")).trim();
		//判断请求方式是GET还是POST
		if ("get".equalsIgnoreCase(this.method)) {//包含请求参数
			if (urlString.contains("?")) {
				String[] urlArray = urlString.split("\\?");
				this.url = urlArray[0];
				paraString = urlArray[1];
			}else {
				this.url = urlString;
			}
		}else {//不包含请求参数
			this.url = urlString;
			paraString = requestInfo.substring(requestInfo.lastIndexOf(CRLF)).trim();
			if (paraString.equals("")) {
				return;
			}
		}
			
			//调用解析参数的方法
			parsePara(paraString);
		
	}
	//分解请求参数,封装入map中
	public void parsePara(String paraString) {
		String[] token = paraString.split("&");
		for (String keyValues : token) {
			String[] keyValue = keyValues.split("=");
			if (keyValue.length==1) {
				keyValue = Arrays.copyOf(keyValue, 2);
				keyValue[1] = null;
			}
			//将表单元素的name和name对应的值存储到map集合中
			String key = keyValue[0].trim();
			String value = keyValue[1]==null?null:decode(keyValue[1], "gbk").trim();
			//关联key value
			if (!parameterMapValues.containsKey(key)) {
				parameterMapValues.put(key, new ArrayList<String>());
			}
			parameterMapValues.get(key).add(value);
		}
	}
	//浏览器发送请求时,会将中文编码,服务器接收请求信息时候进行解码
	public String decode(String value,String code) {
		try {
			return URLDecoder.decode(value, code);
		} catch (UnsupportedEncodingException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		return null;
	}
	
	public static void main(String[] args) {
		Request req = new Request();
		req.parsePara("http://localhost:8888/?username=efe&password=fef&hobby=ball&hobby=paint");
		System.out.println(req.parameterMapValues);
	}
}

效果截图:

image.png

JAVA 全系列/第二阶段:JAVA 基础深化和提高/手写服务器项目(旧) 3773楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/容器(旧) 3774楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/IO 流技术(旧) 3775楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/手写服务器项目(旧) 3776楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/容器(旧) 3777楼

看了这次课的视频,讲原理的时候,有个疑问:

ThreadLocal的内部类ThreadLocalMap是一个Thread的成员变量,key是Tread对象,value即需要存储的值;

但是我疑问的是一个该线程的共享变量只能存一个值吗?

经过验证第二次的set是会覆盖第一次set的值的;

那么如果线程的共享变量ThreadLocal只有一个的话,存成Map来表示一个键值关系有必要吗?为什么要设计成Map集合来存储呢?


查帖子理解了一下,发现,一个线程的共享变量ThreadLocal是可以有多个的,就是可以在共享资源里定义多个静态的ThreadLocal的成员变量来存放不同的值。

比如在共享资源SequenceImple里有两个静态的ThreadLocal成员变量,

一个是numThreadLocal 一个是charThreadLocal。并且重写initialValue()方法分别设置默认值

一个是数字容器,默认值0,一个是字母容器,默认值A。

使用多线程分别实现数字和字母顺序打印(每次打印1A、2B),开启3个线程,run方法打印2次。

打算实现的结果是,

线程A——>打印1A

线程B——>打印1A

线程A——>打印2B

线程C——>打印1A

线程A——>打印3C

线程C——>打印2B

线程B——>打印2B

线程B——>打印3C

线程C——>打印3C

顺序不固定,因为cpu在不停切换,但是要达到的效果与视频当中相比,就多了一个类变量容器,用来打印字母。


这个时候第一个线程执行,run方法中调用numThreadLocal 的set方法的时候,得到的ThreadLocalMap集合里面,就会有两组键值对,

第一组的键是numThreadLocal 对象,值0。

第二组是 charThreadLocal对象,值是A。 

因为对应一个线程,操作的共享资源SequenceImple的时候,出现了两个类变量容器,即numThreadLocal 和charThreadLocal。(课上案例只有一个)

为了给每一个线程,创建多个类变量容器的副本,并且可以set与get它们,在ThreadLocal中定义了内部类 ThreadLocalMap,并将其作为Thread类的成员变量。

一旦作为了Thread类的成员变量,那么每个线程都会拥有一个ThreadLocalMap对象,使用此map集合,键为共享资源中的类变量容器对象,值为需要存储的值(set进去的值)。

也就是说,每个线程当中都有了一个map,这个map存了共享资源类当中的类变量容器(可以有一个也可以有多个)和它的值。

每次使用类变量容器的set方法的时候,会利用当前的类变量容器(key)找到该键值对,重新覆盖原来的value。

使用类变量容器的get方法的时候会利用当前的类变量容器(key)找到该键值对,找到value并返回。

image.png

并且由于每个线程都有自己的map,所以互相不受对方线程的影响。


所以,存储成Map集合,是因为共享资源类里可能出现多个类变量容器。每个类变量容器及其值作为一组键值对。视频中共享资源类SequenceImple2只有一个类变量容器numberContainer ,因此画图的时候对应一组键值对。

private static ThreadLocal<Integer> numberContainer =new ThreadLocal<Integer>(){

        protected Integer initialValue(){

                       return 0;//设置容器存储内容的默认值

     }

}



JAVA 全系列/第二阶段:JAVA 基础深化和提高/多线程和并发编程(旧) 3778楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/智能电话本项目实战 3779楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/多线程技术 3780楼

课程分类

百战程序员微信公众号

百战程序员微信小程序

©2014-2025百战汇智(北京)科技有限公司 All Rights Reserved 北京亦庄经济开发区科创十四街 赛蒂国际工业园
网站维护:百战汇智(北京)科技有限公司
京公网安备 11011402011233号    京ICP备18060230号-3    营业执照    经营许可证:京B2-20212637