java 代理

代理的作用,就是在原有类的代码不发生改动的情况下,添加新功能。起到一个修饰器的作用。

假设我们现在有个类,用于打印helloworld

class hello{
public void gogo(){System.out.println("Hello Wrold");}
}

我们想在不改动hello类的前提下,在输出helloworld时同时输出当前时间到文件,达到日志的功能,该如何解决这个问题呢?以这个问题为切入点,开始学习代理。

静态代理

静态代理很简单,不用接触什么新技术。就是创建一个”继承已有的一个类
“的类,通过重写父类的方法,达到不改动原有类的基础上增添新功能。纸上得来终觉浅,用代码来说明一下吧。

以刚刚的输出helloworld的类为例。我们编写一个它的子类,并在其中重写其gogo方法,添加日志生成功能

class hello{
public void gogo(){System.out.println("Hello Wrold");}
}

class static_hello_proxy extends hello{
hello hello_obj;
static_hello_proxy(hello hello){
this.hello_obj = hello;
}

@Override
public void gogo(){
log();
hello_obj.gogo();
}

public void log(){
Date date = new Date();
String value = "date:"+date+"\n";
FileWriter writer = null;
try {
writer = new FileWriter(new File("log.txt"),true);
writer.write(value);
writer.flush();
} catch (IOException e) {
e.printStackTrace();
}

}


}

然后主函数中我们只需调用static_hello_proxy中的gogo()方法就能达到日志功能了。这就是静态代理。
静态代理的缺点显而易见,如果我们想要代理不同的类,就要写出不同的静态代理类出来,同时我们也可能会遇到需要多个代理类来增添实现一个类的不同功能,有需要定义一大堆类出来。不容易维护,所以动态代理就诞生了,它很好的解决了静态代理会产生大量代理类的难题。

动态代理

使用动态代理,可以不用创建代理类,非常的方便。

动态代理由Proxy.newProxyInstance方法实现。

QQ截图20210219150752

我们看看它的三个参数。
loader,顾名思义,就是被代理类的类加载器。
interfaces 被代理类所需要实现的接口,可通过 类对象.getinterfaces()获得
h 即 InvocationHandler接口类 的实现类,用于实现代理增添的方法

我们看看这个 InvocationHandler 类的结构。这个类只有一个invoke方法,且这个类是接口类。这个invoke方法就用于存放我们的增添的功能。

QQ截图20210219150801

proxy即代理对象,method即对象中的某个方法,args即方法中的参数。

ok,接下来把刚刚那个类动态代理一下吧。但是需要注意的是动态代理只能代理接口实现类。

import java.io.File;
import java.io.FileInputStream;
import java.io.FileWriter;
import java.io.IOException;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Date;

public class test {
public static void main(String[] args){
hello hello_obj = new hello();
ClassLoader Loader = hello_obj.getClass().getClassLoader();
Class<?>[] Interfaces = hello_obj.getClass().getInterfaces();
InvocationHandler ih = new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if(method.getName()=="gogo"){
FileWriter writer = new FileWriter("log.txt");
writer.write("date:"+new Date()+"|by 动态代理\n");
writer.flush();
hello_obj.gogo();
}
return null;
}
};
hello_interface proxy = (hello_interface) Proxy.newProxyInstance(Loader,Interfaces,ih); //注意这里数据类型是接口的数据类型
proxy.gogo();
}
}

interface hello_interface{
public void gogo();
}

class hello implements hello_interface{
public void gogo(){System.out.println("Hello Wrold");}
}

OK,ALL DOWN

JNI

JNI是JAVA的一个接口,用来实现c/c++调用,为JAVA提供了一个操纵底层的能力。

JNI HelloWorld

java IDE:idea C IDE:vs2019

1.写好JAVA文件,定义好native方法。

public class test {
static{
System.loadLibrary("Dll1");
}
public native void hello();
public static void main(String[] args) {
test a = new test();
a.hello();
}
}

2.使用javac指令为java文件的各native方法生成c语言头文件。

javac -cp . test.java -h .

执行后会在当前目录下生成一个头文件

QQ截图20210219150831

3.用vscode创建一个dll项目,将test.h与jdk include目录下的jni.h以及jdk/include/win32/jni_md.h统统移动到项目文件目录下。然后在vscode头文件里添加现有项,把刚刚放到项目文件下的3个头文件添加进去。

4.在c或c++中定义native方法的c/c++原型。

QQ截图20210219150842

通过test.h我们可以看到哪些方法需要我们定义。这里是Java_test_hello方法。

我们随便定义一下

#include "test.h"

JNIEXPORT void JNICALL Java_test_hello(JNIEnv* a, jobject b) {
printf("Hello,World");
}

编译为dll文件。

5.在idea里为当前项目指定依赖dll文件路径

先点击箭头指向处的编辑选项,然后在虚拟机选项里填上-Djava.library.path=你的dll路径

QQ截图20210219151203

6.把dll文件拖进刚刚配置好的依赖路径,运行

