鸿蒙开发(Harmonyos兼容与Harmonyos适配)

布局的实现

Layout_ability_main.xml布局:

<?xml version="1.0" encoding="utf-8"?>
<DirectionalLayoutxmlns:ohos="http://schemas.huawei.com/res/ohos"ohos:height="match_parent"ohos:width="match_parent"ohos:alignment="center"ohos:orientation="vertical"><TextFieldohos:id="$+id:content"ohos:height="0vp"ohos:weight="1"ohos:width="match_parent"ohos:enabled="false"ohos:padding="20vp"ohos:text="0"ohos:text_alignment="center"ohos:text_size="30"/><ScrollViewohos:height="0vp"ohos:weight="1.8"ohos:width="match_parent"><DirectionalLayoutohos:height="match_content"ohos:width="match_parent"ohos:alignment="horizontal_center"ohos:orientation="vertical"><DirectionalLayoutohos:height="match_content"ohos:width="match_parent"ohos:orientation="horizontal"><Buttonohos:id="$+id:reset"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="c"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:except"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="÷"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:ride"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="x"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:delete"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="←"ohos:text_size="30vp"ohos:weight="1"/></DirectionalLayout><DirectionalLayoutohos:height="match_content"ohos:width="match_parent"ohos:orientation="horizontal"><Buttonohos:id="$+id:seven"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="7"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:eight"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="8"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:nine"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="9"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:reduce"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="-"ohos:text_size="30vp"ohos:weight="1"/></DirectionalLayout><DirectionalLayoutohos:height="match_content"ohos:width="match_parent"ohos:orientation="horizontal"><Buttonohos:id="$+id:four"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="4"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:five"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="5"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:six"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="6"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:add"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="+"ohos:text_size="30vp"ohos:weight="1"/></DirectionalLayout><DirectionalLayoutohos:height="match_content"ohos:width="match_parent"ohos:orientation="horizontal"><Buttonohos:id="$+id:one"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="1"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:two"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="2"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:three"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button1"ohos:margin="6vp"ohos:text="3"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:decimal_point"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="."ohos:text_size="30vp"ohos:weight="1"/></DirectionalLayout><DirectionalLayoutohos:height="match_content"ohos:width="match_parent"ohos:orientation="horizontal"><Buttonohos:id="$+id:percentage"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="%"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:zero"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button2"ohos:margin="6vp"ohos:text="0"ohos:text_size="30vp"ohos:weight="1"/><Buttonohos:id="$+id:equal"ohos:height="70vp"ohos:width="0vp"ohos:background_element="$graphic:background_button3"ohos:margin="6vp"ohos:text="="ohos:text_size="30vp"ohos:weight="2"/></DirectionalLayout></DirectionalLayout></ScrollView>
</DirectionalLayout>

background_button1.xml背景样式:

<?xml version="1.0" encoding="utf-8"?>
<shapexmlns:ohos="http://schemas.huawei.com/res/ohos"ohos:shape="rectangle"><solid ohos:color="#CCF1F1F1"/><corners ohos:radius="20vp"/>
</shape>

background_button2.xml背景样式:

<shapexmlns:ohos="http://schemas.huawei.com/res/ohos"ohos:shape="rectangle"><solid ohos:color="#FFE4F2FE"/><corners ohos:radius="20vp"/>
</shape>

background_button3.xml背景样式:

<?xml version="1.0" encoding="utf-8"?>
<shapexmlns:ohos="http://schemas.huawei.com/res/ohos"ohos:shape="rectangle"><solid ohos:color="#FF007CFD"/><corners ohos:radius="20vp"/>
</shape>

嗯,编写布局页面不难、稍微难点的是电视、车载设备、Pad、手机、手表五个端的屏幕适配。

界面编写完,发现各个端的屏幕高度还没有做适配,一开始认为Android与HarmonyOS用Java语言都可以编写,HarmonyOS也可以使用Android的相关框架,便想着如何在HarmonyOS上去使用Android的屏幕适配方案,在用了今日头条的屏幕适配方案开刀后,发现压根行不通,今日头条的屏幕适配方案用的单位是dp,这个单位在HarmonyOS上并没有,只有类似的vp,看来还是我太天真了。

