java中的action是指什么_Struts2【开发Action】知识要点

前言

前面Struts博文基本把Struts的配置信息讲解完了.....本博文主要讲解Struts对数据的处理

Action开发的三种方式

在第一次我们写开发步骤的时候,我们写的Action是继承着ActionSupport类的...为啥我们继承了ActionSupport类呢?下面我就会讲解到

继承ActionSupport类

我们来看一下ActionSupport干了什么:

76041990d75acba8644b402a18447205.png

也就是说,如果我们在Action类中需要用到Struts为我们提供的数据校验等Struts已经帮我们实现的功能,我们就继承着ActionSupport类..

实现Action接口

我们再来看看Action接口干了什么:

0654d378deafe45530713a26556a21ac.png

当然啦,ActionSuppot也继承着Action接口,所以ActionSuppot拥有Action接口的全部功能....因此,这种开发方式我们是比较少用的...

不继承任何类、不实现任何接口

开发此类的Action,它是不继承任何类、不实现任何接口的...也就是说,它就是一个普通的Java类....

Action类

public class PrivilegeAction {

public String login() {

System.out.println("我是普通的javaAction,不继承任何的类、不实现任何的接口");

return "success";

}

}

在配置文件中配置:

/index.jsp

效果:

523ff402620a4572a8a15ae0f00aeba2.png

小总结

如果我们使用到了Struts2一些特用的功能,我们就需要继承ActionSupport

如果我们没用到Struts2的特殊功能,只要平凡写一个Java类行了。

大多情况下,我们还是会继承ActionSupport的。

请求数据封装

一般地,我们使用Servlet的时候都是分为几个步骤的:

得到web层的数据、封装数据

调用service层的逻辑业务代码

将数据保存在域对象中,跳转到对应的JSP页面

现在问题来了,我们自己编写的Action类是没有request、response、Session、application之类的对象的....我们是怎么得到web层的数据、再将数据存到域对象中的呢??

前面已经说过了,Struts预先帮我们完成了对数据封装的功能,它是通过params拦截器来实现数据封装的

register.jsp

首先,我们填写表单页面的数据,请求Action处理数据

用户名:

密码:

年龄:

生日:

Action封装基本信息

在Action设置与JSP页面相同的属性,并为它们编写setter方法

private String username;

private String psd;

private int age;

private Date birthday;

public void setUsername(String username) {

this.username = username;

}

public void setPsd(String psd) {

this.psd = psd;

}

public void setAge(int age) {

this.age = age;

}

public void setBirthday(Date birthday) {

this.birthday = birthday;

}

我们直接在业务方法中访问这些变量,看是否能得到表单的值。

4820a72b33bc7af476da9c6d545823c5.png

Action封装对象

一般地,我们注册的时候,都是在Servlet上把基本信息封装到对象上...那么在Struts怎么做呢?

创建一个User类,基本的信息和JSP页面是相同的。

package qwer;

import java.util.Date;

/**

* Created by ozc on 2017/4/27.

*/