QQ截图20210219151203

JNI 数据类型

QQ截图20210219151238

JNI类型数据在c文件里当作传入参数和返回结果,参数传入后其数据类型会被用指定方法转换为C类型然后被处理,处理完毕后使用指定方法转换为JNI类型返回给JAVA。

这里介绍一下c中JNI类型与c类型互转的函数

jstring转char*:env->GetStringUTFChars(str, &jsCopy)

char*转jstring: env->NewStringUTF("Hello...")

字符串资源释放: env->ReleaseStringUTFChars(javaString, p);

JNI 实现本地命令执行

相当于一个加强版的helloworld.

首先我们看一下,如果在c++中我们想执行本地命令执行该如何执行

#include<iostream>
#include<stdio.h>
#include <string>

std::string gogo(const char* cmd) {
const char* cmd = cmd;
FILE* fp = _popen(cmd, "r");
if (fp != NULL) {
std::string result;
char buf[128];
while (fgets(buf, sizeof(buf), fp) != NULL) {
result += buf;
}
return result;
}
}

int main() {
std::string result = gogo("dir");
std::cout << result;
}

那么我们在JNI里如何实现呢?其实也差不多.

我们现在写好java代码

public class test {
static{
System.loadLibrary("Dll2");
}
public native String gogo(String cmd);
public static void main(String[] args) {
test a = new test();
String str = a.gogo("dir");
System.out.println(str);
}
}

写好c++代码

#include "test.h"
#include<iostream>
#include<stdio.h>
#include <string>
JNIEXPORT jstring JNICALL Java_test_gogo(JNIEnv* env, jobject obj, jstring str) {
const char* cmd = env->GetStringUTFChars(str, 0);
FILE* fd = _popen(cmd, "r");
if (fd != NULL) {
std::string result;
char buf[128];
while (fgets(buf, sizeof(buf), fd) != NULL) {
result += buf;
}
_pclose(fd);
return env->NewStringUTF(result.c_str());
}
}

运行结果

QQ截图20210219151248

类加载器

ClassLoader

在JVM类加载器中最顶层的是Bootstrap ClassLoader(引导类加载器)Extension ClassLoader(扩展类加载器)App ClassLoader(系统类加载器)AppClassLoader是默认的类加载器,如果类加载时我们不指定类加载器的情况下,默认会使用AppClassLoader加载类,ClassLoader.getSystemClassLoader()返回的系统类加载器也是AppClassLoader

ClassLoader的主要方法有以下几个:

  1. loadClass(加载指定的Java类)
  2. findClass(查找指定的Java类)
  3. findLoadedClass(查找JVM已经加载过的类)
  4. defineClass(从byte[]获得一个Java类)
  5. resolveClass(链接指定的Java类)

如何通过ClassLoader获取一个类对象?很简单

Class a = ClassLoader.getSystemClassLoader().loadClass("java.lang.Runtime");

它对一个类的加载流程如下:

  1. ClassLoader调用loadClass(String name)方法加载指定类
  2. 调用findLoadedClass检查指定的类是否已经初始化,若已初始化则直接返回类对象
  3. 如果创建ClassLoader时传入父类加载器,则使用父类加载器加载指定类,否则使用JVM的Bootstrap ClassLoader(引导类加载器)加载
  4. 如果以上步骤没有完成加载,则使用findClass方法尝试加载指定类
  5. 如果当前ClassLoader类没有重写findClass方法则直接返回异常。若重写了该方法且通过findClass找到了传入的类名的对应的类字节码,那么就会使用defineClass去JVM注册该类
  6. 如果loadClass调用时传入resolve的参数为true,则那么还需要调用resolveClass方法连接类。该参数默认为false
  7. 返回被JVM加载后的指定类的类对象

ClassLoader自定义

java.lang.ClassLoader 是所有类加载器的父类,我们可以通过重写其findClass方法来实现自定义ClassLoader。

我们试着自定义一个,当对loadClass传入com.anbai.sec.classloader.TestHelloWorld 时会直接返回一个在加载器里已经定义好了的对应类的类对象。

