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

看到下面的问答,有同学问,在讲使用线程池执行大量的Runnable命令时,案例中局部变量n的final为什么要加:

image.png

相关连接:https://blog.csdn.net/xiaobei3120/article/details/12747117

帖子里提到了:

匿名内部类一般定义在一个方法的内部,如果要访问该方法的参数或者方法中定义的变量,则这些参数或者变量必须使用final修饰


虽然匿名内部类定义在方法的内部,但在编译时内部类与外部类中的方法属于同一个级别,外部类中方法的变量或参数只是方法的局部变量,这些变量或者参数的作用域只在当前方法内部有效。但是如果这些变量用final修饰,内部类就可以保存方法变量的备份,即使方法销毁也能保证内部类在访问时不会出现访问不到的错误。

匿名内部类要访问局部变量,但是函数的局部变量在执行完后会立即退出,销毁掉所有临时变量。而产生的匿名内部类可能会保留.

 实际上:匿名内部类并不是直接调用的局部变量n,而是内部类将局部变量n通过自己的构造器备份到了自己的内部,自己内部的方法调用的实际是自己的属性而不是外部类方法的参数。 这样理解就很容易得出为什么要用final了,因为两者从外表看起来是同一个东西,实际上却不是这样,如果内部类改掉了这些参数的值也不可能影响到原参数,然而这样却失去了参数的一致性,因为从编程人员的角度来看他们是同一个东西,如果编程人员在程序设计的时候在内部类中改掉参数的值,但是外部调用的时候又发现值其实没有被改掉,这就让人非常的难以理解和接受,为了避免这种尴尬的问题存在,所以编译器设计人员把内部类能够使用的参数设定为必须是final来规避这种莫名其妙错误的存在。


并且JDK1.8之后匿名内部类访问方法中的局部变量不用加final修饰.


以上是关于匿名内部类中使用方法当中的变量的问题。顺便研究一下匿名内部类:


思考到了一个问题,那我们可以自己在敲代码的时候在匿名内部类里面定义匿名内部类自己的构造器吗 ?

第一步:研究一下,匿名内部类到底是不是执行了父类的构造器。

查资料:https://www.cnblogs.com/wzhanke/p/4779048.html

这个帖子让我们了解到,编译的时候会给匿名内部类生成自己的字节码文件(.class结尾),并且这个字节码文件当中包含外部类的引用,包含自己的构造器,也就是下面的图片代码第5行的构造器。

通过帖子对帖子中的字节码文件的解读。我们可以发现,第12行:

7: invokespecial #2; //Method Test."<init>":(I)V 这个位置 应该是执行了匿名内部类父类Test的构造器。在帖子作者的案例中匿名内部类的父类是Test类。

image.png

也就是说,匿名内部类实际上是执行了父类的构造器的。


第二步: 上一步已经证明了无参数构造器。那么匿名内部类可以调用父类的有参数构造器吗?也就是在创建匿名内部类对象的时候可以传参吗?


思考:我可以在下面的Inter后的小括号,(写着xxx的那个括号)里面传入参数吗?,xxx是我的参数,如下代码:

new Inter(xxx){

...方法的重写等等操作

}

是否能传取决于Inter这个父类,Inter是我们定义的匿名内部类的父类,或者是接口。

如果Inter是一个类,或者抽象类,并且这个类或者抽象类里定义了有参构造器,那么我们在写匿名内部类的时候,匿名内部类就可以传参数。(通过上面的第一步的帖子分析可推断,如果创建匿名内部类对象的时候指定了xxx的值,也就是传参了,那么编译器就会调用父类的带同参数类型的有参构造方法)

如果Inter是一个接口,接口不能定义有参及无参数构造器,所以匿名内部类的那个位置就不可以传参数。


帖子:https://blog.csdn.net/ggjjzhzz/article/details/321272

这个帖子验证了这上面说到的,传参的问题,并且给了总结:

一、由于匿名内部类没有名字,所以它没有构造函数。因为没有构造函数,所以它必须完全借用父类的构造函数来实例化,换言之:匿名内部类完全把创建对象的任务交给了父类去完成。

二、在匿名内部类里创建新的方法没有太大意义,但它可以通过覆盖父类的方法达到神奇效果,如上例所示。这是多态性的体现。