Android屏幕单位有dp、in、mm、pt、px、sp,HarmonOS屏幕单位有fp、px、vp。

其中两者相同的单位是px,Android的dp与HarmonOS的vp都是为各自设备量身打造的单位,若想要搞一个两者都可以用的屏幕适配框架,也许,只能从px找突破口。今日头条的屏幕适配方案用的单位虽然是HarmonyOS所没有的dp,但其实它最终都是要拿dp来转换成px的喔~

Java代码逻辑

继承AbilitySlice的MainAbilitySlice类:

public class MainAbilitySlice extends AbilitySlice implements Component.ClickedListener {private Utils utils = Utils.getInstance();private TextField content;private String formula = "";@Overridepublic void onStart(Intent intent) {super.onStart(intent);super.setUIContent(ResourceTable.Layout_ability_main);initView();highlyAdaptive();}/*** 各个按钮点击事件* @param component*/@Overridepublic void onClick(Component component) {switch (component.getId()) {case ResourceTable.Id_one:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "1");else formula = "1";break;case ResourceTable.Id_two:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "2");else formula = "2";break;case ResourceTable.Id_three:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "3");else formula = "3";break;case ResourceTable.Id_four:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "4");else formula = "4";break;case ResourceTable.Id_five:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "5");else formula = "5";break;case ResourceTable.Id_six:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "6");else formula = "6";break;case ResourceTable.Id_seven:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "7");else formula = "7";break;case ResourceTable.Id_eight:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "8");else formula = "8";break;case ResourceTable.Id_nine:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "9");else formula = "9";break;case ResourceTable.Id_zero:if (utils.isNumStart(formula)) formula = utils.isZero(formula, "0");else formula = "0";break;case ResourceTable.Id_reset:formula = "0";break;case ResourceTable.Id_except:if (utils.isNumEnd(formula)) formula += "÷";else if (!formula.substring(formula.length() - 1, formula.length()).equals("."))formula = formula.substring(0, formula.length() - 1) + "÷";break;case ResourceTable.Id_ride:formula = utils.isNum(formula, "x");break;case ResourceTable.Id_percentage:formula = utils.isNum(formula, "%");break;case ResourceTable.Id_decimal_point:if (utils.isNumEnd(formula) && !utils.isDecimals(formula)) formula += ".";break;case ResourceTable.Id_delete:if (!formula.equals("") && !formula.equals("0")) {formula = formula.substring(0, formula.length() - 1);if (formula.equals("")) formula = "0";}break;case ResourceTable.Id_reduce:if (utils.isNumEnd(formula)) formula += "-";else formula = formula.substring(0, formula.length() - 1) + "-";break;case ResourceTable.Id_add:if (utils.isNumEnd(formula)) formula += "+";else formula =formula.substring(0, formula.length() - 1) + "+";break;case ResourceTable.Id_equal:equal();break;default:break;}if (component.getId() != ResourceTable.Id_equal) {content.setText(formula);}}private void equal() {if (formula.equals("")) {// 如果没有输入公式utils.toast(this, "还没输入公式呢");return;} else if (!utils.isNumEnd(formula)) {// 如果公式的最后一位数非数字utils.toast(this, "计算器表示没见过这样的数学公式,运算不出来");return;}String[] split;if (!utils.isContains(formula, ".")) {// 计算整数if (utils.isContains(formula, "-")) {// 减法split = formula.split("-");if (split.length > 1)result((Integer.parseInt(split[0]) + Integer.parseInt(split[1])) + "");} else if (utils.isContains(formula, "+")) {// 加法split = formula.split("\\+");if (split.length > 1)result((Integer.parseInt(split[0]) + Integer.parseInt(split[1])) + "");} else if (utils.isContains(formula, "x")) {// 乘法split = formula.split("x");if (split.length > 1)result((Integer.parseInt(split[0]) + Integer.parseInt(split[1])) + "");} else if (utils.isContains(formula, "÷")) {// 除法split = formula.split("÷");if (split.length > 1)result((Integer.parseInt(split[0]) + Integer.parseInt(split[1])) + "");} else if (utils.isContains(formula, "%")) {// 取余split = formula.split("%");if (split.length > 1)result((Integer.parseInt(split[0]) + Integer.parseInt(split[1])) + "");}} else {// 计算小数if (utils.isContains(formula, "-")) {// 减法split = formula.split("-");if (split.length > 1)result((Double.parseDouble(split[0]) - Double.parseDouble(split[1])) + "");} else if (utils.isContains(formula, "+")) {// 加法split = formula.split("\\+");if (split.length > 1)result((Double.parseDouble(split[0]) - Double.parseDouble(split[1])) + "");} else if (utils.isContains(formula, "x")) {// 乘法split = formula.split("x");if (split.length > 1)result((Double.parseDouble(split[0]) - Double.parseDouble(split[1])) + "");} else if (utils.isContains(formula, "÷")) {// 除法`split = formula.split("÷");if (split.length > 1)result((Double.parseDouble(split[0]) - Double.parseDouble(split[1])) + "");} else if (utils.isContains(formula, "%")) {// 取余split = formula.split("%");if (split.length > 1)result((Double.parseDouble(split[0]) - Double.parseDouble(split[1])) + "");}}}private void result(String value) {formula = value;content.setText(value);}/*** 根据不同设备调整高度*/private void highlyAdaptive() {if (DeviceInfo.getDeviceType().equals("phone")) {// 手机设备ComponentContainer.LayoutConfig layoutConfig = new ComponentContainer.LayoutConfig();layoutConfig.height = 1100;content.setLayoutConfig(layoutConfig);} else if (DeviceInfo.getDeviceType().equals("tablet")) {// 平板设备ComponentContainer.LayoutConfig layoutConfig = new ComponentContainer.LayoutConfig();layoutConfig.height = 1200;content.setLayoutConfig(layoutConfig);} else if (DeviceInfo.getDeviceType().equals("tv")) {// TV设备ComponentContainer.LayoutConfig layoutConfig = new ComponentContainer.LayoutConfig();layoutConfig.height = 160;content.setLayoutConfig(layoutConfig);} else if (DeviceInfo.getDeviceType().equals("wearable")) {// 可穿戴设备ComponentContainer.LayoutConfig layoutConfig = new ComponentContainer.LayoutConfig();layoutConfig.height = 150;content.setLayoutConfig(layoutConfig);} else if (DeviceInfo.getDeviceType().equals("car")) {// 车载设备ComponentContainer.LayoutConfig layoutConfig = new ComponentContainer.LayoutConfig();layoutConfig.height = 500;content.setLayoutConfig(layoutConfig);}}/*** 初始化xml布局控件*/private void initView() {content = (TextField) findComponentById(ResourceTable.Id_content);((Button) findComponentById(ResourceTable.Id_one)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_two)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_three)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_four)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_five)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_six)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_seven)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_eight)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_nine)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_zero)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_reset)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_except)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_ride)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_delete)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_reduce)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_add)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_equal)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_decimal_point)).setClickedListener(this);((Button) findComponentById(ResourceTable.Id_percentage)).setClickedListener(this);}
}

由于在编写xml UI时屏幕适配只能做到宽度适配或高度适配,没办法在一个xml界面同时适配宽度与高度,为此写了一个highlyAdaptive方法处理xml没能完成的高度适配,方法通过DeviceInfo.getDeviceType()来得到设备的类型,根据不同的设备去修改它的高度,也算是实现了高度适配。

Utils类:

public class Utils {private static Utils utils = new Utils();private static ToastDialog toastDialog;private String[] symbol = new String[]{"+", "-", "x", "÷", "%"};public static Utils getInstance() {return utils;}public void toast(Context context, String text) {if (toastDialog == null) {toastDialog = new ToastDialog(context);}toastDialog.setAlignment(LayoutAlignment.CENTER);toastDialog.setText(text);toastDialog.show();}/*** 判断最后一位是否数字* @param content*/public boolean isNumber(String content){char[] chars = content.substring(content.length() - 1, content.length()).toCharArray();return Character.isDigit(chars[0]);}/*** 判断是否是小数*/public boolean isDecimals(String str) {if (isDecimal(str)) {for (String s : symbol) {if (isContains(str, s)) {String[] split = str.split(s);if (split != null){if (!isDecimal(split[split.length - 1])) {return false;} else {return true;}}}}return true;}return false;}/*** 判断一位数是否是小数*/public boolean isDecimal(String str) {if (isContains(str, "."))return true;elsereturn false;}/*** 是否包含某一个运算符*/public boolean isContains(String value, String contain) {if (value.indexOf(contain) == -1)return false;elsereturn true;}/*** 最后一个值是数字就加符号,不是数字则替换它* @param str 符号*/public String isNum(String content,String str) {if (isNumEnd(content)) content += str;else content = content.substring(0, content.length() - 1) + str;return content;}/*** 第一个值是0,输入整数则替换掉*/public String isZero(String content,String str) {if (content.equals("0")) {content = str;} else {content += str;}return content;}/*** 得到第一个值是否是数字*/public boolean isNumStart(String str) {if (str.startsWith("+") || str.startsWith("x") || str.startsWith("÷") || str.startsWith("%") || str.equals("")) {return false;}return true;}/*** 得到最后一个值是否是数字*/public boolean isNumEnd(String str) {char[] chars = str.substring(str.length() - 1, str.length()).toCharArray();if (!Character.isDigit(chars[chars.length - 1])) {return false;}return true;}
}

GIF演示实现效果