class TestClassLoader extends  ClassLoader{
String testClassName="com.anbai.sec.classloader.TestHelloWorld";
byte[] testClassBytes = new byte[]{-54, -2, -70, -66, 0, 0, 0, 51, 0, 17, 10, 0, 4, 0, 13, 8, 0, 14, 7, 0, 15, 7, 0,
16, 1, 0, 6, 60, 105, 110, 105, 116, 62, 1, 0, 3, 40, 41, 86, 1, 0, 4, 67, 111, 100,
101, 1, 0, 15, 76, 105, 110, 101, 78, 117, 109, 98, 101, 114, 84, 97, 98, 108, 101,
1, 0, 5, 104, 101, 108, 108, 111, 1, 0, 20, 40, 41, 76, 106, 97, 118, 97, 47, 108,
97, 110, 103, 47, 83, 116, 114, 105, 110, 103, 59, 1, 0, 10, 83, 111, 117, 114, 99,
101, 70, 105, 108, 101, 1, 0, 19, 84, 101, 115, 116, 72, 101, 108, 108, 111, 87, 111,
114, 108, 100, 46, 106, 97, 118, 97, 12, 0, 5, 0, 6, 1, 0, 12, 72, 101, 108, 108, 111,
32, 87, 111, 114, 108, 100, 126, 1, 0, 40, 99, 111, 109, 47, 97, 110, 98, 97, 105, 47,
115, 101, 99, 47, 99, 108, 97, 115, 115, 108, 111, 97, 100, 101, 114, 47, 84, 101, 115,
116, 72, 101, 108, 108, 111, 87, 111, 114, 108, 100, 1, 0, 16, 106, 97, 118, 97, 47, 108,
97, 110, 103, 47, 79, 98, 106, 101, 99, 116, 0, 33, 0, 3, 0, 4, 0, 0, 0, 0, 0, 2, 0, 1,
0, 5, 0, 6, 0, 1, 0, 7, 0, 0, 0, 29, 0, 1, 0, 1, 0, 0, 0, 5, 42, -73, 0, 1, -79, 0, 0, 0,
1, 0, 8, 0, 0, 0, 6, 0, 1, 0, 0, 0, 7, 0, 1, 0, 9, 0, 10, 0, 1, 0, 7, 0, 0, 0, 27, 0, 1,
0, 1, 0, 0, 0, 3, 18, 2, -80, 0, 0, 0, 1, 0, 8, 0, 0, 0, 6, 0, 1, 0, 0, 0, 10, 0, 1, 0, 11,
0, 0, 0, 2, 0, 12}; //这部分是类字节码,是我从别人那里嫖的

@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
if(name.equals("com.anbai.sec.classloader.TestHelloWorld")){
return defineClass(testClassName,testClassBytes,0,testClassBytes.length);
}
else{
return super.findClass(name);
}
}
}

然后主函数我们写

public static void main(String[] args) {
try {
TestClassLoader Loader = new TestClassLoader();
Class clazz = Loader.loadClass("com.anbai.sec.classloader.TestHelloWorld");
Object Instance = clazz.newInstance();
Method method = Instance.getClass().getMethod("hello");
String str = (String) method.invoke(Instance);
System.out.println(str);
}catch (Exception e){}
}

经测试程序正常运行。我们的自定义ClassLoader就写好了。当然还有更多花里胡哨的重写方法,我们这里只是知道怎么重写即可。

URLClassLoader

URLClassLoader也是ClassLoader类下的一个重写类。这个类很好用,它可以通过网络协议获取远程的jar包,然后通过类加载器去获得其中的指定类的类对象。试试吧。

流程是先定义一个URL对象指向我们的jar包,然后实例化一个URLClassLoader对象,然后通过loadClass加载某个类获得类对象。

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;

public class test {
public static void main(String[] args){
try{
// Class clazz = ClassLoader.getSystemClassLoader().loadClass("fuck");
// Method method = clazz.getMethod("func1",int.class);
// method.invoke(clazz.newInstance(),123);
URL url = new URL("http://127.0.0.1/LOL.jar");
URLClassLoader Loader = new URLClassLoader(new URL[]{url});
Class clazz = Loader.loadClass("fuck");
Method method = clazz.getMethod("func1",int.class);
method.setAccessible(true);
method.invoke(clazz.newInstance(),123);

}
catch (Exception e){
e.printStackTrace();
}
}
}

然后我们jar包里的class是这个.JAVA的编译文件

public class fuck{
public void func1(int input1){
System.out.print(input1);
}
}

执行结果

QQ截图20210219151315

反射

反射的基础

反射是java的一个特性,用于获取类的详细信息(方法,变量),并可以执行类中的方法。

获得一个类的类对象

要获取类的详细信息或执行其中的方法,首先肯定是要获取到那个类的类对象

方法一:我们需要创建一个Class类型的变量,用于接收Class.forname(“类”)返回的类对象。这个方法必须通过try..catch 来处理其中ClassNotFoundException

try{
Class clazz = Class.forname("java.lang.String");}
catch(ClassNotFoundException e){}

方法二:我们实例化一个类的对象出来,然后通过 对象.getClass()获得其类对象

String a = new String;
Class clazz = a.getclass();

方法三:使用.class

Class clazz = String.class;

这样,我们就获得一个指定类的类对象了。很简单。我们可以对一个类对象使用getName()方法获取其类名。

System.out.println(clazz.getName());

获取一个类对象的所有成员

获取属性并修改

对类对象使用 getFields() 或 getDeclaredFields() 方法即可获得属性数组,区别在于前者只能获取公有属性,后者能获取私有属性。
然后遍历属性数组,通过对每一项执行getName()获取属性名,get(实例化的对象)获得属性值
然后对某一项进行.set()对其进行修改
这里需要提的是,如果要获得或修改私有属性的值的时候,需要对私有属性使用setAccessible(true)来实现私有访问