三、因为匿名内部类没有名字,所以无法进行向下的强制类型转换,持有对一个匿名内部类对象引用的变量类型一定是它的直接或间接父类类型。

所以经常在定义匿名内部类的时候会给出下面的格式:感觉好像调用的是父类的构造器。

new 父类构造器(实参列表) | 实现接口() 

{

      //匿名内部类的类体部分

}


第三步:回到一开始的问题: 

思考: 那我们可以自己在敲代码的时候在匿名内部类里面定义匿名内部类自己的构造器吗 ?

答案是:不可以。其实使用的时候编译器帮我们生成了构造器,但是我们自己不能写,因为匿名匿名,匿名的含义就是根本不知道这个类的名字是什么。

匿名内部类里面不可以定义自己的构造器,因为匿名内部类是要扩展或实现父类或接口,所以根本无法为其构造器命名(不能用父类名,哪有在子类里给父类写构造器的说法。而子类根本没有名,更别提用子类名做构造器了)。、因此在匿名内部中要想有构造器作用的部分要就应该是“初始化块”。

也就是说如果要初始化匿名内部类对象,可以采用构造代码块进行初始化。也就是 {  }






JAVA 全系列/第二阶段:JAVA 基础深化和提高/多线程和并发编程(旧) 2686楼

老师我这里出现了不能强制类型转换

package com.hao;


import cn.hao.Server;
import com.ha.servlet.Servlet;

import java.util.List;
import java.util.Map;

/**
 * @Author:@Author
 */
public class WebApp {   //App的意思就是应用程序
    private static ServletContext contxt;
    static {
        contxt=new ServletContext();
        //分别获取对应关系的Map集合
        Map<String,String> servlet=contxt.getServlet();
        Map<String,String> mapping=contxt.getMapping();
        //创建解析XML文件的对象
        WebDom4j we=new WebDom4j();
        we.parse(we.getDocument());//解析XML文件
        //获取解析后的XML文件之后的List集合
        List<Entity> entityList=we.getEntityList();
        List<Mapping> mappingList=we.getMappingList();
        //将List集合中的数据存储到Map集合当中
        for(Entity entity:entityList){
            servlet.put(entity.getName(),entity.getClazz());
        }
        //System.out.println(servlet);
        for(Mapping map:mappingList){
            //遍历url-pattern的集合
            List<String> urlPattern=map.getUrlPattern();
            for(String s:urlPattern){
                mapping.put(s,map.getName());
            }
        }
       // System.out.println(mapping);
    }
    public static Servlet getServlet(String url)  {
        if(url==null||url.trim().equals("")){
            return null;

        }

        try {
            //如果url正确
            //根据url的key获取servlet-name的值 /log=login
            String servletName=contxt.getMapping().get(url);
            //根据servletName得到对应的servlet-class
            String servletClass=contxt.getServlet().get(servletName);
            //使用反射创建Servlet对象


            Class clazz=Class.forName(servletClass);
            //调用无参构造方法创建Servlet对象
            Servlet servlet = (Servlet) clazz.newInstance();
            return  servlet;
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }

        return  null;
    }

    public static void main(String[] args) {
        System.out.println(getServlet("/log"));

    }
}
image.png

JAVA 全系列/第二阶段:JAVA 基础深化和提高/手写服务器项目(旧) 2689楼

看到有同学问的这样一个问题,也引发了我的思考

image.png

于是查了查网上的帖子,发现,不是所有泛型都有所谓的泛型擦除效果的。

https://www.zhihu.com/question/346911525/answer/830285753?utm_source=wechat_session&utm_medium=social&utm_oi=661648077853822976&utm_content=sec

这个帖子就举例了,

 

可以通过反射获取到的泛型信息一定是某个class作为成员变量、方法返回值等位置的具体泛型类型,举例来说:

public class Test<T> {
    private Test<Integer> test;

    private T item;}

在上面的代码中,可以通过反射获取成员test的Integer泛型信息,但是无法获取成员item的实际类型。
帖子中作者追溯到JDK8的源码,发现例子中的test成员编译时会将Integer信息编译进class字节码,从而反射系统就可以获取到这个信息,

