Xamarin可以使用C#语言开发原生的Android和iOS应用,被微软收购后更是热度不减;而SignalR是微软推出的实时通信技术。之前在项目中用过SignalR,而Xamarin则是首次接触,本次是想结合二者开发一个Windows、Android、iOS平台间实时通信的示例。

开发环境

说明

操作系统

Windows 10 professional

编译器

VS 2015 update2

插件

Xamarin for VisualStudio

其他

Android SDK,Android Emulater



1.创建服务端

由于之前写过WPF的服务端,所以直接拿来用了。

1.1新建项目ServerApp

选择Windows>WPF应用程序,添加新项目ServerApp。这里框架依然沿用以前的.NET Framework 4.5,并未使用最新的4.6.1 。

Android signal 9 什么时候会触发 signal org android_Android

1.2引用SignalR包

选择“工具>NuGet包管理器>管理解决方案的NuGet程序包”

Android signal 9 什么时候会触发 signal org android_android_02

搜索“SignalR”,相应的包全部安装上(有些包服务端用不上,但是客户端后面会用上)。安装成功后会在项目引用中看到SignalR的几个引用已经添加进来了。

Android signal 9 什么时候会触发 signal org android_android_03

1.3添加启动入口和Hub

添加Startup.cs作为程序启动的入口。

using Owin;//必需的程序集引用

namespace ServerApp
{
    public class Startup
    {
        public void Configuration(IAppBuilder app)
        {
            app.MapSignalR();
        }
    }
}

添加MyHub.cs作为主要服务处理程序。MyHub继承于Hub并实现其接口,IMyHub是自己写的接口,目前只有一个方法“Task SendPlatform(string platform,string message)”。方法SendPlatform就是后面要用到的通信方法。

using Microsoft.AspNet.SignalR;
using System.Collections.Generic;
using System.Data;
using System.Threading.Tasks;
using System.Windows;
using Model;

namespace ServerApp
{
    public class MyHub:Hub,IMyHub
    {
        public override Task OnConnected()
        {
            return base.OnConnected();
        }
        public override Task OnDisconnected(bool stopCalled)
        {
            return base.OnDisconnected(stopCalled);
        }
        public override Task OnReconnected()
        {
            return base.OnReconnected();
        }

        public Task SendPlatform(string platform,string message)
        {
            return Clients.All.sendplatform(platform, message);
        }
    }
}

1.4添加启动服务方法

在主页面的后台MainWindow.cs添加控制服务启动及停止的方法。

namespace ServerApp
{
    public partial class MainWindow : Window
    {
        #region 通信设置
        public IDisposable SignalR { get; set; }
        static string ServerURI = ConfigurationManager.AppSettings["ServiceUri"]; //全局通信链接,地址为服务端所在主机IP地址,端口号可任意指定
        #endregion
        public MainWindow()
        {
            InitializeComponent();
            StartServer();
        }
        private void StartServer()
        {
            try
            {
                SignalR = WebApp.Start(ServerURI);
            }
            catch (TargetInvocationException)
            {
                WriteToConsole("已存在运行的服务:" + ServerURI);
                this.Dispatcher.Invoke(() => btn_Start.IsEnabled = true);
                return;
            }
            this.Dispatcher.Invoke(() => btn_Stop.IsEnabled = true);
            WriteToConsole("服务正在运行:" + ServerURI);
        }
    }
}

服务URI在配置文件app.config中设置。

<?xml version="1.0" encoding="utf-8"?>
<configuration>
  <appSettings>
    <!--通信连接地址-->
    <add key="ServiceUri" value="http://127.0.0.1:46900/" />
  </appSettings>
</configuration>

2创建共享项目ShareMe

选择“Windows>共享项目”,添加项目ShareMe,用于公共方法共享,Android、iOS、Windows客户端都用的到。

Android signal 9 什么时候会触发 signal org android_SignalR_04


添加类文件OSClient.cs

using System;
using Microsoft.AspNet.SignalR.Client;
using System.Threading.Tasks;

namespace ShareMe
{
    public class OSClient
    {
        private readonly string _platform;
        private readonly HubConnection _connection;
        private readonly IHubProxy _proxy;

        public event EventHandler<string> OnSendPlatform;