import java.lang.reflect.Field;

public class test {
public static void main(String[] args){
Class clazz = (new abc()).getClass();
abc test = new abc();
Field[] Fields = clazz.getDeclaredFields();
for(Field Field:Fields){ //遍历Fields数组
try { //执行get()方法时需抛出IllegalAccessException错误
Field.setAccessible(true); //对数组中的每一项实现私有访问
System.out.print(Field.getName());
Object value = Field.get(new abc());
System.out.println(":" + value);
Field.set(test,"new"); //修改test对象中的变量
}
catch (Exception e){

}
}
System.out.println(test.a);
}

}

class abc{
public String a =new String("tom and mary");
private String b = new String("abcdefg");
}

运行结果

QQ截图20210219151405

另外,你也可以通过getFiled(“属性名”)来获取特定属性的Filed对象, getDeclaredFields() 同理

获取方法并执行

接下来是获取方法的一些信息
我们通过getMethod获取方法对象,然后通过getName获取方法名,getReturnType获取返回值类型,getParameterTypes获取传入的参数类型

import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class test {
public static void main(String[] args){
Class clazz = (new abc()).getClass();
Method[] methods = clazz.getDeclaredMethods();
for(Method method:methods){
try{
System.out.print(method.getName());
System.out.print("|retrunType:"+method.getReturnType()+"|");
Class[] ParameterTypes = method.getParameterTypes();
for(Class ParameterType:ParameterTypes){
System.out.print("ParamType:"+ParameterType.getName());
}
System.out.println("");
}
catch (Exception e){

}
}
}
}

class abc{
public int a =1;
private String b = new String("abcdefg");

public void func1(int input1){
System.out.print(input1);
}
private void func2(int input1){
System.out.print(input1);
}
}

接下来是对method对象执行invoke方法调用,通过传入参数,指定对象,即可调用该方法。我在上述代码中添加了修改

import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class test {
public static void main(String[] args){
Class clazz = (new abc()).getClass();
Method[] methods = clazz.getDeclaredMethods();
for(Method method:methods){
try{
System.out.print(method.getName());
System.out.print("|retrunType:"+method.getReturnType()+"|");
Class[] ParameterTypes = method.getParameterTypes();
for(Class ParameterType:ParameterTypes){
System.out.print("ParamType:"+ParameterType.getName());
}
System.out.println("");
method.setAccessible(true); //打开私有访问
method.invoke(new abc(),2); //new 这里invoke第一个参数指定实例化对象,之后的参数代表传入方法的参数
}
catch (Exception e){

}
}
}
}

class abc{
public int a =1;
private String b = new String("abcdefg");

public void func1(int input1){
System.out.print(input1);
}
private void func2(int input1){
System.out.print(input1);
}
}

获取构造方法并执行

获得构造方法,主要是通过 getConstructors()和getDeclaredConstructors() ,后者能访问私有对象,下面是用遍历法获得所有构造函数信息

import java.lang.reflect.Constructor;

public class test {
public static void main(String[] args){
try{
Class clazz = Class.forName("abc");
Constructor[] conArray = clazz.getDeclaredConstructors();
for (Constructor a:conArray){
a.setAccessible(true);
System.out.println(a);
}
}
catch(Exception e){
}
}
}

class abc{
public abc(String name,int age){
System.out.println("姓名:"+name+"年龄:"+ age);
}
private abc(int age){
System.out.println("私有的构造方法 年龄:"+ age);
}

}

img
运行结果

我们刚才提过,getFiled()附带参数可以指定访问某一个属性,同理,getConstructor也一样,我们想要执行某一个类的构造方法,往往这个方法更实用
要执行一个类的构造方法,那我们需要创建一个该类的实例化对象,这个过程我们用newInstance()方法实现

import java.lang.reflect.Constructor;

public class test {
public static void main(String[] args){
try{
Class clazz = Class.forName("abc");

Constructor test = clazz.getDeclaredConstructor(int.class); //指定参数,即可从多个重载的构造函数指定到某个具体的构造函数
test.setAccessible(true);
test.newInstance(12); //私有构造方法只需调用一下newInstance传入参数即可

Constructor test2 = clazz.getDeclaredConstructor(String.class,int.class);
test2.newInstance("tom",15); //公有构造方法也一样
}
catch(Exception e){
}
}
}

class abc{
public abc(String name,int age){
System.out.println("姓名:"+name+"年龄:"+ age);
}
private abc(int age){
System.out.println("私有的构造方法 年龄:"+ age);
}

}

执行结果

QQ截图20210219151513

反射的进阶与安全

Class.forName 实质与类初始化

Class.forName(“…”) 常被我们拿来获得类对象,但是实际上,Class.forname有三个参数,只不过我们默认输第一个参数:类名就能完成工作了。

这是Class.fornName函数原型
public static Class<?> forName(String name, boolean initialize,
ClassLoader loader)

Class.forName(className)
// 等于
Class.forName(className, true, currentLoader)

