前端之H5与App交互总结

交互方式

前端通过将自身的方法挂载到window对象上,App端可以找到并异步回调,通过方法参数的形式将数据传到前端,挂载到window上的方法名字需要两端协议约定。

JS:
	let token = '';
	function foo(token_: string) {
		token = token_;
	}
	//当App端调用getToken的时候触发绑定的foo方法
	window.getToken = foo;
App:
	// 伪代码
	window.getToken('123abc');

封装—让页面更简洁、易维护

封装的目的:1、App有两种系统(IOS、Android),如果直接写在页面上,会使页面有很多冗余的判断。

	if ( /(Android)/i.test(navigator.userAgent)) {
		//Android
		//...
	} else if (/(iPhone|iPad|iPod|iOS)/i.test(navigator.userAgent)) {
		//IOS
		//...
	}

所以我们单独将IOS和Android封装成两个独立的类,再通过工厂模式判断当前的系统类型,自动生产对应的类(以下代码统一用TS做演示)

TS:
	class AppFactory {
		static getUserInstance() {
			//这里考虑到全局使用的情况
			//所以采用单例的模式
			if (this.isIos()) {
				return IosUser.getInstance();
			} else if(this.isAndroid()) {
				return AndroidUser.getInstance();
			} else {
				return PcUser.getInstance();
			}
		}
		
		private static isIos() {
			return /(iPhone|iPad|iPod|iOS)/i.test(navigator.userAgent)
		}

		private static isAndroid() {
			return /(Android)/i.test(navigator.userAgent)
		}
	}
	
	//现在,我们在页面上再也不用写if判断了
	//只需要这样写就可以拿到当前的系统对应的类了
	AppFactory.getUserInstance()

下面我们继续完善IOS类和Android类,我们需要遵循面向抽象类编程的思维,而不是具体的某一类,所以它们还需要一个公共的父类User。

TS:
	//抽象父类
	abstract class User {
		protected some:string|null = null;
		
		//通过init将协议约定的方法挂载到window上
		//这里this指针会发生隐式绑定到window,需要使用bind强制绑定到自身
		init() {
			window.setSome = this.setSome.bind(this);
		}
	
		private setSome(some_: string) {
			this.some = some_;
		}
		
		public getSome_() {
			return this.some;
		}

		abstract applySome():void;	//这是由H5主动调用App端的方法,由各自子类去实现
	}

	//Android类
	class AndroidUser extends User{
		//上面提到两个类会使用单例模式,以下不在赘述
		private static androidUser: AndroidUser|null;
	    private constructor() {
	        super();
	    }

	    static getInstance() {
	        if (!this.androidUser) {
	            this.androidUser = new AndroidUser();
	            return this.androidUser;
	        }
	
	        return this.androidUser;
	    }

		public applySome() {
			//someFunc两端协议约定的方法名
			 window.someFunc.applySome()
		}
    }
	
	//Ios类
	class IosUser extends User {
		private static iosUser: IosUser|null;
	    private constructor() {
	        super();
	    }

	    static getInstance() {
	        if (!this.iosUser) {
	            this.iosUser = new IosUser();
	            return this.iosUser;
	        }
	
	        return this.iosUser;
	    }

		public applySome() {
			//对比AndroidUser类的applySome方法
			//可以看出调用App端方法时,两个系统的处理方式有差别
			window.webkit.messageHandlers.applySome.postMessage(null)
		}
	}

	//底层封装完毕后,在页面调用就会非常清晰、优雅
	//例如我们需要拿some这个字段
	AppFctory.getUserInstance.init();	//这一段代码在全局只需要初始化一次
	let some = AppFctory.getUserInstance.getSome();

如何完美抓住异步调用的时机—发布订阅模式

上面的代码,已经可以支持我们优雅的在页面与App端进行交互,但是还存在一个严重的Bug,App端是异步调用我们的方法,按照上面的写法,App端还没调用我们的方法就进行赋值操作,这样是肯定拿不到值的。
如何解决呢?这里我使用的是发布订阅模式,当监听到App端调用我们的方法后,通知页面