        public OSClient(string platform)
        {
            _platform = platform;
            _connection = new HubConnection("http://127.0.0.1:46900");//url要与服务端的ServerUri保持一致,否则通信不上
            _proxy = _connection.CreateHubProxy("MyHub");//创建Hub,与服务端MyHub类名一致
        }
        public async Task Connect()
        {
            await _connection.Start();
            _proxy.On("sendplatform",(string platform,string message)=> {
                if (OnSendPlatform!=null)
                {
                    OnSendPlatform(this, string.Format("{0}:{1}",platform,message));
                }
            });
            Send("Connected");
        }
        public Task Send(string message)
        {
            return _proxy.Invoke("SendPlatform", _platform, message;//调用方法名与服务端方法名一致
        }
    }
}

3创建安卓客户端AndroidApp

3.1新建项目

选择“Android>Blank App”新建项目AndroidApp。

Android signal 9 什么时候会触发 signal org android_System_05

3.2添加相关引用

首先添加共享项目ShareMe的引用

Android signal 9 什么时候会触发 signal org android_Android_06


然后添加SignalR程序包的引用。在添加SignalR程序包的过程中遇到了比较棘手的问题,使用NuGet为项目添加SignalR引用时总是报错。

Android signal 9 什么时候会触发 signal org android_xamarin_07


考虑是由于Mono还不支持的当前SignalR的版本,在网上查阅了相关资料后发现,SignalR是可以用于Mono的,但是需要使用portbale版本,可以在解决方案根目录下“packages\Microsoft.AspNet.SignalR.Client.2.2.0\lib\portable-net45+sl5+netcore45+wp8+wp81”找到,手动添加引用。

Android signal 9 什么时候会触发 signal org android_SignalR_08


添加引用后编译会出错,把错误提示中出现的未引用程序包通过NuGet添加下(有Newton.Json及System.Net.Http等)。其中可能会有System.Net.Http.Extensions也需要手动添加引用。所有引用添加完后再次编译提示

这当中还有一个小插曲,就是引用添加完之后,编译总是失败并提示:

"aapt.exe"已退出,代码为“-1073741819”

尝试了N多次,即使新建项目后什么都不做依然会报错,这就比较奇怪了。后来发现了一篇Xamarin作者之一的文章,说到是由于Android SDK Bulid-tools 22.0.1自身问题所致,移除这个版本的编译工具即可。但是我在Android SDK Manager中没有安装这个版本,所以更加疑惑,后来向别人请教说可能是Android SDK Bulid-tools 24有问题。于是把这个版本的也卸载了,问题解决。

Android signal 9 什么时候会触发 signal org android_System_09

3.3添加客户端方法

添加完成后的项目文件结构如下:

Android signal 9 什么时候会触发 signal org android_System_10


Resources是项目资源所在,其中drawable主要是图片,layout是布局,values是静态资源,这些文件夹都是项目创建时自动生成的。后台处理主要包含在MainActivity.cs中。

打开布局文件“Resources\layout\Main.xml”,在设计器中添加一个EditText和ListView(把原来的删掉,直接从工具箱里拖,然后再属性中修改id)

Android signal 9 什么时候会触发 signal org android_xamarin_11


Android signal 9 什么时候会触发 signal org android_xamarin_12


添加完成后重新生成一下,不然资源文件在后台找不到。修改完成后的布局文件Main.xml代码如下

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <EditText
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/Input" />
    <ListView
        android:minWidth="25px"
        android:minHeight="25px"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/Messages" />
</LinearLayout>

在后台文件MainActivity.cs中添加相应的方法:

using Android.App;
using Android.Widget;
using Android.OS;
using ShareMe;//共享项目引用
using Android.Views.InputMethods;
using System.Collections.Generic;

namespace AndroidApp
{
    [Activity(Label = "AndroidApp", MainLauncher = true, Icon = "@drawable/icon")]
    public class MainActivity : Activity
    {
        int count = 1;

        protected override async void OnCreate(Bundle bundle)
        {
            base.OnCreate(bundle);

            // Set our view from the "main" layout resource
            SetContentView(Resource.Layout.Main);

            // Get our button from the layout resource,
            // and attach an event to it
            var client = new OSClient("Android");

            var input = FindViewById<EditText>(Resource.Id.Input);
            var messages = FindViewById<ListView>(Resource.Id.Messages);
            var inputManager = (InputMethodManager)GetSystemService(InputMethodService);
            var adapter = new ArrayAdapter<string>(this, Android.Resource.Layout.SimpleListItem1, new List<string>());
            messages.Adapter = adapter;

            await client.Connect();
            input.EditorAction += delegate
            {
                inputManager.HideSoftInputFromWindow(input.WindowToken, HideSoftInputFlags.None);
                if (string.IsNullOrEmpty(input.Text))
                {
                    return;
                }
                client.Send(input.Text);
                input.Text = "";
            };
            client.OnSendPlatform+= (sender, message) => RunOnUiThread(() =>adapter.Add(message));
        }
    }
}

最后启动服务端,在模拟器上运行AndroidApp查看效果。


后记

写这个示例中间不少曲折,但也收获很多。写的过程中查找好多的文章都是几年前的,时效性可能有折扣,还是要多想多看多学多用。