会员可以在此提问,百战程序员老师有问必答
对大家有帮助的问答会被标记为“推荐”
看完课程过来浏览一下别人提的问题,会帮你学得更全面
截止目前,同学们一共提了 133306个问题

说句玩笑话,关于ThreadLocal 分析的视频是认真做的吗?

 

  1. 关于set方法和get方法的顺序问题,视频先介绍的set方法

    set.png

    视屏并未说明为何3个ThreadLocalMap中的Number为何不是同一个对象?因为我们要保证的ThreadLocalMap 键值对对象中的value值为不同的?实际上传入的value对象才是关键吧。

    此set方法只是展示了:将value设置到自己线程的ThreadLocalMap键值对中,其中key为线程对象本身,值为value对象;但是并不能判断对象(图示中的number为不同的对象)

     

2.实际上set方法之所以设置的value为不同的对象,是因为get方法,

QQ截图20190109180939.png

视频中分析的get方法通过线程对象得到value 而get方法源码为:

public T get() {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            ThreadLocalMap.Entry e = map.getEntry(this);
            if (e != null) {
                @SuppressWarnings("unchecked")
                T result = (T)e.value;
                return result;
            }
        }
        return setInitialValue();
    }

正是由于第一次执行get方法时,ThreadLocalMap map = getMap(t);中map为null

所以执行 return setInitialValue();其中setInitialValue()方法源码为:

private T setInitialValue() {
        T value = initialValue();
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
        return value;
    }

是因为第一次执行setInitialValue()方法时,返回的是执行 ,T value = initialValue();中value的值

而 initialValue()方法是我们自己实现的;

也就是说,每个ThreadLocalMap键值对的值的初始值(对象),都是通过initialValue()方法新创建不同对象。因为是不同的对象,但是状态相同,所以才有每个副本的值相同吧.

所以视频的源码分析是故意没讲完全?

    

JAVA 全系列/第二阶段:JAVA 基础深化和提高/多线程和并发编程(旧) 16楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/IO 流技术(旧) 17楼

老师,结合之前电话本儿项目的验证功能做了一些调整, 能跑通 还请老师指点下代码设计上有没有啥毛病

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Scanner;

class SendMsg extends Thread{
    private Socket socket;
    public SendMsg(Socket socket){
        this.socket=socket;
    }

