JAVA网络编程,反射及注解知识总结

文章目录

  • 网络编程
    • 软件架构
    • 三要素
    • IP
    • 端口号
    • 协议
      • UDP协议
        • 发送数据
        • 接收数据
        • 三种通信方式
      • TCP协议
        • 客户端
        • 服务器端
        • 三次握手四次挥手
  • 反射
    • 获取字节码文件
    • 获取构造方法
    • 获取成员变量
    • 获取成员方法
    • 反射的作用
  • 动态代理
  • 注解
    • 作用
    • 格式
    • 使用位置
    • 注解的原理
    • 常见注解
    • 元注解
    • 自定义注解
    • 解析注解

网络编程

在网络通信协议下,不同计算机上运行的程序,进行的数据传输。

应用场景:即时通信、网游对战、金融证券、国际贸易、邮件、等等,

Java中可以使用Java.net包下的技术轻松开发出常见的网络应用程序。

软件架构

  1. 客户端-服务器架构(Client-Server Architecture)

客户端服务端模式需要开发客户端

适合定制专业化的办公类软件如:IDEA、网游

  1. 浏览器-服务器架构(Browser-Server Architecture)

浏览器服务端模式不需要开发客户端。

适合移动互联网应用,可以在任何地方随时访问的系统

三要素

三次握手协议保证连接建立

四次挥手
利用这个协议断开连接
而且保证连接通道里面的数据已经处理完毕了

IP

设备在网络中的地址,是唯一的标识。

Internet Protocol(互联网协议)

是分配给上网设备的数字标签,也被称为IP地址。它是一种网络协议,用于在互联网上标识和定位设备,使得数据能够在网络中正确地传输和路由。

通俗地说,IP地址就像是上网设备在网络中的地址,它是唯一的,就像是我们现实生活中的门牌号一样,用于准确定位和识别设备。

1>常见的IP地址分类

  1. IPv4(Internet Protocol version 4):IPv4是目前广泛使用的IP地址标准,它使用32位地址,通常以四个十进制数表示,每个数的取值范围是0到255
    • IPv4:地址通常采用**“点分十进制”**表示,例如192.168.1.1
  2. IPv6(Internet Protocol version 6):使用128位地址,采用了更加复杂的表示方式,通常以八组十六进制数表示
    • IPv6:地址使用**“冒号分16进制”**的方式进行表示,例如2001:0db8:85a3:0000:0000:8a2e:0370:7334

2>IPv4

公网地址(万维网使用)和私有地址(局域网使用)。

192.168.开头的就是私有址址,范围即为192.168.0.0–192.168.255.255,专门为组织机构内部使用,以此节省IP

3>特殊IP地址

127.0.0.1,也可以是localhost:是回送地址也称本地回环地址,也称本机IP,永远只会寻找当前所在本机。

4>InetAddress

InetAddress 是 Java 中用于表示 IP 地址的类。它可以表示 IPv4 地址或 IPv6 地址,并提供了一系列方法用于获取主机名、IP 地址等信息,以及进行网络通信时 IP 地址的解析和操作。