这里我们看见有三个参数,第一个参数是指定类名就不多讲了,第二个参数是决定类是否初始化(这个稍后会详细阐明),第三个是ClassLoader类加载器(告诉JVM如何加载这个类这里不展开说)

关于一个类的初始化,有三种操作可以实现:构造方法,空块和static块,就像这样

public class TrainPrint {
{
System.out.printf("Empty block initial %s\n", this.getClass());
}
static {
System.out.printf("Static initial %s\n", TrainPrint.class);
}
public TrainPrint() {
System.out.printf("Initial %s\n", this.getClass());
}
}

那么执行顺序是如何呢,在引入父类的情况下又是如何呢?我们写个demo看看

import java.lang.reflect.Constructor;

public class test {
public static void main(String[] args){
abc a = new abc();
}
}




class b{
{
System.out.println("b空块已执行");
}

static{
System.out.println(" b static块已执行");
}
public b(){
System.out.println("b类构造方法已执行");
}
}


class abc extends b{
{
System.out.println("a空块已执行");
}

static{
System.out.println(" a static块已执行");
}

public abc() {
System.out.println("a初始化方法已执行");
}
}

QQ截图20210219151831

我们可以清晰的看到执行顺序,对继承类来说,
1.会先执行父类static块
2.执行自己的static块
3.执行父类空快
4.执行父类构造方法
5.执行自己空快
6.执行自己构造方法

对于一个类来说,执行顺序则是
1.执行static块
2.执行空快
3.执行构造方法

那么对于Class.forName指定的是否进行类初始化参数,指的是哪个部分?static块,空块还是构造方法?答案是只会执行static块里的,且会优先执行父类的static块。
我们把上面代码中的主函数替换为

try {
Class clazz = Class.forName("abc");
}
catch (Exception e){}

执行一下,看结果。发现只执行了static块

QQ截图20210219151840

也就是说,Class.forName()会默认执行类的static代码块,是个比较危险的信号。
另外一提,xxx.class.newinstance() 会自动调用类的构造方法。

Runtime执行命令解析

一般来说,我们调用Runtime类来执行命令时的指定是这样的

Runtime.getRuntime().exe()

我们到源码里分析这段代码

QQ截图20210219151850

可见,当我们对Runtime类执行getRuntime()时会得到一个Runtime对象,然后我们就可以调用我们的exec方法了
同时,Runtime() 被private修饰符修饰了,这说明我们无法通过Runtime a = new Runtime()来实现一个Runtime的对象。
所以说,我们正常地使用runtime来执行命令只能依靠以上代码。

那么我们要是想要依靠反射来写一个Runtime执行任意命令的payload,那么该如何写呢?

先来一个错误示范,当我们用常规的思路去实现时。我们直接调用Runtime类里的exec方法,然后通过newInstance来实例化一个Runtime对象

Class clazz = Class.forName("java.lang.Runtime");
Method method = clazz.getMethod("exec", String.class);
method.setAccessible(true);
method.invoke(clazz.newInstance(),"calc.exe");

最终结果则是报错:class test cannot access a member of class java.lang.Runtime (in module java.base) with modifiers “private”
看来通过反射也不能直接调用exec方法,或者说不能实例化Runtime对象。
那么正确思路该是什么呢?应该是先调用getRuntime获得Runtime对象,然后再调用exec方法

Class clazz = Class.forName("java.lang.Runtime");
Method MgetRuntime = clazz.getMethod("getRuntime");
Object runtime = MgetRuntime.invoke(clazz); //对类对象使用getRuntime(),其实就相当于Runtime.getRuntime()。这种方式仅限static方法
Method Mexec = clazz.getMethod("exec", String.class);
Mexec.invoke(runtime,"calc.exe");

最终弹出计算器。

ProcessBuilder执行命令的反射实现

除了Runtime以外,还可以用ProcessBuilder类来执行命令。
它的正常情况使用如下.

ProcessBuilder pb = new ProcessBuilder("calc.exe");
pb.start();

我们来分析一下ProcessBuilder的构造方法,它的构造方法有很多个重载,我们分析一个吧。

QQ截图20210219151901

可见,ProcessBuilder的构造方法把传入参数保存到了command属性里,然后commad会被以系统命令调用(这部分代码就不贴出来了)。

那么以反射的形式该如何实现呢。

Class clazz = Class.forName("java.lang.ProcessBuilder");
Method start = clazz.getMethod("start");
start.invoke(clazz.getConstructor(List.class).newInstance(Arrays.asList("cmd.exe")));

Servlet

Servlet生命周期

1.调用init()方法进行初始化

init()方法在第一次创建servlet时被调用,用于进行初始化操作。

public void init() throws ServletException{...code...}

2.调用service()方法处理客户端请求

web服务器,即servlet容器调用service()方法处理来自客户端的请求,并返回响应。同时根据客户端请求的类型(Get,Post,delete等),做出不同的行为,这些行为是由service方法抽象出的其他方法(doGet,doPost等)。
所以这里说的调用service()方法也就是等同于调用doGet,doPost等方法。