    @Override
    public void run() {
        this.sendMsg();
    }
    private void sendMsg(){
        try(PrintWriter pw=new PrintWriter(this.socket.getOutputStream());
            Scanner scanner=new Scanner(System.in)) {
            while(true){
                String msgSengding=scanner.nextLine();
                pw.println(msgSengding);
                pw.flush();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            if (socket!=null){
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

class ReceiveMsg extends Thread{
    private Socket socket;
    public ReceiveMsg(Socket socket){
        this.socket=socket;
    }

    @Override
    public void run() {
        this.receiveMsg();
    }
    private void receiveMsg(){
        try (BufferedReader br=new BufferedReader(new InputStreamReader(this.socket.getInputStream()))){
            String msgReceive=br.readLine();
            System.out.println(msgReceive);

        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            if (socket!=null){
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
class Regex {
    public Regex() {
    }
    public int orderValidate(){
        String regex="[1-2]{1}";
        Scanner scanner=new Scanner(System.in);
        while (true){
            System.out.println("请输入启动类型(数字1或2)");
            System.out.println("1.启动服务端\t2.启动客户端");
            String order=scanner.nextLine();
            if (order.matches(regex)){
                if ("1".equals(order)){
                    System.out.println("准备启动服务端...");
                }else{
                    System.out.println("准备启动客户端...");
                }
                return Integer.parseInt(order);
            }else {
                System.out.println("输入有误,请重试");
                System.out.println();
            }
        }
    }
}
public class GoodTcp {
    public static void main(String[] args) {
        Regex regex=new Regex();
        int order=regex.orderValidate();
        if (order==1){
            try(ServerSocket serverSocket=new ServerSocket(8888)) {
                System.out.println("服务端启动成功,正在监听端口8888");
                Socket socket=serverSocket.accept();
                System.out.println("已成功建立连接");
                new SendMsg(socket).start();
                new Receive(socket).start();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }else{
            try{
                Socket socket=new Socket(InetAddress.getLocalHost().getHostAddress(),8888);
                System.out.println("已建立连接");
                new SendMsg(socket).start();
                new Receive(socket).start();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

    }
}


JAVA 全系列/第二阶段:JAVA 基础深化和提高/网络编程(旧) 18楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/反射技术(旧) 21楼

老师帮忙看一下我的代码有什么问题吗?

代码如下:

import java.util.Map;
import java.util.Set;
import java.util.TreeMap;

public class TreeMapCompare {
    public static void main(String[] args) {
        Map<User, String> map = new TreeMap<>();
        User u1 = new User("Leo", 21);
        User u2 = new User("Fish", 23);
        User u3 = new User("Angela", 18);
        User u4 = new User("Orange", 20);

        map.put(u1, "Leo");
        map.put(u2, "Fish");
        map.put(u3, "Angela");
        map.put(u4, "Orange");

        Set<User> keys = map.keySet();
        for(User key: keys) {
            System.out.println(key + "---------" + map.get(key));
        }
    }
}

User代码如下:

import java.util.Objects;

public class User implements Comparable<User> {
    private String name;
    private int age;

    public User() {
    }

    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public boolean equals(Object o) {
        System.out.println("equals start");
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        User user = (User) o;
        System.out.println("equals end");
        return age == user.age &&
                Objects.equals(name, user.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, age);
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }

    @Override
    public int compareTo(User o) {
        if(this.getAge() > o.getAge()) {
            return 1;
        }
        if(this.getAge() == o.getAge()) {
            this.getName().compareTo(o.getName());
        }
        return -1;
    }
}

运行结果:

image.png

通过Key无法获取TreeMap中的Value值,请问一下老师,我的代码中存在什么问题吗?

JAVA 全系列/第二阶段:JAVA 基础深化和提高/容器(旧) 24楼

http_server8.zip

老师我这个是怎么回事?

blob.png

blob.png



JAVA 全系列/第二阶段:JAVA 基础深化和提高/手写服务器项目(旧) 25楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/多线程技术(旧) 26楼

sb.append("1").append("2").append("3");
这句不太理解,这句执行的结果是是什么样的?谢谢老师!
package com.itbaizhan;

public class TestString {
    public static void main(String[] args) {
        String s1 = new String("abcdef");
        String s2 = s1.substring(2,4);
        System.out.println(Integer.toHexString(s1.hashCode()));
        System.out.println(Integer.toHexString(s2.hashCode()));

        String str1 = "hello"+"java";//相当于str1 = "hello java";
        //编译器做了优化,直接在编译的时候将字符串进行拼接
        String str2 = "hellojava";
        System.out.println(str1==str2);//true
        String str3 = "hello";
        String str4 = "java";
        //编译的时候不知道变量中存储的是什么,所以没有办法在编译的时候优化
        String str5 = str3+str4;
        System.out.println(str5);
        System.out.println(str2==str5);//false

        String str = "abc";
        StringBuilder sb = new StringBuilder("abc");
        StringBuffer  sb2 = new StringBuffer("abc");

        /**StringBuilder*/
        StringBuilder sb3 = new StringBuilder();
        for(int i=0;i<7;i++){
            sb.append((char)('a'+i));
        }
        System.out.println(sb.toString());//转换成String输出
        sb.append("1").append("2").append("3");
    }
}



JAVA 全系列/第二阶段:JAVA 基础深化和提高/常用类 28楼

class A implements Runnable{
    private Thread b;
    public A(Thread b){
        this.b = b;
   }
    @Override
    public void run() {
        for(int i=0;i<10;i++){
          
System.out.println(Thread.currentThread().g
etName()+" A "+i);
            if(i == 5){
                try {
                    this.b.join();
               } catch
(InterruptedException e) {
                    e.printStackTrace();
               }
           }
            try {
                Thread.sleep(1000);
        } catch (InterruptedException e)
{
                e.printStackTrace();
           }
       }
   }
}
class B implements Runnable{
    @Override
    public void run() {
        for(int i=0;i<20;i++){
          
System.out.println(Thread.currentThread().g
etName()+" B "+i);
            try {
                Thread.sleep(1000);
           } catch (InterruptedException e)
{
                e.printStackTrace();
           }
       }
   }
}
public class TestJoinThread {
    public static void main(String[] args) {
        Thread t1 = new Thread(new B());
        Thread t = new Thread(new A(t1));
        t.start();
        t1.start();

        for(int i=0;i<10;i++){
          
System.out.println(Thread.currentThread().g
etName()+" "+i);
            if(i ==2){
                try {
                    t.join();
               } catch
(InterruptedException e) {
                    e.printStackTrace();
               }
           }
            try {
                Thread.sleep(1000);
           } catch (InterruptedException e)
{
                e.printStackTrace();
           }
       }
   }
}

上面是课程代码


问题:为什么Class b 实现了Runnable接口后,Class A 中的私有属性b的类型要设置为Thread,而不是B?

image.png

JAVA 全系列/第二阶段:JAVA 基础深化和提高/多线程技术 29楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/异常机制 30楼

课程分类

百战程序员微信公众号

百战程序员微信小程序

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