public class User {

private String username;

private String psd;

private int age;

private Date birthday;

public String getUsername() {

return username;

}

public void setUsername(String username) {

this.username = username;

}

public String getPsd() {

return psd;

}

public void setPsd(String psd) {

this.psd = psd;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

public Date getBirthday() {

return birthday;

}

public void setBirthday(Date birthday) {

this.birthday = birthday;

}

}

在Action中定义User对象出来,并给出setter和getter方法....值得注意的是:基本信息只要setter就够了,封装到对象的话,需要setter和getter

public class ccAction extends ActionSupport {

private User user;

public User getUser() {

return user;

}

public void setUser(User user) {

this.user = user;

}

public String register() {

System.out.println(user.getUsername());

System.out.println(user.getPsd());

System.out.println(user.getAge());

System.out.println(user.getBirthday());

return "success";

}

}

在JSP页面,提交的name要写成user.username之类的

用户名:

密码:

年龄:

生日:

33e5c9bd65676e3c7c48309163836d9f.png

得到域对象

Struts怎么把数据保存在域对象中呢???Struts提供了三种方式

一、得到Servlet API

我们可以通过ServletActionContext得到Servlet API

由于每个用户拥有一个Action对象,那么底层为了维护用户拿到的是当前线程的request等对象,使用ThreadLocal来维护当前线程下的request、response等对象...

//通过ServletActionContext得到Servlet API

javax.servlet.ServletContext context = ServletActionContext.getServletContext();

HttpServletRequest request = ServletActionContext.getRequest();

HttpSession session = request.getSession();

HttpServletResponse response = ServletActionContext.getResponse();

二、ActionContext类

我们还可以通过ActionContext类来得到request、response、session、application被Struts封装的Map集合

//得到ActionContext 对象

ActionContext context = ActionContext.getContext();

Map session = context.getSession();

Map application = context.getApplication();

//这是request的Map

Map request = context.getContextMap();

三、实现接口

当web容器发现该Action实现了Aware接口,会把相对应的资源通过Aware接口注射进去,实际上就是一种IOC。

Aware实际就是一种拦截器,拦截代码在执行Action之前执行、将资源注射到Action中

实现SessionAware, RequestAware, ApplicationAware接口,它就要在程序中实现三个方法:

private Map request;

private Map session;

private Map application;

@Override

public void setApplication(Map map) {

this.application = map;

}

@Override

public void setRequest(Map map) {

this.request = map;

}

@Override

public void setSession(Map map) {

this.session = map;

}

通过这些方法,我们就可以得到对应的Map对象.....

小总结

那么,我们有三种方法可以得到Servlet对应的对象,那么该使用哪一种呢???

分析:

第一种方法:需要导入Servlet的包,与Struts耦合了

第二种方法:只能在业务方法中使用ActionContext类得到对应的Map对象,如果有多个方法,那么每个方法都需要写类似的代码

第三种方法:可以在类上定义成员变量,以至于整个类都能使用。但是需要实现类、实现对应的方法

如果我们需要使用到对象的其他方法,类似getContextPath()之类的,那么只能使用第一种

如果我们就按照平常的开发,我们就使用第二种【获取简单,没有耦合】

至于第三种,当我们将来可能开发BaseAction的时候,就使用它!

日期转换问题

前面博文已经讲解了,Struts2为我们实现了数据自动封装...由上篇的例子我们可以看出,表单提交过去的数据全都是String类型的,但是经过Struts自动封装,就改成是JavaBean对应成员变量的类型了。

但是呢,日期类型只支持是yyyy-MM-dd这种格式的,因为我们在上个例子中直接使用的是Struts支持的格式,因此没有报错...本篇博文就是讲解Struts如何对日期类型的格式更好地支持

当我们使用的是yyyyMMdd这种格式的时候,我们看看Struts的自动封装能不能解析出相对应的日期

5bfe817815c1356552e24043a464348c.png

直接抛出了异常

6c1a17067fd3d02614980d1ab500394f.png

53905a115b5c815d36787503c5f0306e.png

分析

那么,我们怎么让Struts能够支持更多的日期格式呢??比如,我想Struts在自动封装数据的时候支持yyyyMMdd,yyyy年MM月dd日这样的日期格式.....

Struts提供了转换器给我们使用,也就是,我们可以自定义转换器,我们定义了什么格式,Struts就可以根据对应的格式进行自动封装...

当我们写完自定义转换器,是需要向Struts说明我们写了,不然的话,Struts是不知道我们自定义了转换器类的...

也就是说,我们要想实现类型转换,需要两步:

编写自定义转换器类

告诉Struts我们写了转换器类

自定义转换器类

一般地,我们想要编写自定义转换器类,都是实现StrutsTypeConverter类的....

/**

* Created by ozc on 2017/5/1.

* 自定义异常转换器类

*

* 我们要实现的就是:在Struts转换的时候,

*

*/

public class MyConvter extends StrutsTypeConverter {

//需求,当Struts自动封装数据时,也支持yyyyMMdd,yyyy年MM月dd日等格式的支持\

SimpleDateFormat[] format = {new SimpleDateFormat("yyyy-MM-dd"), new SimpleDateFormat("yyyyMMdd"), new SimpleDateFormat("yyyy年MM月dd日")};

/**

* 把String转换为指定的类型 【String To Date】

*

*

* @param map

* 当前上下文环境

* @param strings

* jsp表单提交的字符串的值

* @param aClass

* 要转换为的目标类型

*/

@Override

public Object convertFromString(Map map, String[] strings, Class aClass) {

//判断是否有值

if (strings == null) {

return null;

}

//判断是否是日期类型的

if (Date.class != aClass) {

return null;

}

//遍历循环

for (SimpleDateFormat dateFormat : format) {

try {

//解析传递进来的第一个就行啦

dateFormat.parse(strings[0]);

} catch (ParseException e) {

//如果格式不对,那么就跳出当前的循环

continue;

}

}

return null;

}

@Override

public String convertToString(Map map, Object o) {

return null;

}

}

告诉Struts,我写了转换器类

告诉Struts我写了一个转换器类,也分两种方式

定义了局部转换器类,就当前包下的Action类有效

定义了全局转换器类,整个项目有效

全局转换器

步骤:

在src目录下创建一个名为xwork-conversion.properties的文件

配置文件的内容:需要转换的类类型=转换器类的全名java.util.Date=qwer.MyConvter

局部转换器类

步骤:

在当前的Action包下创建名为Action名-conversion.properties的文件

文件的内容为:需要转换的字段【如果是JavaBean里的字段,需要写上JavaBean的】=转换器类的全名user.birthday=qwer.MyConvter

效果

3d4f28d3cd004a1b9394e61456d534f0.png59f0a10d91e87c9f580624d2154fd6bc.png

错误提示页面

当发生了日期转换的异常时,Struts给出的页面是这样子的:

53905a115b5c815d36787503c5f0306e.png

这个我们称之为input视图,我们要做的就是给出用户更友好的提示,于是在struts.xml文件中配置:如果返回的是input视图,那么跳转到我们相对应的页面上

/error.jsp

7d643ebd164ed3bbf43068d056d162ed.png

文件上传和下载

在讲解开山篇的时候就已经说了,Struts2框架封装了文件上传的功能........本博文主要讲解怎么使用Struts框架来完成文件上传和下载

回顾以前的文件上传

可以使用FileUpload或者SmartUpload组件来完成文件上传的功能。但是呢,FileUpload组件使用起来是比较麻烦的...而SmartUPload解决中文的问题也非常麻烦

使用Struts进行文件上传

从要导入的jar包我们就可以知道:Struts内部还是使用fileUpload上传组件....但是它极大的简化地我们的具体操作

那我们怎么用它呢??看下面的图

6af46d55be49c2ff2b90162e597607b0.png

在Action中使用在表单中定义的name,就可以获取代表的上传文件的File对象

在Action中使用在表单中定义的name+FileName,就得到上传文件的名字

JSP页面

在注册页面上拥有两个上传文件控件

Action

得到相对应的File对象、上传文件名称、上传文件的类型

package fileupload;

import java.io.File;

/**

* Created by ozc on 2017/5/2.

*/

public class FileUploadAction {

//上传文件对应的File对象

private File photo;

private File photo1;

//得到上传文件的名称

private String photoFileName;

private String photo1FileName;

//得到上传文件的类型

private String photoContentType;

private String photo1ContentType;

//给出相对应的setter

public void setPhoto(File photo) {

this.photo = photo;

}

public void setPhoto1(File photo1) {

this.photo1 = photo1;

}

public void setPhotoFileName(String photoFileName) {

this.photoFileName = photoFileName;

}

public void setPhoto1FileName(String photo1FileName) {

this.photo1FileName = photo1FileName;

}

public void setPhotoContentType(String photoContentType) {

this.photoContentType = photoContentType;

}

public void setPhoto1ContentType(String photo1ContentType) {

this.photo1ContentType = photo1ContentType;

}

public String register() {

System.out.println(photo1FileName);

System.out.println(photoFileName);

return "success";

}

}

成功得到数据:

fd5e119664d35457831593844a548d51.png

9dd5d31617af9e0984911146fcd3ab0d.png

Action业务代码:

public String register() throws IOException {

//得到上传的路径

String path = ServletActionContext.getServletContext().getRealPath("upload");

System.out.println(path);

//创建文件对象

File destFile = new File(path,photoFileName);

//调用工具类方法,将文件拷贝过去

FileUtils.copyFile(photo, destFile);

return "success";

}

效果:

c26d15e912a8c74f96efab7d76004c39.png

文件下载

我们以前是通过设置request消息头来实现文件下载的.....那么在Struts又如何实现文件下载呢??

我们请求服务器处理都是通过Action类来完成的,但是呢,Action类的业务方法都是返回字符串。因此,Struts在节点中提供了类型为stream的type值。通过stream来配置相对应的信息,从而实现下载!

列出所有可以下载的文件

Action类的业务方法

public class downLoadAction {

//列出所有可以下载的文件

public String list() {

//得到upload文件夹

String path = ServletActionContext.getServletContext().getRealPath("/upload");

//创建file对象

File file = new File(path);

//列出文件下所有的文件

File[] files = file.listFiles();

//将这些文件存到request域中

HttpServletRequest request = ServletActionContext.getRequest();

request.setAttribute("files", files);

return "list";

}

}

Struts配置文件

/list.jsp

JSP显示页面

对不起,没有下载的页面

编号文件名称操作
${file.count}${fn:substringAfter(fileName, "upload\\")}

下载

Action代码:

/**

* 访问Action的业务方法仅仅返回的是字符串。因此Struts在result节点提供了stream类型的type,

* 指定了stream就代表着我这是要下载的...

*

* 既然要下载文件,那么肯定需要几样东西:

* 1、文件名

* 2、代表文件的流

*/

public String downLoad() {

return "downLoad";

}

//得到要下载的文件名,Struts提供了自动封装的功能

private String fileName;

//如果文件名是中文的,那么需要手动转换,因为超链接是get方法提交

public void setFileName(String fileName) throws UnsupportedEncodingException {

fileName = new String(fileName.getBytes("ISO8859-1"), "UTF-8");

this.fileName = fileName;

System.out.println(fileName);

}

//得到代表下载文件流,该方法由Struts调用

public InputStream getAttrInputStream() {

return ServletActionContext.getServletContext().getResourceAsStream("/upload/" + fileName);

}

//下载时,显示的名称【如果是中文,可能会乱码,因此要URLencode】---->在Struts.xml文件中通过${}可获取

public String getDownFileName() throws UnsupportedEncodingException {

fileName = URLEncoder.encode(fileName, "UTF-8");

return fileName;

}

Struts.xml

/list.jsp

application/octet-stream

attrInputStream

attachment;filename=${downFileName}

1024

效果

1460000013651262?w=1903&h=848

模型驱动

什么是模型驱动

在Struts2中模型驱动就是用来封装数据的..完成数据的自动封装.

为什么要使用模型驱动?

我们之前就使用过Sturts2的数据自动封装功能,是用params拦截器完成的...既然有了params拦截器,为啥还要模型驱动??

当我们使用params拦截器完成数据自动封装的时候,如果要封装的是JavaBean对象,那么在web表单中就必须的name写上javaBean.属性名....

这样的话,web层和Action层就耦合了...因为在web层必须要知道封装的JavaBean对象是什么才能够实现自动封装!

而模型驱动就解决了这个问题!即时不知道Action层的JavaBean对象是什么,也能够完成数据自动封装!

模型驱动的实现原理

实现模型驱动功能也是由拦截器完成的,我们来看看拦截器到底做了什么吧....

拦截方法的源码是这样的:

public String intercept(ActionInvocation invocation) throws Exception {

//得到当前要执行的Action对象

Object action = invocation.getAction();

//判断该Action对象是否实现了ModelDriven接口

if(action instanceof ModelDriven) {

ModelDriven modelDriven = (ModelDriven)action;

//获取值栈对象

ValueStack stack = invocation.getStack();

//得到model的对象

Object model = modelDriven.getModel();

//把对象存到值栈对象中

if(model != null) {

stack.push(model);

}

if(this.refreshModelBeforeResult) {

invocation.addPreResultListener(new ModelDrivenInterceptor.RefreshModelBeforeResult(modelDriven, model));

}

}

return invocation.invoke();

}

把model对象放到值栈对象之后,Parameters 拦截器将把表单字段映射到 ValueStack 栈的栈顶对象的各个属性中.

也就是说,使用模型驱动是需要配合Params拦截器完成的!

使用数据模型驱动

实现ModelDriven接口

实现ModelDriven接口,重写方法....实现接口时,要封装的对象是什么,形参类型就给什么

public class UserAction extends ActionSupport implements ModelDriven {

public String login() {

return SUCCESS;

}

@Override

public User getModel() {

return null;

}

}

对象实例化

public class UserAction extends ActionSupport implements ModelDriven {

//这里一定要实例化

User user = new User();

public User getUser() {

return user;

}

public void setUser(User user) {

this.user = user;

}

@Override

public User getModel() {

return user;

}

}

测试

JSP提交页面,直接写上JavaBean对象的属性就行了..不需要写上JavaBean对象的名称!

用户名:
密码:
电话:
邮箱:

在Action业务方法中输出User对象的数据

@Override

public String execute() throws Exception {

System.out.println(user);

return SUCCESS;

}

3bc99e7141ef0536490a0e168afbca00.png

如果文章有错的地方欢迎指正,大家互相交流。习惯在微信看技术文章,想要获取更多的Java资源的同学,可以关注微信公众号:Java3y

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

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

相关文章

hql 查询条件 set集合_Redis从入门到深入-Sorted_set的value

1. sorted_set 类型新的存储需求,数据排序有利于数据的有效展示,需要提供一种可以根据自身特征进行排序的方式需要的存储结构:新的存储模型,可以保存可排序的数据sorted_set类型:在set的存储结构基础上添加可排序字段2…

pdm生成java_PowerDesigner通过SQL语句生成PDM文件并将name和comment进行互相转换

本篇文章主要介绍了PowerDesigner通过SQL语句生成PDM文件并将name和comment进行互相转换 超详细过程(图文),具有一定的参考价值,感兴趣的小伙伴们可以参考一下1.软件准备软件:Navicat 11.1,Powerdesigner 152.安装步骤第一步&…

java里面如何加入高级的东西_如何成为一名Java高级架构师

近些年来互联网快速发展,现阶段的数据量和高并发的诉求,引起了不少传统的技术人员的力不从心,企业愈发关注到了系统架构的重要性,既需要掌控整体又需要洞悉局部瓶颈并依据具体的业务场景给出解决方案的领导型人物——Java架构师应…

mysql获取最好成绩对应数据的其他项_开源数据同步神器——canal

前言如今大型的IT系统中,都会使用分布式的方式,同时会有非常多的中间件,如redis、消息队列、大数据存储等,但是实际核心的数据存储依然是存储在数据库,作为使用最广泛的数据库,如何将mysql的数据与中间件的…

window访问不到linux的activemq_springboot整合开源项目测试拦截接口访问的频率

本节目标本篇文章主要是介绍以下springboot整合开源项目,实现反爬虫接口防刷的demo,额外的介绍一下axios的基本用法;所以本篇文章阅读起来相对轻松。OK,下面开始整合.引入项目依赖cn.keking.project kk-anti-reptile …

sql server限制查询条数_18. Django 2.1.7 查询集 QuerySet

上一篇Django 2.1.7 模型 - 条件查询 F对象 Q对象 聚合查询讲述了关于Django模型的介绍F对象、Q对象、聚合查询等功能。不管什么查询,返回的结果都基本是查询集QuerySet,如下:In [16]: MiddlewareInfo.objects.all()Out[16]: 1)>, 2)>,…

