Android与iOS输入法开发框架比较谈

对于任何一个使用手机的人,有一样工具是不可能缺少的,它既不是微信之类的社交工具,也不是支付宝之类的金融工具(事实上这两个都越界了),而是输入法这样的输入工具。更重要的是,输入法还是一种特权工具,因为它能够与其它任何可以接受信息录入的应用进行配合,帮助用户完成信息输入,这也就意味着,输入法有更多的机会接触到用户的个人信息和隐私信息。那么,作为一名手机使用者,你对手机输入法背后的运作机制了解吗?你有没有怀疑过输入法会在你不知情的情况下窃取你的个人信息呢?如果你是一位对输入法开发感兴趣的人,你是否想知道Android平台和iOS平台的输入法开发框架有何异同?作为一名输入法的开发者,我将结合我在两个平台上的输入法开发经验,告诉你,我的所见所得与所践所得。

输入法开发入门

两个平台的输入法框架有一定的相似性,根本上都是通过继承某个指定的类来实现输入法开发。Android平台上是InputMethodService,而iOS则是UIInputViewController。

在Android平台上,除了继承InputMethodService类,还要在AndroidManifest.xml文件中加入下列表明

 <service
            android:name=".MyService"
            android:permission="android.permission.BIND_INPUT_METHOD">
            <intent-filter>
                <action android:name="android.view.InputMethod" />
            </intent-filter>

            <meta-data
                android:name="android.view.im"
                android:resource="@xml/method" />
        </service>

其中android:resource="@xml/method",意指在xml文件夹下有个method.xml文件,里面的内容应当是这样的:

<?xml version="1.0" encoding="utf-8"?>

<input-method xmlns:android="http://schemas.android.com/apk/res/android"
android:settingsActivity="com.android.inputmethod.pinyin.SettingsActivity"/>

其中SettingsActivity是输入法相对应的设置程序,以下称为主程序。

相比而言,iOS的入门还要更简单一些。这里有我此前写的一篇介绍iOS输入法开发入门的文章,虽然我使用的技术是Xamarin,但是流程是大同小异的:如何使用Xamarin开发iOS输入法

与输入框如何交互

在Android平台下,我们是这样向输入框发送文本的:

InputConnection ic = getCurrentInputConnection();
        if (null != ic)
            ic.commitText("hello world", 1);

而在iOS下,则是TextDocumentProxy.InsertText("hello world").

如果要向输入框发送一个删除命令,Android的做法是模仿删除键的按下操作:

//调用该方法模仿删除键按下:simulateKeyEventDownUp(KeyEvent.KEYCODE_DEL);
private void simulateKeyEventDownUp(int keyCode) {
        InputConnection ic = getCurrentInputConnection();
        if (null == ic)
            return;

        ic.sendKeyEvent(new KeyEvent(KeyEvent.ACTION_DOWN, keyCode));
        ic.sendKeyEvent(new KeyEvent(KeyEvent.ACTION_UP, keyCode));
    }

而在iOS下,则是TextDocumentProxy.DeleteBackward();

从这个角度看,iOS的API设计要简洁许多,但事实上,Android平台所提供的接口要远比iOS丰富,能做许多iOS下做不了的事情,比如选中输入框的文本,和更丰富的生命周期。

Android输入法服务的生命周期

UI上的异同

在Android平台下,创建键盘的界面与平常开发Activity相似:

@Override
public View onCreateInputView() {
    MyKeyboardView inputView =
        (MyKeyboardView) getLayoutInflater().inflate(R.layout.input, null);

    inputView.setOnKeyboardActionListener(this);
    inputView.setKeyboard(mLatinKeyboard);

    return mInputView;
}

而iOS平台也与开发一般的iOS App类似:

    /*下述iOS平台的代码示例都是Xamarin下的C#实现,
      本质上与swift或Object-C并无不同*/
    public override void ViewDidLoad() {
            try {
                base.ViewDidLoad();
                var v = getKeyBoardView();
            
                View.AddSubview(v);
                v.MakeConstraints(make => {
                    make.Height.EqualTo(new NSNumber(height));
                    make.Width.EqualTo(new NSNumber(width));
                    make.Edges.EqualTo(View);
                });

            } catch (Exception e) {
                ErrorReporter.Report(e);
            }
            // Perform custom UI setup here

        }

但是呢,iOS下,输入法键盘的UI有一个限制,就是不用实现悬浮窗体,或者说,输入法的显示区间无法延展到host App上。而在Android平台上则没有这种限制;所以Android输入法的拼音栏可以悬浮在输入框的上面,而iOS输入法则不行。

其实,不允许越过键盘显示,我说忍也就忍了,我不能忍的是,为什么你他喵的系统键盘就可以这么做!!!

你说我该怎么办?当然是原谅她了,谁让她是苹果呢!

进程空间上的异同

一个输入法在结构上往往分成两个部分,一个是输入法键盘,一个是主程序。在Android平台的输入法中,他们分别对应一个Service和一个Activity,而且这两个对象是运行在同一个进程空间下的。而在iOS平台的输入法中,他们分别是一个App Extension和一个Containing App,他们是两个独立的程序,运行在不同的进程空间中。

