ssm-springmvc-学习笔记

news/2024/12/15 0:08:38/

简介

简单的来说,就是一个在表述层负责和前端数据进行交互的框架

帮我们简化了许多从前端获取数据的步骤

springmvc基本流程

用户在原本的没有框架的时候请求会直接调用到controller这个类,但是其步骤非常繁琐

所以我们就使用springmvc进行简化

当用户发送请求时,首先到达的是Servlet,解析用户的请求,然后去mapping类寻找用户请求的是类中的哪一个方法

接着寻找到是哪个类之后,在发送request到Adapter去,此时adapter会拆解request然后将其中有用的参数发送给handler执行对应的逻辑

接着handler执行完毕后再发送adapter处理JSON串给servlet一个respone报文,返回给前端做出数据响应

核心组件

1. DispatcherServlet :  SpringMVC提供,我们需要使用web.xml配置使其生效,它是整个流程处理的核心,所有请求都经过它的处理和分发![ CEO ]
2. HandlerMapping :  SpringMVC提供,我们需要进行IoC配置使其加入IoC容器方可生效,它内部缓存handler(controller方法)和handler访问路径数据,被DispatcherServlet调用,用于查找路径对应的handler![秘书]
3. HandlerAdapter : SpringMVC提供,我们需要进行IoC配置使其加入IoC容器方可生效,它可以处理请求参数和处理响应数据数据,每次DispatcherServlet都是通过handlerAdapter间接调用handler,他是handler和DispatcherServlet之间的适配器![经理]
4. Handler : handler又称处理器,他是Controller类内部的方法简称,是由我们自己定义,用来接收参数,向后调用业务,最终返回响应结果![打工人]
5. ViewResovler : SpringMVC提供,我们需要进行IoC配置使其加入IoC容器方可生效!视图解析器主要作用简化模版视图页面查找的,但是需要注意,前后端分离项目,后端只返回JSON数据,不返回页面,那就不需要视图解析器!所以,视图解析器,相对其他的组件不是必须的![财务]

springmvc运用大致流程

首先声明一下controller

通过requestMapping指定该类的访问路径

package com.atguigu.controller;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;@Controller
public class HelloController {//     对外访问的地址,到handlerMapping注册的注册@RequestMapping("springmvc/hello")@ResponseBody//直接返回字符串给前端不需要找视图public String Hello(){System.out.println("hello");return "springMvc Hello";}
}

接着我们声明一个配置类,将各种springmvc需要的类放入ioc容器当中

通过Bean组件放入ioc容器


//TODO: 导入handlerMapping和handlerAdapter的三种方式
//1.自动导入handlerMapping和handlerAdapter [推荐]
//2.可以不添加,springmvc会检查是否配置handlerMapping和handlerAdapter,没有配置默认加载
//3.使用@Bean方式配置handlerMapper和handlerAdapter
@EnableWebMvc
@Configuration
@ComponentScan(basePackages = "com.atguigu.controller") //TODO: 进行controller扫
//WebMvcConfigurer springMvc进行组件配置的规范,配置组件,提供各种方法! 前期可以实现
public class MvcConfig implements WebMvcConfigurer {@Beanpublic HandlerMapping handlerMapping(){return new RequestMappingHandlerMapping();}@Beanpublic HandlerAdapter handlerAdapter(){return new RequestMappingHandlerAdapter();}}

那么我们知道spring中将类放入ioc容器中是需要读取一些配置文件的

那么将springmvc需要的各种工具类放入ioc容器中依靠的是如下的接口

AbstractAnnotationConfigDispatcherServletInitializer
package com.atguigu.config;import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;//该类可以被web项目加载,会初始化ioc容器,会设置dispatcherServlet的地址
public class SpringMvcInit extends AbstractAnnotationConfigDispatcherServletInitializer {@Overrideprotected Class<?>[] getRootConfigClasses() {return new Class[0];}//    设置我们项目的配置类@Overrideprotected Class<?>[] getServletConfigClasses() {return new Class[]{MvcConfig.class};}//    配置springmvc内部自带的servlet的访问地址@Overrideprotected String[] getServletMappings() {return new String[]{"/"};
//        单个斜杠就代表着处理所有的用户请求的意思}
}

那么上述的方法是如何实现将类放入ioc容器中呢?