也就说在类加载阶段,JVM就将字节码中写死的泛型信息保存了下来。而反射的时候,反射系统自然就可以获取到该信息,从而可以通过getGenericType()来获取到Type信息,从而解析出泛型类型。

而如同例子中定义的item这类泛型引用来说,它们的泛型信息不来自于自身的class,在编译完成通过类型检查后,类型系统中它们就等同于Object,这种泛型是无法通过反射获取的,也就是说这类类型信息被擦除了。

所以真正擦除的应该是例子中的T这类在源代码当中没有具体指定的泛型,才是反射无法获取到的。


看了另一篇帖子,作者给出以下的见解:

  java中的泛型在jdk1.5之后才加进去的,功能基本都是模仿c++中泛型,但是没有c++的泛型那样强大到可在程序执行时都还有泛型功能.这是跟起初java的虚拟机JVM的设计有关,如果要将泛型功能全部添加到JVM中也可以,但是工程量非常大,sun公司不会因为一个泛型的追求而去几乎翻新一个JVM.所以就放弃了.

但是泛型在java中还是有一定的作用的,所以sun公司就在编译器中实现了一个可以擦出泛型的方法.

这么一来java中泛型的生命周期就只能在源代码到编译期了.这就为什么会出现在字节码中可以获取到泛型的Type对象,而又可以绕过泛型对参数化了的集合进行其他数据类型存储的现象.

那为什么sun公司还要提供我们在字节码中获取到泛型的Type对象呢?
       我想这是sun公司为了伱补java泛型的不完美吧!让我们在程序中可以手动实现一个方法去检测接受的数据类型是否就是源代码中定义的泛型类型,如果
不是那么就做相应处理.这样使得程序执行起来更安全,不会因为项目在部署后因传入其他数据类型而照成程序异常等结果发生.


自己的理解:这位作者的意思与上一个作者的相同,就是那些可以被反射读取到的泛型类型信息已经存在字节码文件当中了,但是sun公司不会因为一个泛型而翻新JVM,不想让泛型的功能太过于强大,不然太复杂了,于是它就在编译器中实现了一个可以擦除泛型的方法,让大家认为,【字节码文件当中没有储存泛型的信息,“泛型被擦除了”】,

让大家停止使用泛型的进一步强大的功能,因为没有提供。

这就是为什么反射可以绕过泛型对参数化了的集合进行其他数据类型存储,造成了泛型的生命周期只存在于源代码期间的感受。(就是视频中的第一个案例,使用反射向List<String> list 里面添加Integer类型的123,并且成功了,按理说正常情况下list.add(123)是不能成功添加的,但是反射成功添加了,所以感觉设置泛型好像没有用一样,我们的解释是因为泛型在编译后的字节码文件被"擦除了"。这个案例的结论是泛型的生命周期只存在源代码期间,编译后的字节码文件是没有出现泛型的,而反射是运行期间起作用,两者的生命周期不一样,所以反射操纵不了泛型,所以更不会想到去使用泛型的更强大的功能)。这也是jdk聪明的一点,它让大家认为,泛型信息被擦除了,让大家认为,泛型信息没有存在字节码文件当中。其生命周期在源代码阶段,止步于编译期间。


但是实际上呢,字节码文件当中确实是存储了那些在源代码当中的具体化了的泛型类的信息的,并且类加载阶段,JVM就将字节码中写死的泛型信息保存了下来,这也是为什么我们能够通过反射得到泛型的Type对象。

并且这个帖子的作者说:既然没有实现泛型的更强大的功能,那为什么还要给我们提供Type对象呢?可能是为了程序的安全性考虑,弥补java泛型的不完美,可以让我们在程序中手动实现一个方法去检测接收的数据类型是否就是源代码中定义的泛型类型。
       

以上纯属查资料的个人理解,仅供参考。。。

JAVA 全系列/第二阶段:JAVA 基础深化和提高/反射技术(旧) 2690楼

package com.bjsxt.server;

import java.util.List;
import java.util.Map;

import com.bjsxt.servlet.Servlet;

/**
 * Web应用程序
 * @version 1.0
 *
 */