ecs 云服务器 管理控制台_阿里云ECS服务器监控资源使用情况

阿里云Ecs服务器运维过程中,无论是Centos系统还是Windows系统,有时候我们需要监控分析最新的服务器资源利用率等运行情况,例如最近3个小时CPU使用率情况、内存使用率、网络流入带宽、网络流出带宽、服务器TCP连接数等数据状况。上述提到的这些…

svg 地图_找地图素材?有这个网站就够了!

文 | 孙宁使用地图类的素材最担心什么?答案是:地图数据的准确性!设想,你辛辛苦苦找来的地图素材,却被人发现是一个假的地图,那可是要犯错误滴~今天给大家推荐一个地图网站,完全不用担心这个问题…

hadoop设置java环境_hadoop安装(3)_Linux配置JDK环境

hadoop分布式集群安装的过程中要先给Linux系统配置好JDK环境,因为hadoop是基于Java开发的。所以本文主要介绍如何在centos上安装jdk环境,主要步骤如下:1、去Oracle官网下载Linux版本的jdk,地址如下:笔者下载的是JDK1.8…

sqlplus怎样将名次显示在表的后面_数据透视表-01 3种分析方法比拼,助你步入透视表的知识宝殿...

数据透视表通俗的讲就是一种在EXCEL表中对数据进行分析汇总的方式,它可以方便的排列、汇总复杂数据,更加清晰的查看、分析隐藏在数据内部的信息。另外在菜单栏数据选项下的合并计算和分类汇总也可以进行简单的统计分析下面我们通过一个小例子来了解一下合…