我们先介绍一个接口

这个接口里面有一个onStartUp函数,其中这个函数会在web项目启动的时候调用

web容器初始化讲解

我们来找寻

AbstractAnnotationConfigDispatcherServletInitializer

的继承关系然后找出web容器初始化的原理

当我们点进去它的父类之后,我们再进去它的父类观察

再父类里面我们发现有一个

registerDispatcherServlet

方法,这个方法就是将web容器放入ioc容器当中的方法,我们点进去观察

点进去这个方法之后,我们发现有一个名为

createServletApplicationContext();

的方法,我们再点进去观察

我们在这里发现,有一个

AnnotationConfigWebApplicationContext();

类,这个类再spring中是用于读取配置类和配置文件将类导入ioc容器中的类

紧接着我们发现,在这个类中,我们发现了一个

getServletConfigClasses();

这个方法。是不是很熟悉?

我们回到最开始的地方观察

在最开始的继承了上述所有类的子类中,是不是有一个这样的方法

将配置类通过反射装入数组,然后再父类的函数中调用数组将其放入了ioc容器

如果对象数组不为空就将其配置类放入ioc容器进行注册

里面还有servletmapping的代码,用于将地址映射的类也装入ioc容器中

SpringMvc接收数据

1.访问路径设置

Java代码展示

该代码中和新注释就是@ResquestMapping

该注释能够使得这个controller响应一个user的网页路径请求

并且写在该类下的所有方法的请求路径都是

/user/方法上的request路径

其余的原理都写在了注释里

package com.atguigu.requestmapping;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;@Controller
@RequestMapping("user")
public class UserController {//    handler->handlerMapping@RequestMapping("login")public String login() {return "login";}//    可以通过一个字符数组装入注册的地址,可以响应多个请求地址,method用于指定前端发送什么请求类型才会响应@RequestMapping(value = {"register","login"},method = RequestMethod.GET)public String register(){return "register";}//    模糊匹配,*表示前面如果是/user,则后面不管是/user/dasdasdasd /user/ds都能匹配 不过多层不可匹配/user/a/a这样就不能匹配
//    **表示前面如果是/user,则后面不管是/user/aa/aaaa/aaaa/,多少层都能够匹配
//    @GetMapping表示该方法以get方式接收请求实际源码只不过是多加了一个method 指明了请求方式罢了@GetMapping({"/user/*","/user/**"})public String modifyPassword(){return "modifyPassword";}}

2.接收参数

 @RequestMapping("data")@ResponseBody
//    形参列表,填写对应的名称参数即可! 请求参数名 = 形式参数名即可,这样会自动赋值public String data(String name,int age){System.out.println("name: "+name+"  age: "+age);return "name: "+name+"  age: "+age;}

通过指定requestMapping注释能够指定接收路径

通过ResponseBody能够指定该类直接返回给前端字符串进行接收

1.@Param注解

通过value指定传入参数时的key名字,就算是传入的key名字和参数名相同也会报错

required用于指定是否是必须被传入的参数,默认值是true代表必须传入,否则报错

 

2.一名多值

可以通过一个列表接收传入的多个值

但是注意,如果未使用requestParam那么就会报错

因为未使用requestParam就会代表着直接将一个字符串赋值给了一个列表报错

3.实体类接收参数

只需要将传入的参数名和实体类名保持一致即可

4.路径传参接收

要点:在路径标签中指定其路径的时候使用花括号将其括起来

然后再参数列表中声明一个与路径中花括号括起来的key值相同的名字即可

这样就可以获取到路径中的值,如下图结果所示

不过要注意,我们需要使用@PathVariable写在参数名前面,不然正常情况下默认规定其通过param方式获取数据

即便参数列表中参数名不相同依旧可以指定value的对应值与mapping注解中的key值传参

5.json数据接收

Java在没有导入对应的json转换依赖的时候是不能够获取到json数据的

我们导入一个json依赖

在配置类上方添加一个

@EnableWebMvc

注解就可以使得适配器转化json数据并发送给类进行接收

@EnableWebMvc解释

@EnableWebMvc注解效果等同于在 XML 配置中,可以使用 `<mvc:annotation-driven>` 元素!我们来解析`<mvc:annotation-driven>`对应的解析工作!

让我们来查看下`<mvc:annotation-driven>`具体的动作!

先查看标签最终对应解析的Java类

查看解析类中具体的动作即可

打开源码:org.springframework.web.servlet.config.MvcNamespaceHandler

打开源码:org.springframework.web.servlet.config.AnnotationDrivenBeanDefinitionParser

在对应的源码中,我们可以发现他将一个Adapter(经理)加入了ioc容器,并且在Adapter对象中加入了json字符串适配器能够解析json串

在源码中由于已经将Adapter和Handler加入了ioc容器,我们就不需要在手动生成这两对象假如ioc容器中,只需要添加一个注解即可

6.接收cookie数据

先来介绍一下cookie是什么

Cookie,即“小甜饼”的意思,在计算机领域中,特指一种由服务器发送到用户浏览器并保存在用户计算机(客户端)上的小型文本文件。它满足RFC6265标准,通常用于辨别用户身份、跟踪用户活动、保存用户设置等。以下是对Cookie的详细解释:

一、Cookie的定义

Cookie是一种技术,允许网站服务器把少量数据储存到客户端的硬盘或内存,或是从客户端的硬盘读取数据。这些数据以“名/值”对(name-value pairs)的形式储存,通常经过了加密处理,因此一般用户看到的只是一些毫无意义的字母数字组合,只有服务器的CGI处理程序才知道它们真正的含义。

二、Cookie的作用