doPost,doDelete等方法的定义形同于下
public void doGet(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{...code...}

3.调用distroy()在Servlet销毁前完成清理活动

distrtoy()函数只调用一次,在Servlet被销毁前完成后台线程停止,数据库链接关闭等一系列清理工作。

public void destroy() {
// 终止化代码...
}

编写一个简单的Servlet

我们写一个程序,用来控制Get请求的响应

import javax.servlet.*;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.*;

@WebServlet("/a") //用于设定路由
public class test extends HttpServlet {

@Override
public void doGet(HttpServletRequest request,
HttpServletResponse response)
throws ServletException, IOException
{
response.setContentType("text/html");

PrintWriter out = response.getWriter();
out.println("<h1>" + "a" + "</h1>");
}
}

至于部署我是参考的这个文章https://blog.csdn.net/gaoqingliang521/article/details/108677301
我的版本时idea 2020.2,可能会有出入。

处理传参

servlet处理传参主要是通过doGet和doPost方法进行的。
接收参数值主要是靠下面的方法

  • getParameter():您可以调用 request.getParameter() 方法来获取表单参数的值。
  • getParameterValues():如果参数出现一次以上,则调用该方法,并返回多个值,例如复选框。
  • getParameterNames():如果您想要得到当前请求中的所有参数的完整列表,则调用该方法。

我们写个demo试试

import javax.servlet.*;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.*;

@WebServlet("/a")
public class test extends HttpServlet {

@Override
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
response.setContentType("text/html");
PrintWriter out = response.getWriter();
String name = new String(request.getParameter("name"));
out.println("your name is "+name);
}
}

这个demo主要用于接收get数据,并做出一些处理返回.
那么如何接收POST数据呢?和doGet差不多

import javax.servlet.*;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.*;

@WebServlet("/a")
public class test extends HttpServlet {

public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
response.setContentType("text/html");
PrintWriter out = response.getWriter();
String name = new String(request.getParameter("name"));
out.println("[Post]your name is "+name);
}
}
需要注意的是,这里的request.getParameter("name")会优先选择在url里的参数,而不是http包体的参数

QQ截图20210219152111

cookie的主要方法如下

序号 方法 & 描述
1 public void setDomain(String pattern) 该方法设置 cookie 适用的域,例如 runoob.com。
2 public String getDomain() 该方法获取 cookie 适用的域,例如 runoob.com。
3 public void setMaxAge(int expiry) 该方法设置 cookie 过期的时间(以秒为单位)。如果不这样设置,cookie 只会在当前 session 会话中持续有效。
4 public int getMaxAge() 该方法返回 cookie 的最大生存周期(以秒为单位),默认情况下,-1 表示 cookie 将持续下去,直到浏览器关闭。
5 public String getName() 该方法返回 cookie 的名称。名称在创建后不能改变。
6 public void setValue(String newValue) 该方法设置与 cookie 关联的值。
7 public String getValue() 该方法获取与 cookie 关联的值。
8 public void setPath(String uri) 该方法设置 cookie 适用的路径。如果您不指定路径,与当前页面相同目录下的(包括子目录下的)所有 URL 都会返回 cookie。
9 public String getPath() 该方法获取 cookie 适用的路径。
10 public void setSecure(boolean flag) 该方法设置布尔值,表示 cookie 是否应该只在加密的(即 SSL)连接上发送。
11 public void setComment(String purpose) 设置cookie的注释。该注释在浏览器向用户呈现 cookie 时非常有用。
12 public String getComment() 获取 cookie 的注释,如果 cookie 没有注释则返回 null。

我们马上写一个demo,这个demo会用到 response.addCookie用于在响应头里添加cookie,cookie.setMaxAge 用于设置cookie的有效期,request.getCookies 用于获取请求里的cookie数组
这里的demo有两个,一个用于添加cookie,一个用于输出cookie

用于添加cookie
import javax.servlet.*;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.*;

@WebServlet("/a")
public class test extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
Cookie name = new Cookie("name", request.getParameter("name")); // 中文转码

name.setMaxAge(60*60*24);

response.addCookie( name );

}
}


输出cookie值
import javax.servlet.*;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.*;

@WebServlet("/b")
public class test2 extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
Cookie cookie = null;
Cookie[] cookies = null;
// 获取与该域相关的 Cookie 的数组
cookies = request.getCookies();

// 设置响应内容类型
response.setContentType("text/html;charset=UTF-8");

PrintWriter out = response.getWriter();
for (int i = 0; i < cookies.length; i++){
if((cookies[i].getName()).compareTo("name")==0){
out.println("your name is "+cookies[i].getValue());
}
}

}
}

QQ截图20210219152122

session

servle提供了一系列session接口辅助这个流程

