【软通动力】安卓VS鸿蒙第三方件切换宝典 V1.0 原创 精华
众所周知,安卓应用开发经过这么多年的发展相对成熟和稳定,鸿蒙OS作为后来者兼容一个成熟的开发体系会节省很多推广和开发成本。但在实际开发中,代码层面仍然有很多细节上的差异,会给初次开发人员造成困扰。
本宝典旨在汇总实际开发中第三方件接入时的代码差异,以期帮助开发人员更好的进行开发作业,由于目前接触的开发类型有限,所汇总的内容多少会有疏漏,后期我们会进一步完善和补全。
欢迎关注我们以及我们的专栏,方便您及时获得相关内容的更新。
※基础功能
1.获取屏幕分辨率
安卓:
getWindowManager().getDefaultDisplay();
鸿蒙:
Optional<Display>
display = DisplayManager.getInstance().getDefaultDisplay(this.getContext());
Point pt = new Point();
display.get().getSize(pt);
2.隐藏标题栏TitleBar
安卓:
略
鸿蒙:
confi.json中添加如下描述:
""metaData"":{
""customizeData"":[
{
""name"": ""hwc-theme"",
""value"": ""androidhwext:style/Theme.Emui.NoTitleBar"",
""extra"":""""
}
]
}
3.获取屏幕密度
安卓:
Resources.getSystem().getDisplayMetrics().density
鸿蒙:
// 获取屏幕密度
Optional<Display>
display = DisplayManager.getInstance().getDefaultDisplay(this.getContext());
DisplayAttributes displayAttributes = display.get().getAttributes();
//displayAttributes.xDpi;
//displayAttributes.yDpi;
4.获取上下文
安卓:
context
鸿蒙:
getContext()
5.组件的父类
安卓:
android.view.View; class ProgressBar extends View
鸿蒙:
class ProgressBar extends Component
6.沉浸式显示
安卓:
略
鸿蒙:
A:在config.json ability 中添加
"metaData"": {
""customizeData"": [
{
""extra"": """",
""name"": ""hwc-theme"",
""value"": ""androidhwext:style/Theme.Emui.Light.NoTitleBar""
}
]
}
B:在AbilitySlice的onStart函数内增加如下代码,注意要在setUIContent之前。
getWindow().addFlags(WindowManager.LayoutConfig.MARK_TRANSLUCENT_STATUS);
7.获取运行时权限
安卓:
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.CALL_PHONE}, 1)
鸿蒙:
requestPermissionsFromUser(
new String[]{""ohos.permission.READ_MEDIA"", ""ohos.permission.WRITE_MEDIA"", ""ohos.permission.READ_USER_STORAGE"", ""ohos.permission.WRITE_USER_STORAGE"",}, 1);
※布局&组件
1.页面跳转(显示跳转)
安卓:
A.从A跳转至B,没有参数,并且不接收返回值
Intent intent = new Intent();
intent.setClass(A.this, B.class);
startActivity(intent);
B.从A跳转至B,有参数,不接收返回值
Intent intent = new Intent(this, B.class);
intent.putExtra(""name"", ""lily"");
startActivity(intent);
C.从A跳转至B,有参数,接收返回值
Intent intent = new Intent(this, B.class);
intent.putExtra(""name"", ""lily"");
startActivityForResult(intent, 2);
鸿蒙:
A.从A跳转至B,没有参数,并且不接收返回值
present(new BSlice(), new Intent());
B.从A跳转至B,有参数,不接收返回值
Intent intent = new Intent();
Operation operation = new Intent.OperationBuilder()
.withDeviceId("""") .withBundleName(""com.test"") .withAbilityName(""com.test.BAbility"")
.build();
intent.setParam(""name"",""lily"");
intent.setOperation(operation);
startAbility(intent);
C.从A跳转至B,有参数,接收返回值
Intent intent = new Intent();
Operation operation = new Intent.OperationBuilder()
.withDeviceId("""") .withBundleName(""com.test"") .withAbilityName(""com.test.BAbility"")
.build();
intent.setParam(""name"",""lily"");
intent.setOperation(operation);
startAbilityForResult(intent,100);
2.页面跳转(隐式跳转)
安卓:
A.配置
<activity android:name="".B"">
<intent-filter>
<action android:name=""com.hly.view.fling""/>
</intent-filter>
</activity>
B.启动
Intent intent = new Intent(); intent.setAction(""com.hly.view.fling""); intent.putExtra(""key"", ""name""); startActivity(intent);
鸿蒙:
A.在config.json文件ability 中添加以下信息
"skills"":[
{
""actions"":[
""ability.intent.gotopage""
]
}
]
B.在MainAbility的onStart函数中,增加页面路由
addActionRoute( ""ability.intent.gotopage"", BSlice.class.getName());
C.跳转
Intent intent = new Intent();
intent.setAction(""ability.intent.gotopage"");
startAbility(intent);
3.页面碎片
安卓:
Fragment
鸿蒙:
Fraction
A:Ability继承FractionAbility
B:获取Fraction调度器
getFractionManager().startFractionScheduler()
C:构造Fraction
D:调用调度器管理Fraction
FractionScheduler.add()
FractionScheduler.remove()
FractionScheduler.replace()
备注:
参考demo
https://www.jianshu.com/p/58558dc6673a"
4.从xml文件创建一个组件实例
安卓:
LayoutInflater.from(mContext).inflate(R.layout.banner_viewpager_layout, null);
鸿蒙:
LayoutScatter.getInstance(getContext()).parse(ResourceTable.Layout_ability_main, null, false);
5.组件自定义绘制
安卓:
ImageView.setImageDrawable(Drawable drawable);
并重写Drawable 的draw函数
鸿蒙:
Component.addDrawTask(Component.DrawTask task);
并实现Component.DrawTask接口的onDraw函数
6.自定义组件的自定义属性(在xml中使用)
安卓:
需要3步
A.在 values/attrs.xml,在其中编写 styleable 和 item 等标签元素。
B.在layout.xml中,增加
xmln:app= ""http://schemas.android.com/apk/res/-auto""
C.在自定义组件的构造函数中,调用array.getInteger(R.styleable.***, 100);获取属性
鸿蒙:
只需2步
A. 在组件定义的layout.xml中增加 xmlns:app=""http://schemas.huawei.com/apk/res/ohos""
然后就可以使用app:***(***为任意字符串)来增加自定义属性了,为了区分建议加上组件名前缀。
B. 在自定义组件的带AttrSet参数的构造函数中,使用下面代码获取属性。attrSet.getAttr(""***"").get().getStringValue();
7.触摸事件
安卓:
android.view.MotionEvent
鸿蒙:
ohos.multimodalinput.event.TouchEvent
8.事件处理
安卓:
android.os.Handler
鸿蒙:
ohos.eventhandler.EventHandler
9.控件触摸事件回调
安卓:
android.view.View.OnTouchListener
鸿蒙:
ohos.agp.components.Component.
TouchEventListener
10.轮播图继承的父类
安卓:
extends ViewPager
鸿蒙:
extends PageSlider
11.实现监听轮播图组件事件
安卓:
implements PageSlider.PageChangedListener
鸿蒙:
Implements OnPageChangedListener
12.touch事件监听
安卓:
直接重写onTouchEvent
鸿蒙:
继承 Component.TouchEventListener然后在构造方法中设置监听 setTouchEventListener(this::onTouchEvent);实现onTouchEvent
13.获取点击事件的坐标点
安卓:
event.getX(), event.getY()
鸿蒙:
MmiPoint point = touchEvent.getPointerPosition(touchEvent.getIndex());
14.调节滚轮中内容间距
安卓:
setLineSpacingMultiplier(float f)
鸿蒙:
setSelectedNormalTextMarginRatio(float f)
15.滚轮定位
安卓:
setPosition
鸿蒙:
setValue
16.Layout布局改变监听
安卓:
View.OnLayoutChangeListener
鸿蒙:
Component.LayoutRefreshedListener
17.组件容器
安卓:
ViewGroup
鸿蒙:
ComponentContainer
18.添加组件
安卓:
addView()
鸿蒙:
addComponent()
19.动态列表的适配器
安卓:
extends RecyclerView.Adapter<>
鸿蒙:
extends RecycleItemProvider
20.动态列表
安卓:
RecyclerView
鸿蒙:
ListContainer
21.文本域动态监听
安卓:
TextWatcher
鸿蒙:
Component.ComponentStateChangedListener
22.组件绘制自定义布局
安卓:
重写onLayout(boolean changed, int left, int top, int right, int bottom)
鸿蒙:
重写Component.LayoutRefreshedListener的onRefreshed方法
23.List组件
安卓:
ListView
鸿蒙:
ListContainer
24.设置背景颜色
安卓:
setBackgroundColor(maskColor);
鸿蒙:
// 创建背景元素
ShapeElement shapeElement = new ShapeElement();
// 设置颜色
shapeElement.setRgbColor(new RgbColor(255, 0, 0));
view.setBackground(shapeElement);
25.可以在控件上、下、左、右设置图标,大小按比例自适应
安卓:
setCompoundDrawablesWithIntrinsicBounds
鸿蒙:
setAroundElements
26.RadioButton组件在xml中如何设置checked属性
安卓:
在xml中可以设置
鸿蒙:
radioButton = findComponentById();
radioButton.setChecked(true);
备注:
sdk2.0后 xml中没有了checked属性,如果使用,可以在java代码中实现"
27.文本域动态监听
安卓:
TextWatcher
鸿蒙:
Component.ComponentStateChangedListener
28.颜色类
安卓:
java.awt.Color
鸿蒙:
ohos.agb.colors.rgbcolor
29.为ckeckbox或者Switch按钮设置资源图片
安卓:
略
鸿蒙:
VectorElement vectorElement = new VectorElement(this, ResourceTable.Graphic_candy);
setBackground(vectorElement)
30.子组件将拖拽事件传递给父组件
安卓:
略
鸿蒙:
注册setDraggedListener侦听,实现onDragPreAccept方法,再方法内根据拖拽方向判断是否需要父组件处理,如果需要则返回false,否则返回true
※资源管理
1.管理资源
安卓:
AssertManager
鸿蒙:
ResourceManager
2.获取应用的资源文件rawFile,并返回InputStream
安卓:
getResources()
AssetManager类
鸿蒙:
ResourceManager resourceManager = getContext().getResourceManager();
RawFileEntry rawFileEntry = resourceManager.getRawFileEntry(jsonFile);
Resource resource = null;
try {
resource = rawFileEntry.openRawFile();
} catch (IOException e) {
e.printStackTrace();
}
备注:
Resource是InputStream的子类,可以直接作为InputStream使用。"
3.获取文件路径
安卓:
Environment.getExternalStorageDirectory().getAbsolutePath()
鸿蒙:
获取文档(DIRECTORY_DOCUMENTS)、下载(DIRECTORY_DOWNLOADS)、视频(DIRECTORY_MOVIES)、音乐(DIRECTORY_MUSIC)、图片(DIRECTORY_PICTURES)
GetExternalFilesDir(Environment.DIRECTORY_PICTURES).getAbsolutePath()
※消息&多线程
1.事件循环器
安卓:
android.os.Looper
鸿蒙:
EventRunner.create(true)
备注:
有参数且为true,表示队列被托管,参数为false,或无参表示不被托管,需要eventRunner.run()调用
2.消息
安卓:
android.os.Message
鸿蒙:
InnerEvent
3.休眠
安卓:
android.os.SystemClock.sleep()
鸿蒙:
ohos.miscservices.timeutility.time;
Time.sleep(int millesend)
4.事件通知延迟消息
安卓:
Handler.postDelayed(MESSAGE_LOGIN, 5000);
鸿蒙:
Handler.postTask(task, 5000);
5.Intent传递参数
安卓:
Intent.putExtra or add Bundle
鸿蒙:
Intent.setParam
6.消息发送
安卓:
Handler handler = new Handler,通过handlerMsg发消息
鸿蒙:
InnerEvent event1 = InnerEvent.get(eventId1, param, object);
myHandler.sendEvent(event1, 0, Priority.IMMEDIATE);
7.更新UI
安卓:
class MyHandle extends Handler{
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case 1:
//更新UI的操作
break;
default:
Break;
}
}
}
鸿蒙:
abilitySlice.getUITaskDispatcher().asyncDispatch(() -> {
//更新UI的操作
});
※图片处理
1.位图资源
安卓:
Bitmap
鸿蒙:
PixelMap
2.图像缩放,拉伸到视图边界
安卓:
ImageView.ScaleType
image.setScaleType(ScaleType.FXY);
鸿蒙:
Image.ScaleMode
image.setScaleMode(Image.ScaleMode.STRETCH);
3.List组件&内容适配器
安卓:
ListView
extends BaeAdapter
ViewPage.setAdapter(BaeAdapter);
鸿蒙:
ListContainer
extends PageSlider
PageSlider.setProvider(PageSlider);
4.图片显示组件
安卓:
androidx.appcompat.widget.AppCompatImageView
Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(),drawableId);
Image.setImageBitmap(bitmap);
鸿蒙:
ohos.agp.components.Image
根据实际情况可传递其他参数
ImageSource imageSource = ImageSource.create(file, new ImageSource.SourceOptions());
pixelMap = imageSource.createPixelmap(new ImageSource.DecodingOptions());
image.setPixelMap(pixelMap);
5.图片填充整个控件
安卓:
image.setScaleType(ScaleType.FXY);
鸿蒙:
image.setScaleMode(Image.ScaleMode.STRETCH);
6.通过资源id获取位图
安卓:
getBitmapFromDrawable
鸿蒙:
/**
* 通过资源ID获取位图对象
**/
private PixelMap getPixelMap(int resId) {
InputStream drawableInputStream = null;
try {
drawableInputStream = getResourceManager().getResource(resId);
ImageSource.SourceOptions sourceOptions = new ImageSource.SourceOptions();
sourceOptions.formatHint = ""image/png"";
ImageSource imageSource = ImageSource.create(drawableInputStream, null);
ImageSource.DecodingOptions decodingOptions = new ImageSource.DecodingOptions();
decodingOptions.desiredSize = new Size(0, 0);
decodingOptions.desiredRegion = new Rect(0, 0, 0, 0);
decodingOptions.desiredPixelFormat = PixelFormat.ARGB_8888;
PixelMap pixelMap = imageSource.createPixelmap(decodingOptions);
return pixelMap;
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (drawableInputStream != null) {
drawableInputStream.close();
}
} catch (Exception e) {
e.printStackTrace();
}
}
return null;
}
7.获取Gif图片帧
安卓:
需要自定frame类,通过decoder获取
鸿蒙:
ImageSource.createPixelmap(int index, ImageSource.DecodingOptions opts)
8.BMP位图裁剪
安卓:
Bitmap.createBitmap(Bitmap source, int x, int y, int width, int height)
鸿蒙:
PixelMap.create(PixelMap source, Rect srcRegion, PixelMap.InitializationOptions opts)
※视频播放
在视频播放窗口上层增加控件
安卓:
略
鸿蒙:
A.pinToTop设置false,保证其他控件与surfaceProvider在同一layout下,并且不能设置背景
B.增加以下代码设置顶部窗口透明
WindowManager.getInstance().getTopWindow().get().setTransparent(true);
※数据库
数据库获取索引
安卓:
android.database.Cursor;
cursor.getString()/cursor.getColumnIndex()
鸿蒙:
ohos.data.resultset
※数据结构
1.应用程序数据共享
安卓:
context.getContentResolver();
resolver.getType(uri)
鸿蒙:
ohos.aafwk.ability.DataAbilityHelper
2.JSON解析
安卓:
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import org.json.JSONTokener;
鸿蒙:
Gson,fastJson
3.对象序列化
安卓:
android.os.Parcel;
parcel.readParcelable();parcel.writeParcelable()
鸿蒙:
ohos.utils.parcel
4.浮点数矩形,获取中心点
安卓:
RectF.centerX()
鸿蒙:
RectFloat.getCenter().getPointX()
5.数据结构类
安卓:
LongSparseArray
SparseArrayCompat
鸿蒙:
使用HashMap<Long,Ojbect>和HashMap<Integer,Ojbect>替代
备注:
内存使用和查找性能会有影响。"
6.浮点数矩形
安卓:
RectF
鸿蒙:
RectFloat
7.浮点坐标
安卓:
PointF
鸿蒙:
可使用Point
※对话框
1.对话框销毁
安卓:
mDialog.dismiss()
鸿蒙:
mDialog.destroy();
2.对话框中加载布局
安卓:
mDialog.setContentView(ViewGroup dialogView)
鸿蒙:
setContentCustomComponent(Componnet comp)
3.点击对话框外部关闭对话框
安卓:
mDialog.setCancelable(mPickerOptions.cancelable)
鸿蒙:
mDialog.setDialogListener(new BaseDialog.DialogListener() {
@Override
public boolean isTouchOutside() {
mDialog.destroy(); dialogView.getComponentParent().removeComponent(dialogView);
return true;
}
});
备注:
鸿蒙对话框销毁之后需移除对话框中加载的布局,否则再次加载会报错"
※动画
1.旋转动画
安卓:
android.view.animation.RotateAnimation
鸿蒙:
ohos.agp.animation.AnimatorProperty
2.值动画及相关回调
安卓:
android.animation.ValueAnimator
ValueAnimator.AnimatorUpdateListener
Animator.AnimatorListener
Animator.AnimatorPauseListener
鸿蒙:
ohos.agp.animation.AnimatorValue
AnimatorValue.ValueUpdateListener
Animator.StateChangedListener
Animator.LoopedListener
备注:
启动动画时,AnimatorValue必须作为类的成员变量,而不能时函数局部变量,否则动画不会启动"
3.线性插值器
安卓:
LinearInterpolator
鸿蒙:
自己写一个
public interface Interpolator {
float getInterpolation(float input);
}
public class LinearInterpolator implements Interpolator {
public LinearInterpolator() {
}
public LinearInterpolator(Context context, AttrSet attrs) {
}
public float getInterpolation(float input) {
return input;
}
}
4.设置动画循环次数
安卓:
animation.setRepeatCount(Animation.INFINITE)
鸿蒙:
animator.setLoopedCount(Animator.INFINITE)
※存储
获取存储根路径
安卓:
Environment.getExternalStorageDirectory().getAbsolutePath();
鸿蒙:
System.getProperty("user.dir")
※Canvas绘图
1.绘制圆弧
安卓:
Android canvas.drawArc()
鸿蒙:
ohos.agp.render; class Arc
2.绘制圆形的两种方式
安卓:
canvas.drawCircle(float x, float y, float radius, Paint paint)
鸿蒙:
A.canvas.drawPixelMapHolderRoundRectShape(PixelMapHolder holder, RectFloat rectSrc, RectFloat rectDst, float radiusX, float radiusY)
B.canvas.drawCircle(float x, float y, float radius, Paint paint)
3.绘制文本的方法
安卓:
drawText (String text, float x, float y, Paint paint)
鸿蒙:
drawText(Paint paint, String text, float x, float y)
4.获取text 的宽度
安卓:
text.getWidth();
鸿蒙:
Paint paint = new Paint();
paint.setTextSize(text.getTextSize());
float childWidth = paint.measureText(text.getText());
欢迎交流:HWIS-HOS@isoftstone.com
先🐎
Android开发工程师转鸿蒙的福音!这个总结实在是太棒了!👍👍👍
附件good.
这附件,厉害了!
关于数据结构,鸿蒙同样提供了相关类。
PlainArray
LightweightMap
LightweightSet
非常好的分享
非常好的分享
感谢分享!非常有用,已经收藏
欢迎欢迎
谢谢张老师
要是早点看到这个贴子我就不用手写viewPager和Fragment了
谢谢交流,欢迎常来
欢迎常来交流学习
欢迎欢迎
感谢,欢迎常来
感谢,欢迎常来
谢谢夏老师
感谢,欢迎常来
大佬,没看到鸿蒙java端titlebar上实现menu菜单弹窗的实现方式,类似
大佬 手势识别器这个GestureDetector有替代的吗 还有View.post(Runnable方法)