php引用计数器,PHP 引用计数器 通俗版解释_PHP教程

PHP 引用计数器 通俗版解释概述最近看PHP中的引用计数器部分,首先被各种绕晕,然后通过看博客和分析后,总结了一个比较通俗的解释,能帮助自己很好地记忆,也希望能帮助到各位读者。这里分享一遍博文,是比较正…

windows c语言 http https检测_C语言编程工具的选择

1.先说说我的故事我是2007年上的大学,一所双一流工科大学。专业是计算机科学与技术。我们计算机专业没有C语言这门课。其他工科专业会学C语言这门课。据我所知,其他的工科专业需要过C语言2级考试。计算机专业不用考2级、3级...这些考试。我们计算机专业学…

php 按照laravel5.5,Laravel5.5 综合使用

使用 Laravel5.5 开发一个自动交割的项目,把使用到的开源扩展包及特性整理起来,以供后续使用。一、安装IDE提示工具Laravel IDE Helper 是一个极其好用的代码提示及补全工具,可以给编写代码带来极大的便利。1、安装# 如果只想在开发环境安装请…

usb声卡驱动_iCON ProDrive第三代USB声卡驱动全新发布!

2019年3月,iCON官方发布了一款全球首创——全新一代“ProDrive III”USB声卡驱动,iCON所有系列USB声卡(包括带声卡的MIDI键盘)已全面启用ProDrive III !iCON ProDrive III由iCON的国际工程师团队微调的高品质模拟元件和电路,可同时…