1 public Object getAttribute(String name) 该方法返回在该 session 会话中具有指定名称的对象,如果没有指定名称的对象,则返回 null。
2 public Enumeration getAttributeNames() 该方法返回 String 对象的枚举,String 对象包含所有绑定到该 session 会话的对象的名称。
3 public long getCreationTime() 该方法返回该 session 会话被创建的时间,自格林尼治标准时间 1970 年 1 月 1 日午夜算起,以毫秒为单位。
4 public String getId() 该方法返回一个包含分配给该 session 会话的唯一标识符的字符串。
5 public long getLastAccessedTime() 该方法返回客户端最后一次发送与该 session 会话相关的请求的时间自格林尼治标准时间 1970 年 1 月 1 日午夜算起,以毫秒为单位。
6 public int getMaxInactiveInterval() 该方法返回 Servlet 容器在客户端访问时保持 session 会话打开的最大时间间隔,以秒为单位。
7 public void invalidate() 该方法指示该 session 会话无效,并解除绑定到它上面的任何对象。
8 public boolean isNew() 如果客户端还不知道该 session 会话,或者如果客户选择不参入该 session 会话,则该方法返回 true。
9 public void removeAttribute(String name) 该方法将从该 session 会话移除指定名称的对象。
10 public void setAttribute(String name, Object value) 该方法使用指定的名称绑定一个对象到该 session 会话。
11 public void setMaxInactiveInterval(int interval) 该方法在 Servlet 容器指示该 session 会话无效之前,指定客户端请求之间的时间,以秒为单位。

写个demo吧,过一过就行了

import javax.servlet.*;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.*;
import java.text.*;
import java.util.*;

@WebServlet("/a")
public class test extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
HttpSession session = request.getSession(); //创建一个session对象
Date createtime = new Date(session.getCreationTime());
Date lastAccessTime = new Date(session.getLastAccessedTime());
PrintWriter out = response.getWriter();

SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd:mm:ss");
String countkey = new String("key"); //访问次数的键
Integer count = Integer.valueOf(0); //访问次数的值
String userID = new String("Tom"); //用户ID的值
String userIDKey = new String("userIDKey"); //用户ID的键


if(session.getAttribute(countkey)==null){ //如果获得该session的countkey这个键的值不存在时,为当前session初始化这个键
session.setAttribute(countkey,Integer.valueOf(0));
}

if(session.isNew()){ //如果session是第一次创建,则为其初始化userID键
session.setAttribute(userIDKey,userID);
}
else{
count = (Integer) session.getAttribute(countkey);
count = count+1;
userID = (String)session.getAttribute(userIDKey);
}
session.setAttribute(countkey,count);
out.println("hello"+userID+"\nthis is the "+count+" to visit this website\n");
out.println("first time:"+createtime+"\n this time:"+lastAccessTime);
}
}

JSP

概览

JSP是脚本语言,与PHP等语言类似。JSP本质就是servlet,它存在的意义就是简化servlet复杂的程序过程

JSP基础语法

代码段

JSP和PHP等可以嵌入到HTML的脚本语言类似,需要用一个代码段包裹代码。
其格式如下

<% code %>

同时在代码段中还存在着指令标签:Page,include,taglib
他们的定义和作用如下

<%@ page ...... %>    
page指令用于定义网页的各属性,如<%@ page import="..." %> 表示要导入哪些java模块,<%@ page language="..." %>表示JSP页面所用的语言,默认是java

<%@ include file="文件相对url地址" %> 表示要包含哪些其他文件,可以包含JSP,HTML或txt等文件

<%@ taglib ......%> 主要用于用户自定义标签

动作元素

与指令元素比较相似,但不同的是动作元素可以动态的插入,不像指令元素一样一开头就会被执行。

QQ截图20210219152132

九大对象

JSP中有九大对象用于实现各种操作。

QQ截图20210219152143

至于request和response,这俩用的是最多的是他们,
out对象主要用于在response中写入输出内容, out.println或者out.print用于输出内容,out.flush用于刷新输出流,其余在新手阶段都不咋用。

传参处理

  • getParameter(): 使用 request.getParameter() 方法来获取表单参数的值。
  • getParameterValues(): 获得如checkbox类(名字相同,但值有多个)的数据。 接收数组变量 ,如checkbox类型
  • **getParameterNames():**该方法可以取得所有变量的名称,该方法返回一个 Enumeration。
  • **getInputStream():**调用此方法来读取来自客户端的二进制数据流。
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>test jsp</title>
</head>
<body>
your name is <%=request.getParameter("name")%>
</body>
</html>

QQ截图20210219152207

QQ截图20210219152242

cookie,session在JSP里的设置与servlet如出一辙

cookie和session在jsp的<%%>代码区里设置,设置方法与servlet如出一辙,因为<%%>代码区里本身就是调用的java代码,所以不再赘述

RMI

何为RMI

RMI,即远程方法调用,允许运行在一个JAVA虚拟机调用另一个JAVA虚拟机上的对象的方法.

RMI:实现一个远程接口

RMI的远程接口用于其他java虚拟机远程调用该接口下的对象的方法.