  • Phone 设备实现效果

  • Pad 设备实现效果

  • TV 设备实现效果

  • Wearable 设备实现效果

    目前所有设备中,Wearable是几个设备中最不好适配、最难适配的设备,但,想实现也并非不可能。
    如果继续适配Wearable,目前能想到Wearable屏幕适配的方法有三种:
    1、需要将背景换成一个圆,按钮都放进一个自动换行的组件。只是,这个想法不是很现实,Android的RecycleView组件也只是一行固定多少个才会换行,HarmonyOS的ListContainer组件能否实现效果还是个未知数。
    2、使用他人开源的屏幕适配框架。不过,这个很遗憾,截止至发稿,还未能了解到有相关的适配框架。
    3、另外写一个适配Wearable的布局。在onState方法执行super.setUIContent前更换专门为Wearable而写的xml,如:

@Overridepublic void onStart(Intent intent) {super.onStart(intent);// wearable设备换一个布局if (DeviceInfo.getDeviceType().equals("wearable")){super.setUIContent(Wearable布局);}else{super.setUIContent(ResourceTable.Layout_ability_main);}}

Car 实现效果

截止至发稿,Car还没有开放对应的机型,没能使用远程真机进行测试查看最终效果。这个效果图也只是点击Previewer进行查看的样式及效果。

Previewer注意事项:

1、点击Previewer查看xml,偶尔点击xml的一些样式并不会有响应,需要关闭Previewer并重新打开。

2、Previewer展示的样式不会显示ToastDialog等对话框、不会打印日志、不能点击Debug进行测试,还是使用真机测试真机测试香。

此次是我自HarmonyOS的DevEco Studio开发工具发布以来第一次开发的APP,身为一个Android开发工程师,做起HarmonyOS开发并不是很难,其中有很多东西都类似。DevEco Studio的远程真机测试与Previewer,效果杠杠的,要知道网上很多远程真机测试可都是收费制,且按使用时间收费,这一功能的出现可降低了不少开发费用。

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

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

相关文章

docker容器生命周期管理命令

文章目录 前言1、docker create2、docker run2.1、常用选项2.2、系统2.3、网络2.4、健康检查 3、docker start/stop/restart4、docker kill5、docker rm6、docker pause/unpause总结 前言 在云原生时代&#xff0c;Docker已成为必不可少的容器管理工具。通过掌握Docker常用的容…

2. MySQL 多实例

重点&#xff1a; MySQL 的 三种安装方式&#xff1a;包安装&#xff0c;二进制安装&#xff0c;源码编译安装。 MySQL 的 基本使用 MySQL 多实例 DDLcreate alter drop DML insert update delete DQL select 2.5&#xff09;通用 二进制格式安装 MySQL 2.5.1&#xff…

立创EDA学习:设计收尾工作

布线整理 ShiftM&#xff0c;关闭铺铜显示 调整结束后再使用快捷键”ShiftM“打开铺铜 过孔 在空白区域加上一些GND过孔&#xff0c;连接顶层与底层的铺铜。放置好”过孔“后&#xff0c;隐藏铺铜&#xff0c;观察刚才放置的过孔有没有妨碍到其他器件 调整铺铜 先打开铺铜区&…

用可视化案例讲Rust编程4. 用泛型和特性实现自适配shapefile的读取

本节已经涉及Rust学习曲线上的一个大坑&#xff1a;泛型和特性了&#xff0c;属于语言的深水区&#xff0c;如果初学者&#xff0c;建议看一眼知道有这个功能即可。 如果我们立足于功能实现&#xff0c;那么做到像上一节那样就可以了&#xff0c;从原理上来说&#xff0c;每个…

TCP 状态转换以及半关闭

TCP 状态转换&#xff1a; 上图中还没有进行握手的时候状态是关闭的。 三次握手状态的改变&#xff1a; 客户端发起握手。 调用 connect() 函数时状态转化为&#xff1a;SYN_SENT。调用 listen() 函数时状态转换为&#xff1a;LISTEN。ESTABLISHED是被连接的状态。 四次挥手…

C++核心编程:C++ 中的引用 笔记

2.引用 2.1 引用的基本使用 - 作用&#xff1a;给变量起别名 - 语法&#xff1a;数据类型 &别名 原名 #include<iostream> using namespace std; int main() {// 引用基本语法// 数据类型 &别名 原名int a 10;// 创建引用int &ref_a a;cout<<&qu…

代码随想录刷题笔记 DAY15 | 翻转二叉树 No.226 | 对称二叉树 No.101

Day 15 01. 翻转二叉树&#xff08;No. 226&#xff09; 题目链接 代码随想录题解 1.1 题目 给你一棵二叉树的根节点 root &#xff0c;翻转这棵二叉树&#xff0c;并返回其根节点。 示例 1&#xff1a; 输入&#xff1a;root [4,2,7,1,3,6,9] 输出&#xff1a;[4,7,2,9…

PCB制板基础知识

一、PCB概念 PCB&#xff08;PrintedCircuitBoard&#xff09;&#xff0c;中文名称为印制电路板&#xff0c;又称印刷电路板、印刷线路板&#xff0c;是重要的电子部件&#xff0c;是电子元器件的支撑体&#xff0c;是电子元器件电气连接的提供者。由于它是采用电子印刷术制作…

使用vs2022将.net8的应用程序发布为一个单独文件

在使用.NetCore3.1时&#xff0c;可以通过设置以下工程配置文本来将项目发布为一个单独的应用程序文件&#xff1a; <Project Sdk"Microsoft.NET.Sdk.WindowsDesktop"><PropertyGroup><TargetFramework>netcoreapp3.1</TargetFramework><…

Python open函数

在Python编程中&#xff0c;open()函数是一个重要的文件操作函数&#xff0c;用于打开文件并进行读取、写入、追加等操作。本文将深入探讨open()函数的用法、语法、文件模式、示例代码&#xff0c;并探讨其在实际编程中的应用场景。 什么是open()函数&#xff1f; open()函数…

机房及设备安全智慧监管AI+视频方案的设计和应用

一、背景分析 随着互联网的迅猛发展&#xff0c;机房及其配套设施的数量持续攀升&#xff0c;它们的运行状况对于企业运营效率和服务质量的影响日益显著。作为企业信息化的基石&#xff0c;机房的安全监测与管理的重要性不容忽视。它不仅关乎企业的稳定运营&#xff0c;同时也…

《WebKit技术内幕》学习之十五(3): Web前端之未来

3 Web应用和Web运行环境 3.1 Web应用 HTML5提供了强大的能力&#xff0c;而不是支持Web网页这么简单。就目前而言&#xff0c;它已经初步提供了支持Web网页向Web应用方向发展的能力。相对于本地应用&#xff08;Native Application&#xff09;&#xff0c;Web前端领域也能够…