public class WebApp {
	private static ServletContext contxt;
	static{
		contxt=new ServletContext();
		//分别获取对应关系的Map集合
		Map<String,String> servlet=contxt.getServlet();
		Map<String,String> mapping=contxt.getMapping();
		//创建解析XML文件对象
		WebDom4j web=new WebDom4j();
		web.parse(web.getDocument());//解析xml
		//获取解析XML之后的List集合
		List<Entity> entityList=web.getEntityList();
		List<Mapping> mappingList=web.getMappingList();
		
		//将List集合中的数据存储到Map集合
		for (Entity entity : entityList) {
			servlet.put(entity.getName(), entity.getClazz());
		}
		//	System.out.println(servlet);
		for( Mapping map:mappingList){
			//遍历url-pattern的集合
			List<String> urlPattern=map.getUrlPattern();
			for(String s:urlPattern){
				mapping.put(s, map.getName());
			}
		}
		System.out.println(mapping);
	}
	/**
	 * 通过反射创建Servlet对象
	 * @param url
	 * @return Servlet
	 */
	public static Servlet getServlet(String url){
		if (url==null||url.trim().equals("")) {
			return null;
		}
		//调用无参构造方法创建Servlet对象
		
		try {
			//如果url正确
			//根据url的key获取servlet-name的值 /log=login, /reg=register
			String servletName=contxt.getMapping().get(url);
			//根据servletName得到对应的servlet-class
			String servletClass=contxt.getServlet().get(servletName);  //得到的是一个完整个的包名+类的字符串
			System.out.println(servletClass);
			//使用反射创建 Servlet对象
			Class<?> clazz=Class.forName(servletClass);
			Servlet servlet = (Servlet) clazz.newInstance();
			return servlet;
		} catch (ClassNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		return null;
	}
	public static void main(String[] args) {
		System.out.println(getServlet("/log"));
		System.out.println(getServlet("/login"));		
	}
}

image.png

老师,为什么这里得到的是一个包名+类的字符串,过程具体是怎么实现的?

JAVA 全系列/第二阶段:JAVA 基础深化和提高/手写服务器项目(旧) 2691楼
JAVA 全系列/第二阶段:JAVA 基础深化和提高/网络编程(旧) 2692楼

学了这次的视频,我了解了下一直不太懂的命名空间

命名空间的出现是为了解决dtd不能解决的xml的命名冲突问题,这次课的视频里,xsd文件没有定义自己的命名空间。

然后在book.xml中引入book.xsd的时候属性使用的是

xsi:noNamespaceSchemaLocation="book.xsd"

不指定命名空间,只指定用于模式校验的xsd文档的位置,这种写法,XML模式处理器已经可以校验到xml里的写法问题了。如图:具体报错:

The content of element 'book' is not complete. One of '{price}' is expected.

说明已经检验成功了

image.png

在这里老师用的是

xsi:noNamespaceSchemaLocation="{book.xsd}" 而这种写法在我的idea里面是报红的。 至于视频里老师不报红,我猜测不同工具的不同文件,可能工具没有识别出来语法错误,把验证插件给关闭了。image.png为什么老师要多此一举,使用java的相关类,SchemaFactory进行解析, 是不是因为要让我们了解如果脱离eclipse或者相关IDE,没有XML解析器,在只有xml文件及xsd文件的基础上,java也可以完成使用schema对xml文档的校验。

(扩充:自己测试了一下似乎谷歌浏览器无法完成使用schema对xml文档的校验,不报任何错误。)

这点在视频里没有提到。。这里存在疑问:(******) 


另外给大家一种使用命名空间的方式校验:targetNamespace在book.xsd中给当前命名空间起名字,名称任意:http://www.example.org/books 这个是可以更改的

<?xml version="1.0" encoding="UTF-8" ?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"
targetNamespace="http://www.example.org/books" 
elementFormDefault="qualified">
    <xs:element name="books">
        <xs:complexType>
            <xs:sequence>
                <xs:element name="book" maxOccurs="unbounded">
                    <xs:complexType>
                        <xs:sequence>
                            <xs:element name="name" type="xs:string"></xs:element>
                            <xs:element name="author" type="xs:string"></xs:element>
                            <xs:element name="price" type="xs:double"></xs:element>
                        </xs:sequence>
                        <xs:attribute name="id" type="xs:positiveInteger"></xs:attribute>
                    </xs:complexType>
                </xs:element>
            </xs:sequence>
        </xs:complexType>
    </xs:element>
</xs:schema>

book.xml 使用命名空间http://www.example.org/books

<?xml version="1.0" encoding="UTF-8" ?>
<books xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns="http://www.example.org/books"
        xsi:schemaLocation="http://www.example.org/books book.xsd">
    <book id="1001">
        <name>html的使用</name>
        <author>张三</author>
        <price>51.6</price>
    </book>
    <book id="1002">
        <name>css的使用</name>
        <author>李四</author>
        <price>40.6</price>
    </book>
</books>

注释掉第一个book元素的price元素

结果:具体报错:The content of element 'book' is not complete. One of '{"http://www.example.org/books":price}' is expected.

image.png

想让老师解答一下带****号的那里的疑问!谢谢!

JAVA 全系列/第二阶段:JAVA 基础深化和提高/XML 技术(旧) 2695楼

以下是我自己对UDP 一个发送者与一个接受者,多次咨询,双方互相发送的代码的优化,参考TCP的多线程

首先是Clinet项目

为一个发送者建立两个线程类,一个Send类,一个Receive类。用于发送和接收。

Send类

package com.gz.udp2;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Date;
import java.util.Scanner;

public class Send implements Runnable{
    private DatagramSocket ds;
    private Scanner sc;
    private String destIp;//目标ip
    private int destPort;//目标端口号
    private String role;//角色名称
    private boolean flag=true;