package com.feng.MyInetAddressDemo1;import java.net.InetAddress;
import java.net.UnknownHostException;public class MyInetAddressDemo1 {public static void main(String[] args) throws UnknownHostException {//1.获取InetAdderss对象// 获取本地主机的 InetAddress 对象InetAddress address1 = InetAddress.getLocalHost();System.out.println("本地主机信息:" + address1);System.out.println("本地主机名:" + address1.getHostName());System.out.println("本地 IP 地址:" + address1.getHostAddress());// 根据主机名获取 InetAddress 对象InetAddress address2 = InetAddress.getByName("xuanlaptop");System.out.println("指定主机信息:" + address2);System.out.println("主机名:" + address2.getHostName());System.out.println("IP 地址:" + address2.getHostAddress());}
}

端口号

应用程序在设备中唯一的标识

端口号是应用程序在设备中的唯一标识,用于区分同一设备上的不同网络应用程序。以下是对端口号的详细解释:

1.定义

端口号是网络通信中的一个重要概念,它用于标识设备上运行的应用程序。每个网络应用程序通过一个特定的端口号来接收和发送数据。

2.范围

端口号由两个字节表示,取值范围是0到65535。

3.分类

  • 0~1023:这些端口号被称为“知名端口号”或“系统端口号”,用于一些著名的网络服务和应用,例如:
    • HTTP:80
    • HTTPS:443
    • FTP:21
    • SSH:22
    • SMTP:25
  • 1024~49151:这些端口号被称为“注册端口号”,可以由用户或公司注册使用,但需要遵循相关规范。
  • 49152~65535:这些端口号被称为“动态端口号”或“私有端口号”,通常用于临时性或私有应用,适合用户自行分配。

4.使用规则

  • 一个端口号只能被一个应用程序使用。同一时间一个端口号不能被多个应用程序同时占用。
  • 为了避免冲突,用户和开发者应当选择1024以上的端口号来开发和部署自己的应用程序。

协议

数据在网络中传输的规则,常见的协议有UDP、TCP、http、https、ftp。

计算机网络中,连接和通信的规则被称为网络通信协议。

网络通信协议定义了数据在计算机网络中的传输方式、格式、控制信息等,为网络设备之间的互操作提供了标准和规范。

xieyi

UDP协议

用户数据报协议(User Datagram Protocol)

UDP是面向无连接通信协议。

速度快,有大小限制一次最多发送64K,数据不安全,易丢失数据

DatagramSocket

  • 定义DatagramSocket 类用于创建用于发送和接收数据报文的套接字。

  • 功能

    • 发送数据:通过 send(DatagramPacket p) 方法将数据报文发送到指定的目标地址和端口。
    • 接收数据:通过 receive(DatagramPacket p) 方法接收来自网络的数据报文。
  • 子类:MulticastSocket

    Java中用于组播通信的类。是 DatagramSocket 的子类,因此具有 DatagramSocket 的所有功能,并且能够额外支持组播功能。

DatagramPacket

  • 定义DatagramPacket 类表示数据报文,它封装了发送和接收的数据。
  • 功能
    • 发送数据包:包含要发送的数据、目标地址和端口。
    • 接收数据包:用于接收从网络中到达的数据
发送数据
  1. 创建DatagramSocker对象
  2. 打包数据
  3. 发送数据
  4. 释放资源
package com.feng.MyInetAddressDemo2;import java.io.IOException;
import java.net.*;// 发送消息演示类
public class SendMessageDemo {public static void main(String[] args) throws IOException {//1.创建DatagramSocker对象(快递公司)//绑定端口,以后我们就是通过这个端口往外发送//空参:所有可用的端口中随机一个进行使用//有参:指定端口号进行绑定DatagramSocket ds = new DatagramSocket();//2.打包数据String str = "你好世界!";  // 待发送的字符串数据byte[] bytes = str.getBytes();  // 将字符串转换为字节数组InetAddress address = InetAddress.getByName("xuanlaptop"); // 接收方主机名int port = 10086;  // 接收方端口号DatagramPacket dp = new DatagramPacket(bytes, bytes.length, address, port); // 构造数据包//3.发送数据ds.send(dp);  // 发送数据包//4.释放资源ds.close();  // 关闭DatagramSocket}
}
接收数据
  1. 创建接收端的Datagramsocket对象
  2. 接收打包好的数据
  3. 解析数据包
  4. 释放资源

先接受后发送

package com.feng.MyInetAddressDemo2;import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;public class ReceiveMessageDemo {public static void main(String[] args) throws IOException {//1.创建DatagramSocket对象DatagramSocket ds = new DatagramSocket(10086);//2.接收数据包byte[] bytes = new byte[1024];DatagramPacket dp = new DatagramPacket(bytes, bytes.length);//该方法阻塞,程序执行到这一步时,会在这里死等,等待发送端发送消息ds.receive(dp);//3.解析数据包byte[] data = dp.getData();int len = dp.getLength();InetAddress address = dp.getAddress();int port = dp.getPort();System.out.println("接收到的数据:" + new String(data, 0, len));System.out.println("发送来源于" + address + "该台电脑中" + port + "端口");System.out.println();//4.释放资源ds.close();}
}
三种通信方式

一、单播

单播(Unicast)是指一种一对一的通信方式,即通信的两个端点之间建立一条独立的通信连接,消息从发送端点直接传输到接收端点。在单播通信中,每个数据包都有一个确定的目的地址,只有指定目的地址的接收端才会接收到该数据包。

二、组播

它允许一个设备向多个接收设备发送数据包,而不需要为每个接收设备单独发送一个数据包。

组播地址是IPv4中D类地址,即从 224.0.0.0239.255.255.255

预留的组播地址

  • 224.0.0.0224.0.0.255 是预留的组播地址,用于在本地网络段内使用,不应该被路由到其他网络。这些地址通常用于协议控制信息和本地链路多播。
  • 常见的预留组播地址:
    • 224.0.0.1:所有系统组播地址,所有支持组播的设备都应该监听这个地址。
    • 224.0.0.2:所有路由器组播地址。
    • 224.0.0.5224.0.0.6:分别用于OSPF路由协议的所有OSPF路由器和DR(Designated Router)。

在Java中,可以使用 MulticastSocket 类来实现组播通信

接受代码示例:

public class ReceiveMessageDemo2 {public static void main(String[] args) throws IOException {//1.绑定到指定端口。创建MulticastSocket对象MulticastSocket ms= new MulticastSocket(10000);//2.指定组播地址。将当前本机,添加到224.0.0.1的这一组当中InetAddress address = InetAddress.getByName("224.0.0.1");// 加入组播组ms.joinGroup(address);//3.创建DatagramPacket数据包对象byte []bytes = new byte[1024];DatagramPacket dp = new DatagramPacket(bytes, bytes.length);//4.接受数据ms.receive(dp);//5.解析数据byte[] data = dp.getData();int len = dp.getLength();String ip = dp.getAddress().getHostAddress();String name = dp.getAddress().getHostName();System.out.println("ip:" + ip + " name:" + name);System.out.println(new String(data, 0, len));//6.释放资源ms.close();}}

发送代码示例:

public class MulticastExample {public static void main(String[] args) {try {// 创建 MulticastSocket 实例MulticastSocket multicastSocket = new MulticastSocket(8888);// 加入组播组InetAddress group = InetAddress.getByName("230.0.0.0");multicastSocket.joinGroup(group);// 发送数据报String message = "Hello, Multicast!";byte[] buffer = message.getBytes();DatagramPacket packet = new DatagramPacket(buffer, buffer.length, group, 8888);multicastSocket.send(packet);// 接收数据报byte[] receiveBuffer = new byte[1024];DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);multicastSocket.receive(receivePacket);String receivedMessage = new String(receivePacket.getData(), 0, receivePacket.getLength());System.out.println("Received message: " + receivedMessage);// 离开组播组multicastSocket.leaveGroup(group);// 关闭 MulticastSocketmulticastSocket.close();} catch (IOException e) {e.printStackTrace();}}
}

三、广播

广播地址 255.255.255.255 表示发送到网络中的所有主机。使用 UDP 协议可以很容易地实现广播通信。

将目标IP地址设置为255.255.255.255

public class SendMessageDemo1 {public static void main(String[] args) throws IOException {DatagramSocket ds = new DatagramSocket();while (true) {Scanner sc = new Scanner(System.in);System.out.print("请输入消息:");String str = sc.nextLine();if (str.equals("886")) {break;}byte[] bytes = str.getBytes();InetAddress address = InetAddress.getByName("255.255.255.255");int port = 10086;DatagramPacket dp = new DatagramPacket(bytes, bytes.length, address, port);ds.send(dp);}ds.close();}
}

TCP协议

TCP(Transmission Control Protocol,传输控制协议)是一个面向连接的、可靠的、基于字节流的传输层通信协议。

特点

  • 面向连接:通信双方在传输数据之前必须先建立连接,数据传输结束后要释放连接。
  • 可靠性:TCP通过确认和重传机制确保数据无差错、不丢失、不重复并且按序到达。
  • 流控制:TCP通过滑动窗口协议进行流量控制,防止发送方发送速率过快而导致接收方来不及接收。
  • 拥塞控制:TCP具有拥塞控制机制,以减少因网络中拥塞而导致的数据包丢失。
客户端

客户端(Socket)操作

在Java中,使用Socket类来创建TCP客户端。

  1. 创建客户端的Socket对象
    使用Socket类的构造函数来指定要连接的服务器的IP地址和端口号。

    java复制代码Socket socket = new Socket("server_ip_address", port_number);
    
  2. 获取输出流,写数据
    通过getOutputStream()方法获取一个OutputStream对象,用于向服务器发送数据。

    OutputStream outputStream = socket.getOutputStream();  
    // 使用outputStream.write(...)方法发送数据
    

    注意:这里您提到的Outputstream拼写错误,应该是OutputStream

  3. 释放资源
    完成数据传输后,应该关闭Socket以释放资源。这通常包括关闭输出流和输入流(如果你还打开了一个),然后关闭Socket本身。

    outputStream.close(); // 如果打开了输入流,也要关闭它  
    socket.close();
    

    代码示例:

public class Client {public static void main(String[] args) throws IOException {//1.创建socket对象//链接不上代码报错Socket socket = new Socket("127.0.0.1",10000);//2.从链接通道中获取输出流OutputStream os =socket.getOutputStream();//写出数据os.write("这是中文".getBytes());os.close();socket.close();}
}
服务器端

服务器端ServerSocket操作

在Java中,使用ServerSocket类来创建TCP服务器。

  1. 创建服务器端的Socket对象
    使用ServerSocket类的构造函数来指定服务器要监听的端口号。

    ServerSocket serverSocket = new ServerSocket(port_number);
    
  2. 监听客户端连接,返回一个Socket对象
    调用ServerSocket对象的accept()方法来等待客户端的连接。这个方法会阻塞,直到有客户端连接为止。当客户端连接建立后,accept()方法会返回一个与客户端通信的Socket对象。

    Socket clientSocket = serverSocket.accept();
    

    此时可以使用返回的clientSocket对象与客户端进行通信。

  3. 获取输入流,读数据,并把数据显示在控制台
    通过clientSocket对象的getInputStream()方法获取一个InputStream对象,用于从客户端读取数据。然后,您可以将这些数据转换为可以在控制台上显示的格式。

    InputStream inputStream = clientSocket.getInputStream();  
    // InputStreamReader读中文,BufferedReader提高效率
    BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));  
    int b;
    while ((b = br.read()) != -1) {System.out.print((char) b);
    }
    

    使用BufferedReader来读取文本数据

  4. 释放资源
    完成与客户端的通信后,应该关闭所有打开的流和Socket,以释放系统资源。

    reader.close(); // 关闭读取器  
    inputStream.close(); // 如果直接操作InputStream,则关闭它  
    // 如果有输出流,也要关闭它  
    // clientSocket.close(); // 当与特定客户端的通信结束时关闭  
    // 当服务器不再接受任何连接时,关闭serverSocket  
    // serverSocket.close();
    

    代码示例:

public class Server {public static void main(String[] args) throws IOException {//1.创建ServerSocket对象ServerSocket ss = new ServerSocket(10000);//2.监听客户端链接Socket socket = ss.accept();//3.从连接通道中获取输入流读取数据InputStream is = socket.getInputStream();InputStreamReader isr = new InputStreamReader(is);BufferedReader br = new BufferedReader(isr);int b;while ((b = br.read()) != -1) {System.out.print((char) b);}socket.close();ss.close();}
}
三次握手四次挥手

三次握手(Three-way Handshake)

三次握手是 TCP 协议用于建立连接的过程,其步骤如下:

  1. 客户端发送 SYN 包:客户端向服务器发送一个 SYN(同步)包,表示客户端请求连接。
  2. 服务器回应 SYN-ACK 包:服务器接收到客户端的 SYN 包后,回复一个 SYN-ACK(同步-确认)包,表示服务器已经收到请求,并准备好建立连接。
  3. 客户端发送 ACK 包:客户端再次发送一个 ACK(确认)包,表示客户端确认连接请求,连接建立成功。

这样,连接就建立起来了,客户端和服务器可以开始进行数据传输。

四次挥手(Four-way Handshake)

四次挥手是 TCP 协议用于关闭连接的过程,其步骤如下:

  1. 客户端发送 FIN 包:客户端向服务器发送一个 FIN(结束)包,表示客户端不再发送数据,但仍愿意接收数据。
  2. 服务器回应 ACK 包:服务器接收到客户端的 FIN 包后,发送一个 ACK(确认)包,表示已收到客户端的关闭请求。
  3. 服务器发送 FIN 包:服务器不再发送数据后,向客户端发送一个 FIN 包,表示服务器也准备关闭连接。
  4. 客户端回应 ACK 包:客户端接收到服务器的 FIN 包后,发送一个 ACK 包作为确认,表示客户端已收到服务器的关闭请求。

反射

反射(Reflection)是Java语言中的一种机制,它允许程序在运行时获取有关自身的信息并能动态地调用对象的方法、属性等。通过反射,可以在运行时检查类、接口、字段和方法等结构,还可以实例化对象、调用方法、访问和修改字段。

反射允许对封装类的字段,方法和构造函数的信息进行编程访问。

反射允许对成员变量,成员方法和构造方法的信息进行编程访问

获取字节码文件

一、获取class对象的三种方式

  1. Class.forName(”全类名“);
  2. 类名.class
  3. 对象.getClass();
package com.feng.myreflect;public class MyReflectDemo1 {public static void main(String[] args) throws ClassNotFoundException {//第一种方式(常用)//全类名:包名+类名Class clazz1 = Class.forName("com.feng.myreflect.Student");//第二种方式//当做参数传递Class clazz2 = Student.class;//第三种方式//已经存在类的对象,才可以使用Student s = new Student();Class clazz3 = s.getClass();System.out.println(clazz1);System.out.println(clazz2);System.out.println(clazz3);}
}

获取构造方法

Class类方法描述
Constructor<?>[] getConstructors()返回此Class对象所表示的类的所有公共构造方法对象的数组。
Constructor<?>[] getDeclaredConstructors()返回此Class对象所表示的类的所有构造方法对象(包括私有)的数组。
Constructor<T> getConstructor(Class<?>... parameterTypes)返回此Class对象所表示的类的具有指定参数类型的公共构造方法对象。
Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)返回此Class对象所表示的类的具有指定参数类型的构造方法对象(包括私有)。
Constructor类方法描述
T newInstance(Object... initargs)使用此Constructor对象表示的构造方法来创建该构造方法的类的新实例,并用指定的初始化参数初始化它。
void setAccessible(boolean flag)将此对象的accessible标志设置为指示的布尔值。如果值为true,则允许访问,否则不允许。这可以用于访问私有或受保护的构造方法。
public class MyReflectDemo2 {public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {//1.获取class字节码文件对象Class clazz = Class.forName("com.feng.myreflect.Student");//2.获取构造方法//公共构造方法Constructor[] cons1 = clazz.getConstructors();for (Constructor con : cons1) {System.out.println(con);}System.out.println("===========");//所有构造方法Constructor[] cons2 = clazz.getDeclaredConstructors();for(Constructor con: cons2){System.out.println(con);}System.out.println("===========");//单个获取所有构造方法Constructor con1 = clazz.getDeclaredConstructor();Constructor con2 = clazz.getDeclaredConstructor(int.class);Constructor con3 = clazz.getDeclaredConstructor(String.class);Constructor con4 = clazz.getDeclaredConstructor(String.class, int.class);System.out.println(con1);System.out.println(con2);System.out.println(con3);System.out.println(con4);System.out.println("===========");//获取修饰符整数值int modifiers = con4.getModifiers();System.out.println(modifiers);//获取参数类型Parameter[] parameters = con4.getParameters();for (Parameter parameter : parameters) {System.out.println(parameter);}//临时取消权限校验con4.setAccessible(true);Student stu = (Student) con4.newInstance("张三", 10);System.out.println(stu);}
}

获取成员变量

一、Class类中用于获取成员变量的方法

方法签名描述
Field[] getFields()返回所有公共成员变量对象的数组
Field[] getDeclaredFields()返回所有成员变量对象的数组
Field getField(String name)返回单个公共成员变量对象
Field getDeclaredField(String name)返回单个成员变量对象

二、Field类中用于创建对象的方法

方法签名描述
void set(Object obj, Object value)赋值
Object get(Object obj)获取值
public class MyReflectDemo3 {public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException {Class clazz = Class.forName("com.feng.myreflect.Student");//获取所有的成员变量Field[] fields = clazz.getDeclaredFields();for(Field field:fields){System.out.println(field);}System.out.println("==========");//获取单个成员变量Field field = clazz.getDeclaredField("name");System.out.println(field);//获取权限修饰符int modifiers = field.getModifiers();System.out.println(modifiers);//获取成员变量名String n = field.getName();System.out.println(n);//获取成员变量名Class<?> type = field.getType();System.out.println(type);//获取成员变量记录的值//先创建一个,把私有的名字设置为可访问的Student s = new Student("张三", 10,"男");field.setAccessible(true);String o = (String) field.get(s);System.out.println(o);System.out.println(s);//修改对象里面记录的值field.set(s,"lisi");System.out.println(s);}
}

获取成员方法

一、Class类中用于获取成员方法的方法

方法签名描述
Method[] getMethods()返回所有公共成员方法对象的数组,包括继承的
Method[] getDeclaredMethods()返回所有成员方法对象的数组,不包括继承的
Method getMethod(String name, Class<?>... parameterTypes)返回单个公共成员方法对象
Method getDeclaredMethod(String name, Class<?>... parameterTypes)返回单个成员方法对象

二、Method类中用于创建对象的方法

方法签名描述
Object invoke(Object obj, Object... args)运行方法

参数一:用obj对象调用该方法
参数二:调用方法时传递的参数(如果没有参数就不写)
返回值:方法的返回值(如果没有返回值就不写)

三、示例

public class Test {public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {//1.获取class字节码文件对象Class clazz = Class.forName("com.feng.MyReflectDemo1.Student1");//2.获取里面所有的方法对象(包含父类中的所有公共方法)Method[] methods1 = clazz.getMethods();for (Method method : methods1) {System.out.println(method);}System.out.println("=============");//获取里面所有的方法对象(不能获取父类中的,可以获取本类中的私有方法)Method[] methods2 = clazz.getDeclaredMethods();for (Method method : methods2) {System.out.println(method);}System.out.println("=============");//获取指定单个方法Method m = clazz.getDeclaredMethod("eat", String.class);System.out.println(m);//获取修饰符int modifiers = clazz.getModifiers();System.out.println(modifiers);//获取名字System.out.println(m.getName());//获取方法形参int parameterCount = m.getParameterCount();Parameter[] parameters = m.getParameters();for (Parameter parameter : parameters) {System.out.println(parameter);}//获取抛出的异常Class[] exceptionTypes = m.getExceptionTypes();for (Class exceptionType : exceptionTypes) {System.out.println(exceptionType);}//运行方法   获取方法返回值Student1 s = new Student1();m.setAccessible(true);//参数一:方法的调用者//参数二:调用传递的参数String ret = (String) m.invoke(s, "火锅");System.out.println(ret); }
}

反射的作用

  1. 获取一个类里面所有的信息,获取到了之后,再执行其他的业务逻辑
  2. 结合配置文件,动态的创建对象并调用方法

动态代理

代理模式是一种常用的设计模式,用于为对象提供一个代理,以控制对对象的访问。在Java中,动态代理的主要用途包括:

  • 无侵入式增强:可以在不修改原始类代码的情况下,为对象添加额外的功能。
  • 访问控制:可以在代理中控制对原始对象的访问,例如实现权限检查、日志记录等。
  • 延迟加载:当需要时才加载资源,例如数据库连接或网络请求。

一、代理

在Java中,动态代理是通过接口来实现的。代理对象必须实现与原始对象相同的接口,以便可以调用相同的方法。但代理对象实际上并不执行这些方法的具体实现,而是将方法调用转发给InvocationHandler进行处理。

Java通过接口来保证代理对象与原始对象具有相同的“外观”(即方法签名)。代理对象和原始对象都必须实现相同的接口,这样调用者才能通过接口来调用代理对象的方法。

二、java.lang.reflect.Proxy

Proxy类是Java提供的用于创建动态代理的工具类。它有一个静态方法newProxyInstance,用于生成代理对象。这个方法的参数包括:

  • ClassLoader loader:指定类加载器,用于加载生成的代理类。通常使用原始对象的类加载器。
  • Class<?>[] interfaces:指定代理对象需要实现的接口列表。这些接口决定了代理对象的方法签名。
  • InvocationHandler h:指定代理对象的方法调用处理器。当代理对象的方法被调用时,会调用这个处理器的invoke方法。

注解

Annotation

注解是从JDK 5.0开始引入的新技术

作用

java代码里的特殊标记,比如:@Override、@Test等

作用:让其他程序根据注解信息来决定怎么执行该程序

可以用在类上、构造器上、方法上、成员变量上、参数上、等位置处。

  • 不是程序本身:注解对程序作出解释,类似于注释(comment)。
  • 可被读取:注解可以被其他程序(如编译器等)读取,用于辅助程序处理。

格式

注解以 @注释名 的形式存在于代码中

可以添加一些参数值。例如:

@SuppressWarnings(value="unchecked")

使用位置

注解可以附加在 package、class、method、field 等上面,相当于为它们添加了额外的辅助信息。我们可以通过反射机制编程实现对这些元数据的访问。

注解的原理

注解本质是一个接口,Java中所有注解都是继承了Annotation接口的

Annotation2

@注解(…):其实就是一个实现类对象,实现了该注解以及Annotation接口

常见注解

  1. @Override

    • 定义在 java.lang.Override 中。
    • 适用于修饰方法,表示一个方法声明打算重写超类中的另一个方法声明。
    @Override
    public String toString() {return "Override example";
    }
    
  2. @Deprecated

    • 定义在 java.lang.Deprecated 中。
    • 可以用于修饰方法、属性、类,表示不鼓励程序员使用这样的元素,通常因为它很危险或者存在更好的选择。
    @Deprecated
    public void oldMethod() {// This method is deprecated
    }
    
  3. @SuppressWarnings

    • 定义在 java.lang.SuppressWarnings 中。
    • 用来抑制编译时的警告信息。与前两个注释不同,你需要添加一个参数才能正确使用,这些参数都是已经定义好的,可以选择性地使用。
    @SuppressWarnings("all")
    public void someMethod() {// Suppress all warnings
    }@SuppressWarnings("unchecked")
    public void anotherMethod() {// Suppress unchecked warnings
    }@SuppressWarnings(value={"unchecked", "deprecation"})
    public void multipleWarnings() {// Suppress unchecked and deprecation warnings
    }
    

从 Java 7 开始,额外添加了 3 个注解

  • @SafeVarargs - Java 7 开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。
  • @FunctionalInterface - Java 8 开始支持,标识一个匿名函数或函数式接口。
  • @Repeatable - Java 8 开始支持,标识某注解可以在同一个声明上使用多次。

元注解

元注解:负责注解其他注解

Java定义了4个标准的meta-annotation类型,他们被用来提供对其他annotation类型作说明

这些类型和它们所支持的类在java.ang.annotation包中可以找到(@Target,@Retention,@Documented ,@inherited)

  • @Target: 用于描述注解的使用范围(即:被描述的注解可以用在什么地方)
  • @Retention: 表示需要在什么级别保存该注释信息,用于描述注解的生命周期(SOURCE < CLASS < RUNTIME)
  • @Documented: 说明该注解将被包含在Javadoc中
  • @Inherited: 说明子类可以继承父类中的该注解

示例:

package com.feng.annotation;import java.lang.annotation.*;public class Annotation {@myAnnotationpublic void test() {}
}//定义一个注解
//Target 表示我们的注解可以用在哪些地方
@Target(value = {ElementType.METHOD, ElementType.TYPE})//Retention 表示我们的注解在哪些地方还有效
//SOURCE < CLASS < RUNTIME
@Retention(value = RetentionPolicy.RUNTIME)//Documented 表示是否将我们的注解生成在Javadoc中
@Documented//子类可以继承父类的注解
@Inherited
@interface myAnnotation {}

自定义注解

使用 @interface 可以自定义注解。自定义注解自动继承了 java.lang.annotation.Annotation 接口。

一、定义

@interface 用来声明一个注解,格式如下:

public @interface 注解名 {public 属性类型 属性名() default 默认值;
}

二、注解的参数类型

  • 基本类型(如 int、long、double 等)
  • Class 类型
  • String 类型
  • enum 类型

三、自定义注解示例

public @interface MyAnnotation {// 定义一个int类型的参数,默认值为0int id() default 0;// 定义一个String类型的参数,默认值为空字符串String description() default "";// 定义一个Class类型的参数,默认值为Object.classClass<?> clazz() default Object.class;// 定义一个枚举类型的参数enum Status { ACTIVE, INACTIVE }Status status() default Status.ACTIVE;// 如果只有一个参数成员,参数名一般为valueString value() default "";
}

自定义注解的使用示例:

@MyAnnotation(id = 1, description = "Test Annotation", clazz = String.class, status = MyAnnotation.Status.ACTIVE)
public class TestClass {// 类的内容
}@MyAnnotation("Single parameter value")
public class AnotherTestClass {// 类的内容
}

四、细节

  1. 注解元素必须有值

    注解元素必须有值,在定义注解元素时,通过default来声明参数的默认值,通常使用空字符串0作为默认值

  2. 特殊属性名value

    如果注解中只有一个value属性,使用注解时,value名称可以不写

解析注解

一、注解解析

注解解析是指判断类、方法、成员变量上是否存在注解,并解析注解中的内容。

解析注解时,首先需要获取目标对象的 ClassMethodFieldConstructor 对象,再通过这些对象解析其上的注解。

ClassMethodFieldConstructor都实现了AnnotatedElement接口,它们都拥有解析注解的能力。
AnnotatedElement接口提供了解析注解的方法

方法说明
public Annotation[] getDeclaredAnnotations()获取当前对象上面的注解。
public <T> T getDeclaredAnnotation(Class<T> annotationClass)获取指定的注解对象
public boolean isAnnotationPresent(Class<Annotation> annotationClass)判断当前对象上是否存在某个注解

二、步骤

要解析谁上面的注解,就应该先拿到谁

比如要解析类上面的注解,则应该先获取该类的Class对象,再通过Class对象解析其上面的注解

比如要解析成员方法上的注解,则应该获取到该成员方法的Method对象,再通过Method对象解析其上面的注解

1. 解析类上的注解

  • 获取类的 Class 对象
  • 通过 Class 对象解析其上的注解
// 获取类的Class对象
Class clazz = MyClass.class;// 判断类上是否存在某个注解
if (clazz.isAnnotationPresent(MyAnnotation.class)) {// 获取类上的注解MyAnnotation annotation = clazz.getAnnotation(MyAnnotation.class);// 解析注解内容String value = annotation.value();System.out.println(value);
}

2. 解析方法上的注解

  • 获取方法的 Method 对象
  • 通过 Method 对象解析其上的注解
// 获取方法的Method对象
Method method = clazz.getMethod("myMethod");// 判断方法上是否存在某个注解
if (method.isAnnotationPresent(MyAnnotation.class)) {// 获取方法上的注解MyAnnotation annotation = method.getAnnotation(MyAnnotation.class);// 解析注解内容String value = annotation.value();System.out.println(value);
}

3. 解析成员变量上的注解

  • 获取成员变量的 Field 对象
  • 通过 Field 对象解析其上的注解
// 获取成员变量的Field对象
Field field = clazz.getField("myField");// 判断成员变量上是否存在某个注解
if (field.isAnnotationPresent(MyAnnotation.class)) {// 获取成员变量上的注解MyAnnotation annotation = field.getAnnotation(MyAnnotation.class);// 解析注解内容String value = annotation.value();System.out.println(value);
}

如有错误烦请指正

感谢您的阅读

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/849385.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

自然语言处理(NLP)—— 语言检测器

1. 文章概述 1.1 目的 在本篇文章中&#xff0c;我们将构建一个语言检测器&#xff0c;这是一个能够识别文本语言的简单分类器。这是一个能够识别文本是用哪种语言写的程序。想象一下&#xff0c;你给这个程序一段文字&#xff0c;它就能告诉你这是英语、法语还是其他语言。 …

Moonshot AI API使用(1)-获取MOONSHOT_API_KEY

Moonshot AI 开放平台 用户注册&#xff0c;使用微信扫码登录 把这个key复制下来

用你熟悉的语言就能开发智能合约,Vara Network 以 WASM 解锁未来应用创新

Vara Network 自推出以来&#xff0c;凭借其基于 Gear Protocol 的独特架构和强大的开发工具&#xff0c;为开发者提供了一个高效、安全的智能合约构建平台。Vara Network 通过采用先进的 Actor 模型、持久内存概念和 WebAssembly 技术&#xff0c;实现了异步消息处理、并行计算…

OpenFeign --学习笔记

什么是OpenFeign&#xff1f; OpenFeign可以想象成一座连接客户端&#xff08;服务器&#xff09;和服务器之间的桥梁。在微服务架构中&#xff0c;各个服务之间像小岛屿一样分布在网络上&#xff0c;它们需要相互通信才能协同工作。但是&#xff0c;这些岛屿之间并没有现成的…

SVNCloud 与 Navicat和IDEA的连接

文章目录 SVNCloud 配置Navicat访问云端数据库与IDEA Java jdbc 的连接 SVNCloud 配置 访问网址&#xff1a;SVN注册账号&#xff0c;进入mysql区域&#xff1a; 数据库管理->创建数据库&#xff0c;输入数据库名称和密码&#xff0c;注意&#xff0c;这里的数据库名称实际…

6、后端项目初始化

打开idea后&#xff0c; New Project &#xff0c;用Maven构建 Spring Boot 项目 点击Next后&#xff1a;先勾选两个基本的依赖&#xff0c;后面再手动添加其它需要的依赖 Spring Web: 表示是一个web应用程序 Lombok&#xff1a;写实体类的时候添加Data注解后就会自动加上g…

Linux网络编程:网络层协议|IP

目录 前言&#xff1a; 1.IP协议 1.1.IP协议格式 1.2.网段划分 1.2.1.知识引入 1.2.2.IP地址划分和子网掩码 1.3.IP地址分类 1.3.1.特殊IP地址 ​编辑 1.3.2.私有IP和公网IP 1.3.3.浅谈NAT技术 1.4.路由 1.4.1.什么是路由 1.4.2.路由表 1.5.网络层数据切片和组装…

MYSQL基础_01_数据库概述

第01章_数据库概述 1. 为什么要使用数据库 持久化(persistence)&#xff1a;把数据保存到可掉电式存储设备中以供之后使用。大多数情况下&#xff0c;特别是企业级应用&#xff0c;数据持久化意味着将内存中的数据保存到硬盘上加以”固化”&#xff0c;而持久化的实现过程大多…

三招搞定“找不到msvcp140.dll无法继续执行代码”问题

在计算机使用过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中之一就是“找不到msvcp140.dll”。那么&#xff0c;这个错误提示到底是什么意思呢&#xff1f;又该如何解决这个问题呢&#xff1f;本文将从以下几个方面进行详细阐述。 一&#xff0c;msvcp140.dll文…

如何轻松将Android同步到 PC? 【6个最适合你的方法!】

尽管许多Android手机都配备了充足的数据存储空间&#xff0c;但将手机中的重要数据备份到电脑上始终是明智之举&#xff0c;以防止数据丢失。那么&#xff0c;如何将Android手机与电脑同步呢&#xff1f;虽然大多数Android用户可能会使用USB线或蓝牙传输文件到PC&#xff0c;但…

水电表自动抄表系统

1.简述 水电表自动抄表系统是一种现代化智能化管理系统&#xff0c;它利用先进的物联网&#xff0c;完成了远程控制、即时、零接触的水电表读值收集&#xff0c;大大提升了公共事业服务项目的效率和准确性。该系统不仅减少了人工抄表工作量&#xff0c;还避免了人为失误&#…

动画制作软件有哪些?最后一款动画渲染必备

探索动画的无限可能&#xff0c;从简单的线条到复杂的三维世界&#xff0c;动画制作软件是艺术家们实现创意的得力助手。无论是手绘动画的细腻笔触&#xff0c;还是3D建模的立体展现&#xff0c;这些软件都极大地丰富了动画制作的手法和表现形式。接下来&#xff0c;我们将介绍…

应急管理大数据指挥中心解决方案(51页PPT)

方案介绍&#xff1a; 本应急管理大数据指挥中心解决方案充分利用了大数据技术的优势&#xff0c;实现了信息的快速收集、分析和决策支持。通过数据融合、协同指挥、智慧化决策和平台建设等方面的努力&#xff0c;提高了应急管理的效率和准确性&#xff0c;为应对各类突发事件…

珠海鸿瑞毛利率持续下滑:核心产品销量大降,偿债能力偏弱

《港湾商业观察》黄懿 日前&#xff0c;珠海市鸿瑞信息技术股份有限公司&#xff08;下称“珠海鸿瑞”&#xff09;收到了北京证券交易所发出的第三轮审核问询函。 此前&#xff0c;2020年11月&#xff0c;珠海鸿瑞曾向深交所报送上市申请。IPO申请文件获受理后&#xff0c;珠…

普通人如何找到合适的创业方向

作为普通人创业&#xff0c;试错的成本是很高的&#xff0c;哪怕是低成本创业&#xff0c;你起码也得花费大量的时间&#xff0c;所以&#xff0c;在方向的选择上&#xff0c;我们需要谨慎&#xff0c;避免因为方向的选择错误&#xff0c;导致陷入进退两难的地步。 创业方向如何…

男士内裤比较好的品牌有哪些?五款物超所值的男款内裤安利

挑选男士内裤时&#xff0c;哪一款更合适呢&#xff1f;这个问题想必让许多人感到困惑。现在市场上的男士内裤种类繁多&#xff0c;确实让人眼花缭乱&#xff0c;不知从何下手。为了帮助大家解决这一难题&#xff0c;今天特地为大家整理了一些选购男士内裤的技巧&#xff0c;并…

怎么样选择合适的erp|如何选择公司erp

在当今高度竞争的商业环境中&#xff0c;企业资源计划(ERP)系统对于企业管理各种业务流程至关重要。选择合适的ERP系统可以极大地提高效率、降低成本并促进增长。然而&#xff0c;市场上有众多的ERP解决方案&#xff0c;如何选择适合您企业的系统是一个复杂的过程。在本篇文章中…

CSS 渐变背景在线生成工具源码

CSS 渐变背景在线生成工具源码 效果图部分源码领取源码下期更新预报 效果图 部分源码 :root {--transition-fast: 0.2s ease;--box-shadow-light: 0 4px 1px rgba(15, 12, 12, 0.349);--box-shadow-heavy: 0px 5px 1px rgb(15, 12, 12);--border-radius-full: 9999px;--border…

你工作中最推荐的 C/C++ 程序库有哪些,为什么?

我主要做计算力学&#xff0c;说说平时用的一些c库1、前处理划网格用netgen&#xff0c;非结构网格功能强大&#xff0c;有可执行的软件和供调用的库&#xff0c;使用方便。 刚好我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「C的资料从专业入门到高级教程」&…

转让海南投资集团公司变更条件和流程

近年来&#xff0c;随着经济全球化的加速&#xff0c;无地域限制的投资公司&#xff0c;成为了越来越多企业的选择。作为一种快速、高效的发展模式&#xff0c;投资公司不仅可以帮助企业快速进入新市场&#xff0c;还可以获取更多资源和资本支持。本文将从基本概念、研究进展和…