运行在不同的进程空间意味着什么呢?假使某个类下定义了一个静态变量,在Android平台下Service和Activity访问到的都是同一个对象,而对于iOS平台则不然,他们读写的会是两个不同的对象,这也就意味着,iOS下输入法和主程序之间无法通过静态变量进行信息交流。

iOS输入法的基本结构图

文件空间上的异同

输入法与主程序之间不可避免的是要交换信息,甚至是读写文件,否则主程序也就起不到对输入法的设置作用。在Android平台下,这一点相对比较容易实现,因为输入法和主程序既然运行在相同的进程空间,那么他们也就必然共享同一个文件空间。

而iOS上则不同了,输入法和主程序之间不仅运行在不同的进程空间,他们还各拥有互相独立的文件空间,彼此都不能读取对方空间中的文件。注意,是都不能。

那么他们之间的文件交流共享如何实现呢?答案是App Groups。App Groups可以用于创造第三个文件空间,输入法键盘和主程序通过这个空间进行文件共享。他们的关系是这样的:

iOS平台下输入法与主程序的关系图

必须指出的是,主程序对App Groups中的文件拥有读写权限的,而输入法在“允许完全访问”没有打开之前,对App Groups只有读取权限,而没有写入权限。

这在客观上给开发者设置了许多困难,比如我要为岁寒输入法开发一个词条导入功能,首先我无法将文件直接共享到输入法键盘的文件空间下,而只能共享到主程序的文件空间下,再由主程序将文件复制到App Groups下,供输入法键盘进行访问。而由输入法键盘去将这个文件中的信息导入的其文件空间下的词库中是非常耗时的操作,会导致输入法卡死,因此我只能将导入的信息放在App Groups下的词库中,这存在一个小问题,就是在“允许完全访问”不开启的情况下,输入法键盘无法更新App Groups中的词频信息。

值得欣慰的是,这还只是众多困难中的一个。

权限管理上的异同

在Android平台上,输入法和主程序的权限声明是一体的,都放在AndroidManifest.xml文件下。

换言之,主程序可以访问网络,则输入法也可以访问网络;主程序可以读写SD卡,则输入法也可以读写SD卡;主程序可以访问通讯录,则输入法也可以访问通讯录;所以呢:

而在iOS平台上,主程序和输入法的权限相互独立,而且

主程序:嗨,兄弟,我能访问图片库了!
输入法:。。。(关我屁事)
主程序:嗨,兄弟,我能访问通讯录了!
输入法:。。。(关我屁事)
主程序:嗨,兄弟,我能访问网络了!
输入法:。。。(关我屁事)

据我所知,在“允许完全访问”关闭的情况下,输入法几乎无法向主程序传递任何信息,哪怕是一个字。只有一种情况例外,那就是输入法通过Open Url调用主程序时,可以通过设置参数传递一些信息。

值得一提的是,“允许完全访问”机制确实给开发工作带来了极大的不便。一开始,我发现输入法甚至无法向App Groups写入数据时,我是极为不解的。我当时想,哪怕将输入法的权限细分也好呀,网络的归网络,存储的归存储。但仔细一想,不得不承认,只有一个“允许完全访问”开关的设计是正确的。因为如果只是授予了存储,输入法就可以向App Groups写入数据,主程序获得数据之后就可以通过网络将数据发送出去。也就是说,只要有一处授权,其它的权限限制就会变得毫无意义,变成只防君子,不防小人的假把式。于是,开则全开,闭则全闭,没有中间状态。

还有一点必须说到,就是iOS输入法是无论如何也获取不了录音权限的。当然了,你会说,系统输入法可以呀,你都知道那是系统输入法啦~据说现在讯飞输入法可以在输入法界面录音,我要告诉你的是,执行录音操作的一定不是输入法,而是另外一种服务,只不过讯飞的工程师使了障眼法,让用户误以为输入法可以录音而已。

到底哪个平台更安全

最后一个问题,那么到底哪个平台的输入法更加安全呢?

你可能会觉得,答案显而易见,果然是iOS啦。但我要说的是,这也分情况。iOS输入法只有在关闭“允许完全访问”才比Androis输入法安全,一旦打开,其实两者没差。所以,你觉得两个大佬的提醒都只是在给你开玩笑吗?其实他们的潜台词是,你可想好了啊,按了确定之后,出了啥事,你就自己兜着了哈。

事情到了这个层次就已然不是技术可以解决的问题了。惟一可以解决这个问题的方法是信任。

今时今日,我们每个人都在信息的海洋里裸泳,你相信我一直闭着眼睛,不去看你在水里的样子吗?

我是岁寒,感谢你使用岁寒输入法,感谢你的信任。

本文中Android平台代码为java语言,iOS平台代码为C#语言。

更多的信息可以参见各自的官方文档:
iOS:Custom Keyboard
Android: Creating an Input Method

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351

推荐阅读更多精彩内容