    public Send(DatagramSocket ds,String destIp,int destPort,String role) {
        this.ds = ds;
        this.destIp=destIp;
        this.destPort=destPort;
        this.role=role;
        sc=new Scanner(System.in);
    }
    private String getMessage(){
        return sc.next();
    }
    private void send(String str){
        //发送时间
        String date=new Date().toLocaleString();
        str=date+"   "+role+":"+str;
        byte[] buf=str.getBytes();
        try {
            DatagramPacket dp=new DatagramPacket(buf,buf.length, InetAddress.getByName(destIp),destPort);
            ds.send(dp);
        } catch (IOException e) {
            flag=false;
            if(ds!=null) {
                ds.close();
            }
        }
    }

    @Override
    public void run() {
        while(flag){
            this.send(this.getMessage());
        }

    }
}

Receive类

package com.gz.udp2;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.util.Scanner;

public class Receive implements Runnable{

    private DatagramSocket ds;
    private Scanner sc;
    private boolean flag=true;

    public Receive(DatagramSocket ds) {
        this.ds = ds;
        sc=new Scanner(System.in);
    }
    private String receive(){
        String str="";
        try {
            byte [] buf=new byte[1024];//等待接收数据
            DatagramPacket dp=new DatagramPacket(buf,buf.length);
            ds.receive(dp);
            str=new String(dp.getData(),0,dp.getLength());
        } catch (IOException e) {
            flag=false;
            if(ds!=null){
                ds.close();
            }
        }
        return str;
    }