定义一个远程接口

import java.rmi.Remote;   //引入Remote接口
public interface IHello extends Remote{ //实现一个Remote接口
public String sayHello(String name) throws java.rmi.RemoteException;
//若是Remote接口里某个方法抛出RemoteException异常,就意味着这是一个可以被远程调用的方法,但目前这个方法还未被具体实现.
}

远程接口的实现类

刚刚我们定义了一个Remote接口,但是他的方法还未具体实现.我们就来实现远程接口的实现类吧

import java.rmi.RemoteException;
import java.rmi.server.UnicastRemoteObject;
public class HelloImpl extends UnicastRemoteObject implements IHello{ //定义一个远程接口实现类,注意这个远程实现类必须继承自UnicastRemoteObject,不然服务端不会发送该类的对象的存根(stub(下文讲))
protected HelloImpl throws RemoteException{//必须要有一个抛出RemoteException异常的显示构造函数
super();
}
public String sayHello(String name) throws RemoteException{
return "Hello"+name; //定义实现类方法的具体代码
}
}

两台JVM通讯的第一步:RMI Registry

何为RMI Registry
可以理解为部署在被远程调用的JAVA虚拟机上的一个应用,用于将stub绑定到Registry服务的URL上。

服务端绑定stub到指定url

java.rmi.Naming.rebind("rmi://localhost:1099/hello", hello);   
//如此处就是把hello这个对象绑定在了rmi://localhost:1099/hello这个URL

随后客户端想要远程调用hello这个对象的方法时,就需

IHello hello = (IHello) Naming.lookup("rmi://localhost:1099/hello");
//此刻客户端查找出了hello对象的stub所在url,服务端返回hello对象的Stub
//此时的数据类型似乎必须是接口类

QQ截图20210219152319

Stub和Skeleton

QQ截图20210219152334

由上文可知当客户机远程调用一个对象时,返回的其实不是对象本身,而是stub.
你可以理解stub为一个中继站,当客户机调用该对象某个方法时,实际上是通过stub以socket的方式向服务器端的skeleton发送序列化处理的方法名和参数(skeleton可以理解为服务器端上的真实对象)
服务端的skeletion反序列化得到的方法名和参数并处理后再以socket的方式把该方法生成的结果传回stub,stub再把数据返回给客户机

代码实现远程调用

服务器端
import java.rmi.Remote;
import java.rmi.registry.*;
import java.rmi.RemoteException;
import java.rmi.server.UnicastRemoteObject;

interface Hello extends Remote{
public String sayHello(String name) throws RemoteException;
}

class HelloImpl extends UnicastRemoteObject implements Hello{
protected HelloImpl()throws RemoteException {
super();
}
public String sayHello(String name) throws RemoteException{
return "Hello"+name;
}
}
------------------------------------------------------------------
public class Server{
public static void main(String[] args){
try{//必须有try..catch
Hello hello = new HelloImpl(); //似乎数据类型只能是接口类
LocateRegistry.createRegistry(1080); //将RMI registry的端口设置为1080
java.rmi.Naming.rebind("rmi://localhost:1080/hello", hello);//
将hello这个方法对象绑定在指定url上,rmi协议
System.out.print("OK");
}catch(Exception e){
e.printStackTrace();
}
}
}
客户机端
import java.rmi.Remote;
import java.rmi.registry.*;
import java.rmi.RemoteException;
import java.rmi.server.UnicastRemoteObject;
import java.rmi.Naming;
public class Client{
public static void main(String[] args){
try{//必须有try..catch
Hello hello_client = (Hello) Naming.lookup("rmi://localhost:1080/hello"); //服务器端绑定在该url的对象数据类型是Hello,所以这里的数据类型也是Hello
System.out.print(hello_client.sayHello("fuck"));
}catch(Exception e){
e.printStackTrace();
}
}
}

结果

QQ截图20210219152343

javassist

javassist 是一个类库,用于操作java字节码。

比如我们接下来就会使用javassist来动态操作字节码。

test2.java
public class test2 {
public void gogo(){
System.out.println("yuansheng");
}
}

test1.java
import javassist.*;
public class test1 {
public static void main(String[] args) throws Exception{
ClassPool pool = ClassPool.getDefault();
CtClass class1 = pool.get(test2.class.getName());
String cmd = "System.out.print(\"hello\");";
class1.makeClassInitializer().insertBefore(cmd); \\插入static静态代码块,内容为cmd中内容
class1.setName("class1"); \\为新类命名为class1
class1.writeFile("d:\\test"); \\将类写入d:\\test文件夹
}
}

反编译d:\\test\\class1.class

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by FernFlower decompiler)
//

public class class1 {
public class1() {
}

public void gogo() {
System.out.println("yuansheng");
}

static {
System.out.print("hello");
}
}

发现我们已经往test2.java的内容中插入了输出hello的静态代码块,并以class1.class写入d:\\test文件夹。

javassist的功能除此之外还有很多