  1. 会话管理:Cookie最初也是最主要的作用就是用于会话管理。当用户登录一个网站时,服务器会生成一个包含会话ID的Cookie并发送给浏览器,浏览器将这个Cookie保存在本地。此后,每次用户发送请求时,浏览器都会自动将这个Cookie发送给服务器,服务器通过会话ID识别用户身份,从而保持用户的登录状态。
  2. 个性化设置:Cookie还可以用来保存用户的个性化设置,如主题、语言、字体大小等。这样,当用户再次访问网站时,网站可以根据Cookie中的信息为用户提供更加个性化的体验。
  3. 购物车功能:在电子商务网站中,Cookie经常被用来实现购物车功能。当用户将商品添加到购物车时,这些信息会被保存在Cookie中。这样,即使用户关闭了浏览器或换了一台电脑,只要Cookie还在,购物车中的商品信息就不会丢失。
  4. 跟踪用户行为:网站可以使用Cookie跟踪用户在网站上的行为,如访问了哪些页面、停留了多长时间、点击了哪些链接等。这有助于网站分析用户行为,优化网站设计和内容。
  5. 广告定向:除了网站自己设置的Cookie外,还有一些第三方Cookie,它们通常由广告商或数据分析公司设置。这些Cookie可以用来跟踪用户在多个网站上的行为,从而为用户提供更加精准的广告定向服务。

三、Cookie的类型

  1. 会话Cookie(Session Cookies):这种类型的Cookie在浏览器关闭后就会被删除,主要用于保存用户的会话信息。
  2. 持久Cookie(Persistent Cookies):与会话Cookie不同,持久Cookie会在用户的计算机上长期保存,直到其过期时间到达或被用户手动删除。这种类型的Cookie常用于保存用户的登录状态、个性化设置等信息。
  3. 安全Cookie(Secure Cookies):安全Cookie只能通过HTTPS协议传输,不能通过未加密的HTTP协议传输。这增加了Cookie在传输过程中的安全性。
  4. HttpOnly Cookie:HttpOnly是一个标志属性,用于防止JavaScript代码访问特定的Cookie。当设置了HttpOnly属性的Cookie被创建后,它将无法通过客户端脚本(如JavaScript)进行访问。这有助于减少跨站脚本攻击(XSS)的风险。

四、Cookie的安全性问题及防范措施

虽然Cookie在许多方面都非常有用,但它们也存在一些潜在的安全风险,如XSS攻击、CSRF攻击、Cookie劫持和隐私泄露等。为了防范这些风险,网站可以采取以下措施:

  1. 设置HttpOnly属性、对输出进行编码等,以防止XSS攻击。
  2. 使用Token验证、检查请求的来源等,以防止CSRF攻击。
  3. 使用HTTPS协议对Cookie进行加密传输、设置SameSite属性等,以防止Cookie劫持。
  4. 遵循最小必要原则收集和使用用户数据,并采取加密、匿名化等安全措施,以保护用户隐私。

综上所述,Cookie作为一种重要的客户端技术,在互联网应用中发挥着举足轻重的作用。然而,在使用Cookie的过程中,我们也需要注意其潜在的安全风险,并采取相应的措施进行防范和保护。

那么我们如何接收cookie数据呢

我们在Java类中编写如下一个类,在参数中通过cookievalue指定cookiename的请求的参数名字

我们先用save通过响应报文存储一个cookie数据

可以通过图片看到存储成功了

接着我们通过cookiename这个key值从cookie中取出对应的value值显示在屏幕上

7.接收请求头数据

通过RequestHeader标签写在参数上,就能够通过参数获取到对应请求头的中的key值为Host的数据

8.获取原生api对象

SpringMVC接收数据总结

SpringMVC响应数据(输出数据)

两种开发模式介绍

1.前后端不分离介绍

整体调用流程

1.浏览器通过向服务端发送请求进行一些数据交互和页面跳转请求

2.controller层接收后通过三层架构一步步发送到mapper层(Dao层)

3.dao层通过和数据库交互获取数据返回到controller进行数据封装处理

4.在前后端不分离项目中,controller会将数据放入共享域中,并且找到需要跳转的html页面

5.然后前端页面会从共享域中拿取数据,并且将模板页面发送给controller层

6.接着将controller层中返回到的html页面发送给浏览器

7.主要用于小型的管理系统的开发

开发流程

我们首先在webapp下面的web-inf的文件中声明一个.jsp文件这是一个网页

接着我们创建一个配置类,让他继承WebMvcConfigurer

继承这个类之后我们就可以将视图解析器装入ioc容器中了(该类还封装了很多mvc容器需要的类,视图解析器的作用是将.jsp文件进行解析)

我们在重写的方法中,通过registry将文件的index.jsp的文件进行前后缀注册,以便能够定位到视图的位置

接着我们创建一个jspcontroller能够将视图返回给前端

其注意要点图里面就有

其中setAttribute是将数据放入共享数据域的

观察jsp文件,在共享数据区域获取数据

这是返回的结果

转发与重定向实现

转发指的是访问项目内的文件,范围只能是项目内

重定向指可以是向内也可以是向外

其在springMVC中的实现如下

先观察一下资源文件的访问路径是/jsp/index

转发

在转发中,我们先指定转发的访问路径将其设置为forward

然后我们在方法内返回视图文件的访问路径即可,但是要注意

在访问路径前面需要加一个forward编译器才能判定他是在进行转发

否则会直接识别为这是资源文件的位置进行解析,就如开发流程那里所说的一眼

通过转发的方式我们也到达了forward的文件的位置

重定向

其注意要点如图所示

这是向内重定向

同样也有向外重定向

2.前后端分离模式

主要通过json和前端进行数据交互

前端都是调用的一个后端接口获取数据

一般用于大型商业部署项目开发,多端部署

返回json串

其原理如下图所示

核心的注解是一个@ResponeseBody

通过responesebody能够告知adapter我们即将返回的是一个json串

然后让adapater进行处理,将Java类对象转化为一个json串发送给前端

通过使用responseBody能够使得返回视图和转发重定向的语法全部生效

具体controller类的编写如下

通过controller添加到ioc容器

通过requestmapping指定访问路径

接着在需要返回json串的类/方法上添加@ResponseBody即可返回一个json串

也可以是返回一个list对象

这两个方法的返回结果如下图所示

RestController注解

可以观察源码,同时包含了controller和responsebody注解

能够使用这一个注解替代controller和responsebody注解

返回静态资源

我们现在文件目录下面放入一个静态图片资源

我们在平常的客户端请求文件的路径时是访问不到的

因为默认情况下当服务端接收请求之后,回去handlerMapping()中寻找有没有对应的handler请求路径

此时会发现没有对应的请求路径,那么就会发生错误

那么我们如何解决这种情况呢

在配置类中重写一个如红圈所示的接口就可以实现找到静态资源

那么其原理是什么呢

我们点进去接口里面进行观察,如红箭头所指的类是返回静态资源得以成功的方法

接着点进去那个添加Servlet进行观察

我们可以发现,他添加了一个handler对静态资源进行处理

那么这个handler是如何对静态资源进行处理的呢

在这个handler是这样处理的,假如我们没有找到路径,那么就会对图片资源的路径进行内部转发

通过转发去寻找请求地址所需要的资源

原理图

Servlet接收到一个请求,先去handlerMapping寻找有没有对应的controller类查找数据

发现没有,那么handlerMapping就会找DefaultServletHandler去转发图片的路径寻找资源

结果如下

RESTFul风格

一种基于http协议的规范编码风格

其请求路径的设计如下图所示

全局异常处理机制

什么是声明式异常和编程式异常?

编程式:在代码内部有详细的异常捕捉

声明式:在外部使用一个配置文件对需要进行异常捕获的文件进行包裹,有异常时就会调用外部异常处理代码进行处理

全局异常处理注释

所有需要的异常都在图中写清楚了

详解再图片下方写

@ControllerAdvice和@RestControllerAdvice解析

ControllerAdvice:标识这是一个全局异常处理类,当全局异常发生时就会寻找该类进行处理,可以返回逻辑视图和转发重定向

RestControllerAdvice:点进去后其实可以发现源代码里面有controlleradvice和responsebody注解在里面

responseBody是告知这个类会返回一个json字符串

@ExceptionHandler

异常处理的类,在其后面的括号内的参数是用于指定该方法处理什么类型的异常

拦截器

类似于javaweb中的Filter(过滤器)

对一些特殊的请求进行拦截处理

拦截器 Springmvc VS 过滤器 javaWeb:

- 相似点
    - 拦截:必须先把请求拦住,才能执行后续操作
    - 过滤:拦截器或过滤器存在的意义就是对请求进行统一处理
    - 放行:对请求执行了必要操作后,放请求过去,让它访问原本想要访问的资源
- 不同点
    - 工作平台不同
        - 过滤器工作在 Servlet 容器中
        - 拦截器工作在 SpringMVC 的基础上
    - 拦截的范围
        - 过滤器:能够拦截到的最大范围是整个 Web 应用
        - 拦截器:能够拦截到的最大范围是整个 SpringMVC 负责的请求
    - IOC 容器支持
        - 过滤器:想得到 IOC 容器需要调用专门的工具方法,是间接的
        - 拦截器:它自己就在 IOC 容器中,所以可以直接从 IOC 容器中装配组件,也就是可以直接得到 IOC 容器的支持

其作用范围如图所示

因为使用了springmvc框架,那么在程序内部是只有一个DispatcherServlet进行请求处理的

假如使用filter的话,只能在Servlet外部即整个springMVC框架外部进行拦截处理,一刀切,并不能在里面的各种资源调用的内部进一些业务逻辑的处理

那么使用拦截器就可以解决这一问题

拦截器作用的位置

拦截器如何使用

首先在配置类中注册一下拦截器(注意,图中的这种注册方式是拦截全部请求的,至于具体到方法的后续会讲解到)

拦截器类的书写,我们需要继承一个HanlerInterceptor类

其需要重写如下三个方法

preHandle

postHandle

afterCompletion

preHandle方法

其参数和注意事项如下

其中handler就是我们进行拦截的目标方法

preHandler是在目标方法执行前执行的代码

postHandle方法

其参数和主义的事项如下

其中modelAndView返回的是一个视图或者共享域对象

不需要时为空

afterCompletion方法

在前面两个方法执行完毕后调用

其中ex参数是当目标方法发生异常时传入,并进行输出

拦截器的指定拦截

链式调用

在将拦截器类进行注册后添加一个addpathPatterns就可以进行拦截

排除拦截

在前一个大的路径内,指定一个在大路径内的小路径就可以进行小路径的拦截

拦截的顺序

根据注册的先后进行拦截,其源码其实是所有的拦截类放入一个数组

然后prehandler是正序遍历数组进行方法调用,所以先进入的类会先执行

接着post和after方法是逆序遍历,也就是说,先进入的类后遍历

源码

通过mapping(秘书)对对应的拦截方法进行获取

观察下面的代码,下方红箭头所指通过handler获取拦截方法执行的目标方法

在红箭头的上方,有一个拦截方法,他会判断前置prehadler的返回结果,假如是一个false那么就进入方法执行return代表着剩下的目标方法的逻辑不执行,实现了拦截

参数校验

可以通过给实体类的变量上添加注解,不符合对应的要求的就会报错

在对应的实体类中添加对应的注解

接着我们需要在对应的接收前端数据的controller层接收对应实体类参数的方法中打的参数上添加

@Validdated注解!!!

如果不添加那自动校验就不会生效

以字符串不能为空举例当字符串为空时,会将报错信息默认的返回

但是我们发现这种方法前端不容易读懂,所以我们需要自定义的定义报错信息的返回

我们需要定义一个BindingResult类进行错误的接收

注意,该类一定要在参数列表中紧挨着需要校验的实体类

接着我们通过hasErrors判断是否报错

如果报错,我们定义一个集合将错误参数放入并返回

总结

调用流程,很关键


http://www.ppmy.cn/news/1555146.html

相关文章

docker简单私有仓库的创建

1&#xff1a;下载Registry镜像 导入镜像到本地中 [rootlocalhost ~]# docker load -i registry.tag.gz 进行检查 2&#xff1a;开启Registry registry开启的端口号为5000 [rootlocalhost ~]# docker run -d -p 5000:5000 --restartalways registry [rootlocalhost ~]# dock…

【Golang】Go语言编程思想(六):Channel,第三节,使用Channel实现树的遍历

使用 Channel 实现树的遍历 tree 在此处简单回顾一下之前学过的二叉树遍历&#xff0c;首先新建一个名为 tree 的目录&#xff0c;并在其下对文件和子目录进行如下组织&#xff1a; 其中 node.go 存放的是 Node 的定义&#xff1a; package treeimport "fmt"type…

pytest -s执行的路径

pytest -s执行的路径&#xff1a; 直接写pytest -s&#xff0c;表示从当前路径下开始执行全部.py的文件。 执行具体指定文件&#xff1a;pytest -s .\testXdist\test_dandu.py 下面这样执行pytest -s 会报找不到文件或没权限访问&#xff0c; 必须要加上具体文件路径pytest -s…

Linux网络基础知识————网络编程

计算机网络的体系结构 网络采用分而治之的方法设计&#xff0c;将网络的功能划分为不同的模块&#xff0c;以分层的形式有机结合在一起 每层实现不同的功能&#xff0c;其内部实现的方法对外部其他层次来说是透明的&#xff0c;每层向上一层提供服务&#xff0c;使用下一层提供…

【初识数据结构】堆的详解以及Topk问题

数据结构第七课 1. 前言&#x1f6a9;2. 堆的概念以及结构&#x1f6a9;3. 堆的实现&#x1f6a9;3.1 初始化结构&#x1f3f4;3.2 初始化函数&#x1f3f4;3.3 插入函数&#x1f3f4;3.4 向上调整函数&#x1f3f4;3.5 删除函数&#x1f3f4;3.6 向下调整函数&#x1f3f4;3.7…

电脑文件夹打不开了,能打开但是会闪退,提示“找不到iUtils.dll”是什么原因?

电脑运行时常见问题解析&#xff1a;文件夹打不开、闪退及“找不到iUtils.dll”报错 在使用电脑的过程中&#xff0c;我们可能会遇到文件夹打不开、软件闪退或系统报错等问题&#xff0c;特别是提示“找不到iUtils.dll”的报错&#xff0c;更是让人困惑不已。今天我将为大家详…

第四章 认识JFrame和JDialog

第四章 认识JFrame和JDialog JFrame是java图形用户界面(GUI)编程中的一个重要类&#xff0c;它代表一个窗口或框架。JDialog是Java中的一种对话框组件&#xff0c;通常用于和用户进行交互。 4.1 认识JFrame和JDialog JFrame它拥有的特效以下几种特效&#xff1a; 1.窗口特效&a…

蒙特卡洛模拟(Monte Carlo Simulation)详解

简介&#xff1a;个人学习分享&#xff0c;如有错误&#xff0c;欢迎批评指正。 历史背景 蒙特卡洛模拟的名称来源于摩纳哥的蒙特卡洛赌场&#xff0c;因其依赖于随机性和概率&#xff0c;与赌博中的随机过程有相似之处。该方法的雏形可以追溯到20世纪40年代&#xff0c;二战期…