    @Override
    public void run() {
        while(flag){
            System.out.println(this.receive());
        }
    }
}

Test1类,开启线程的类

package com.gz.udp2;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.util.Scanner;

public class Test1 {
    public static void main(String[] args) throws IOException {
        //DS
        DatagramSocket ds=new DatagramSocket(9777);//自己的端口号

        Send s=new Send(ds,"localhost",9888,"咨询者");
        //目标ip,目标端口号,自己角色
        Receive r=new Receive(ds);

        new Thread(s).start();
        new Thread(r).start();


    }
}


对于接收者,也是拥有两个类,Send类和Receive类,及同样的开启线程类,只不过是端口号不同,角色不同。

接收者的Test2类

package com.gz.udp2;

import java.io.IOException;
import java.net.DatagramSocket;

public class Test2 {
    public static void main(String[] args) throws IOException {
        //DS
        DatagramSocket ds=new DatagramSocket(9888);

        Send s=new Send(ds,"localhost",9777,"客服人员");
        //目标ip,目标端口号,自己角色
        Receive r=new Receive(ds);

        new Thread(s).start();
        new Thread(r).start();


    }
}


结果:成功实现了,客服多次回复:又优化加上了发送时间:

咨询者端:

image.png

客服端:

image.png


想让老师看一下我的代码还有哪里值得优化,还有一个问题,就是,单对单的时候,接收者在回复的时候可以获得发送者发送来的ip与端口号,但是我现在写成了多线程,并且在我的发送者与接收者的测试类里,ip跟端口号是单独指定的,也就是说,当线程开启之前,ip与端口号就固定下来了,下面的交互都是这两个ip与端口号之间的交互。我想问的是有办法像单对单那样,获取到上一个接收过来的数据来自的ip和端口号吗,这样的话就要找到上一个接收线程的DatagramPacket数据包dp,从dp能够获取到上一个发送端的ip及端口号,可是由于多线程,我编辑的逻辑是,每次接收和发送的方法都会创建新的DatagramPacket,当上一个线程调用发送方法以后,随着方法结束,dp数据包也会随之在内存中释放消失。所以,我觉得是完不成根据dp获取上一个发送端的ip及端口号的。因此这也是我的发送端和接收端都要重新制定ip及端口号的原因。

思考到这,(小总结******)我在想UDP是不是就是适合于两个人在通信的时候进行的交互,比如qq聊天,客服聊天,并且这两个人的聊天是指定两个人的单次会话的多次聊天。

而TCP适合群聊,中间的服务器可以将信息转发给其他人,也可以满足随时有人加入群聊,为随时加入群聊的客户端开启处理线程。


以上是我的思考,不知道有没有误解,想跟老师求证最后说的总结(带******号的)的正确性。

还有上面多线程的代码是否可以进一步优化,以及我对我的程序中发送端和接收端都要重新制定ip及端口号的理解是否正确。

还有就是UDP的单次会话的多线程如果真的实现是不是像我写的那样的,发送端和接收端都要重新制定ip及端口号。

JAVA 全系列/第二阶段:JAVA 基础深化和提高/网络编程(旧) 2697楼

客户端不显示

package com.bjsxt.net;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;

/** 
 * <p>Title: TcpServerScoket01</p>  
 * <p>Description: 使用TCP Socket实现单向通信(服务器端)</p>  
 * @author xiaoding
 * @date Jul 29, 2020  
 * @version 1.0 
 */
public class TcpServerScoket01 {
    public static void main(String[] args) throws IOException {
        System.out.println("服务器启动");
        //创建服务器对象
        ServerSocket my = new ServerSocket(9999);
        //使用accept方法监听,服务器端口
        Socket p = my.accept();
        //接受客户端发送的数据
        InputStream fis = p.getInputStream();
        //定义一个中转站
        byte[] buf = new byte[1024];
        int len = 0;    //获取读取的次数
        //将读取的数据打印出来
        while((len = fis.read(buf)) != -1) {
            System.out.println(new String(buf,0,len));
        }
//        System.out.println((char)fis.read());
        
        //发送消息
        OutputStream fos = p.getOutputStream();
        //写入数据
        fos.write("收到了".getBytes());
        //关闭IO流
        if (fos != null) {
            fos.close();
        }
        if (fis != null) {
            fis.close();
        }
    }
}

package com.bjsxt.net;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Arrays;

/** 
 * <p>Title: TestTcpSocket01</p>  
 * <p>Description: 使用Tcp Scoket单向通信(客户端)</p>  
 * @author xiaoding
 * @date Jul 29, 2020  
 * @version 1.0 
 */
public class TestTcpSocket01 {
    public static void main(String[] args) throws UnknownHostException, IOException {
        //创建Socket对象
        Socket my = new Socket("127.0.0.1",9999);
        //发送消息
        OutputStream fis = my.getOutputStream();
        fis.write("你好".getBytes());
        
        //接受消息
        InputStream fos = my.getInputStream();
                
        //定义一个中转站
        byte[] buf = new byte[1024];
        int len = 0;    //获取字符
        while((len = fos.read(buf)) != -1) {
            System.out.println(new String(buf,0,len));
        }
        //关闭IO
        if (fos != null) {
            fos.close();
        }
        if (fis != null) {
            fis.close();
        }
        
        if (my != null) {
            my.close();
        }
    }
}


JAVA 全系列/第二阶段:JAVA 基础深化和提高/网络编程(旧) 2698楼

课程分类

百战程序员微信公众号

百战程序员微信小程序

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