TS:
	//首先我们需要一个调度中心的类 NotificationCenter
	class NotificationCenter {
		public eventId: symbol;
		constuctor() {
			this.eventId = Symbol('eventId');
		}
		
		public register<T extends Event>(observe: object|symbol, event: Class<T>, cb: ()=>void) {
			let map = (<any>event).prototype[this.eventId]
					= (<any>event).prototype[this.eventId] || new Map<object|symbol, ()=>void>();

			map.set(observe, cb);
		}

		public notify(event: Event) {
			let map: Map<object|symbol, () => void> 
				= (<any>event)[this.eventId] || new Map();

			for (let [key, value] of map) {
				value();
			}
		}

		public unRegister(evemt: Event, observe: object|symbol) {
			let map =  Map<object|symbol, () => void> 
				= (<any>event)[this.eventId] || new Map();

			if(!map.has(observe)) {
				return;
			}

			map.delect(observe);
		}
	}

	interface Class<T> {
		prototype: T;
	}

这里实现方式大家可以不用太在意细节,主要的思想还是运用发布订阅模式的注册和监听,将这个类挂到我们的User类上,在页面注册监听,当方法被App端调用后,在最后发起通知,这样就可以监听到方法被成功调用了。

TS:
	class User {
		public nc_: NotificationCenter = new NotificationCenter();
	
		public setSome() {
			//...
			this.nc_.notify()	//通知页面方法已经调用完毕
		}
	}

	//页面
	AppFactory.getUserInstance.nc_.register(..., () => {
		//监听到变化后执行
		let some = AppFactory.getUserInstance.getSome_();
	})

拓展和总结

以上基本的架构就成型了,可能交互的时候会通过JSON的形式,那么就需要定义一个JSON类来处理格式等等,可以很好的拓展和集成。其实两端交互本质上很简单,但如果交互非常频繁就会导致页面冗余判断过多,难于维护,以上是我自己的一点经验和总结,也算是抛砖引玉,如有不足的地方,希望大家多多指正、评价。

  • 11
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
随着互联网的发展,移动互联网的普及,手机APP已成为人们使用最多的移动终端产品之一。随着越来越多的APP应用的涌现,越来越多的APP需要内嵌H5页面WebView进行网页的展示和交互。WebView是一种可在应用程序中嵌入Web页面的控件,可以用来显示来自互联网上的Web页面。它可以实现在应用中展示网页或在线功能,解决一些本地应用无法实现的功能。下面,我们分别从用户、开发者两个方面来探讨APP内嵌H5页面WebView的优缺点。 一、用户方面: 优点: 1.节省时间:在APP中直接查看嵌入的网页,省去了用户手动打开浏览器输入网址的步骤; 2.良好的用户体验:页面加载速度相对较快,而且对主应用对内存占用少,不影响其他应用的使用; 3.方便分享:在WebView中打开的网页可以长按复制网址链接,方便分享给其它用户; 4.强大的交互能力:在APP中嵌入H5页面,拓展了应用的交互能力。 缺点: 1.便捷性带来的安全隐患:一些App会在内嵌的H5页面中嵌入第三方广告,导致用户隐私泄露; 2.缺乏统一标准:因为浏览器的内核和引擎都是不同的,所以在不同的WebView中,同一网页的显示效果和交互体验可能会有差异。 二、开发者方面: 优点: 1.拓展应用功能:借助WebView内嵌H5页面,应用的功能可以得到极大的拓展; 2.代码复用:WebView可以实现HTML、CSS等内容的兼容,减轻了移动开发者负担; 3.节省开发成本:相对于开发单独的H5 APP,内嵌方式更为灵活,可以适用于不同场景和需求。 缺点: 1.不支持多线程:WebView在JS调用本地方法的时候是在同一线程下执行的,如果WebView的内容较为复杂,可能会导致主线程卡顿; 2.性能问题:在内存管理、布局排版等方面还不如原生应用; 3.浏览器兼容性:WebView的内核并没有得到很好的统一,不同的WebView之间有兼容性问题,会导致页面显示和交互问题; 4.安全问题:WebView加载HTML页面时容易受到跨站脚本(XSS)和恶意代码注入等安全漏洞的攻击。为此,开发者应该加强前端页面安全防范,对浏览器缓存和Cookie进行管理并开启CSP(内容安全策略)。 综合来看,在APP内嵌H5页面WebView上,开发者需要在开发时注意安全方面的问题,同时还需要加强对WebView性能和兼容性的了解和优化。用户便捷性和良好的用户体验,也需要开发者在开发时重视。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值