分数小数互换图_重复控制器学习心得(二)——超前环节的分数化和校正因子的引入...

本文仅作为学习记录,欢迎各位提出宝贵建议1、超前补偿环节的分数化在上周我学习了分数阶RC的实现方式,无独有偶,超前补偿环节也可以应用插值的方式来近似分数阶补偿。整数阶超前补偿的弊端主要有二:一、当采样频率和固定频率的比值…

java中execquery(),javaee登录界面

首先在此之前我们应该正确安装数据库,以及eclipse(javaee)文件包。1.首先在javaee中建立新的项目2.右键点击WebContent-New-JSP File,新建jsp(动态)文件。(想在哪个文件夹里添加jsp文件,就直接右击文件夹新建,注意一定不要展开文件…

电子工程专业评副高总结_微电子科学与工程专业怎么样?

专业简介080704 微电子科学与工程培养目标:本专业培养德、智、体等方面全面发展,具备微电子科学与工程专业扎实的自然科学基础、系统的专业知识和较强的实验技能与工程实践能力,能在微电子科学技术领域从事研究、开发、制造和管理等方面工…

vue if判断_vue循环判断

<!DOCTYPE html> <html><head><meta charset"utf-8"><title>条件与循环</title><script src"https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script></head><body><h1>判断示例<…

显示器驱动有什么用_科普一下:电脑显示器用什么接口好,主流接口有哪些?...

如今是新的“视”界&#xff0c;生活中总与各种屏幕打交道&#xff0c;难免会遇到选择视频接线的问题&#xff0c;要想搞清楚这点&#xff0c;我们只要通过了解现今常用的几种视频接线就会有个大致的认识。经常会有一些朋友困扰&#xff0c;电脑显示器用什么接口最好&#xff0…

更改应用程序图标_在 Windows 10 version 1903 中查看应用程序是否支持 DPI 感知

在显示屏幕上&#xff0c;每英寸点数(DPI)定义构成面板的每个物理单元的像素数。传统上&#xff0c;显示器每英寸输出 96 个像素。但随着技术的发展&#xff0c;各大公司的产品已经开始增加同一物理空间中的像素数&#xff0c;也就造成了屏幕的 DPI 达到 200 甚至更高&#xff…