一步一步将你的C#程序部署到Android.......

hyblusea 2013-07-30 10:07:26
加精
C#是一种优秀的编程语言,语法之优雅,代码之简洁使得众多软粉多年来对她不离不弃。 但是如何将C#程序部署到Linux, Android等平台,这当然是得依靠众所周知的Mono。

本文Demo程序比较简单,实现了通过HttpRequest 查询天气,最终效果如下:



1. 下载并安装 Xamarin
http://xamarin.com/download
注册Xamarin账号, Role 选择 Academic(学者)即可;
运行 XamarinInstaller.exe在线安装程序,需要一个良好的网络环境,所需的安装程序体积大约 1.44G ,安装过程一路默认即可。

2.破解
网上找的破解:http://download.csdn.net/detail/flydoos/5820325
解压缩后,将文件覆盖到指定位置即可。

3.开发
开发Mono Android程序可以使用Xamarin Studio,也可以使用Visual Studio,建议使用VS,因为Xamarin对VS提供有强大的插件 再配合VS本身强大的功能,会使开发工作如鱼得水,另外Xamarin Studio目前还不够完善,比如添加引用 之后,需要重启。

a.创建 Android项目 ,如下图:


b.项目文件结构,如下图:


c.页面文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<LinearLayout
android:orientation="horizontal"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:id="@+id/linearLayout1"
android:layout_marginBottom="5dip"
android:layout_marginLeft="5dip"
android:layout_marginRight="5dip"
android:layout_marginTop="5dip">
<TextView
android:id="@+id/LblCity"
android:text="@string/PressCity"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:layout_weight="1" />
<EditText
android:id="@+id/TxtWeather"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:layout_weight="1"
android:lines="1" />
</LinearLayout>

我们可以看出,所有的控件均放在LinearLayout中,这是一个部局控件,LinearLayout又分为水平布局和垂直布局,比如一行中需要放置多个控件,这时候就需要用到水平布局。
d. cs文件
我们所熟悉的C#,将在这里大展拳脚,一切看上去都是那么亲切。

using System;

using Android.App;
using Android.Content;
using Android.Runtime;
using Android.Views;
using Android.Widget;
using Android.OS;
using System.IO;


namespace AndroidHelloWorld
{
[Activity(Label = "EasyWeather", MainLauncher = true, Icon = "@drawable/icon")]
public class Activity1 : Activity
{
protected override 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
Button button = FindViewById<Button>(Resource.Id.MyButton);
EditText txtWeather = FindViewById<EditText>(Resource.Id.TxtWeather);

// 天气查询结果
TextView lblCity = FindViewById<TextView>(Resource.Id.LblCityRst); // 城市
TextView lblCurTmp = FindViewById<TextView>(Resource.Id.LabCurTempRst); // 当前温度
TextView lblWeather = FindViewById<TextView>(Resource.Id.LabWeatherRst); // 天气
TextView lblRange = FindViewById<TextView>(Resource.Id.LabRangeRst); // 范围
TextView lblUptTime = FindViewById<TextView>(Resource.Id.LabUptTimeRst); // 更新时间


button.Click += (sender, e) => {
HttpHelper helper = new HttpHelper();

string sUrl = String.Format(@"http://cgi.appx.qq.com/cgi/qqweb/weather/wth/weather.do?
retype=1&city={0}&t={1}",
Uri.EscapeDataString(txtWeather.Text),
DateTime.Now.ToFileTime().ToString());

try
{
var v = helper.HttpGetRequest(sUrl, null, 10000, null);
var rst = new StreamReader(v.GetResponseStream(), System.Text.Encoding.GetEncoding(v.CharacterSet));


var vWeather = Newtonsoft.Json.JsonConvert.DeserializeObject<EtWeather>(rst.ReadToEnd());
//var vWeather = jss.Deserialize<EtWeather>(rst.ReadToEnd());

lblCity.Text = vWeather.city;
lblCurTmp.Text = vWeather.now_temperature;
lblWeather.Text = vWeather.now_pic_name;
lblRange.Text = vWeather.temperature_range;
lblUptTime.Text = vWeather.update_time;
}
catch (Exception Err)
{
var builder = new AlertDialog.Builder(this);
builder.SetMessage(Err.Message);
builder.SetCancelable(false);
builder.SetPositiveButton("OK", delegate { });
var dialog = builder.Create();
dialog.Show();

}

};
}
}
}

需要注意的是,我们怎样使用前台页面所定义的控件?目前我知道的这种方法,比较繁琐,不知道是否还有更直接一点的方式:
TextView lblCity = FindViewById<TextView>(Resource.Id.LblCityRst);


然后编写HttpHelper类:

public class HttpHelper
{
private readonly string DefaultUserAgent = "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.2; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)";

public HttpWebResponse HttpGetRequest(string url, string referer, int? timeout, CookieCollection cookies)
{
HttpWebRequest request = null;

if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase))
{
ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateValidationCallback(CheckValidationResult);
request = WebRequest.Create(url) as HttpWebRequest;
request.ProtocolVersion = HttpVersion.Version10;
}
else
request = WebRequest.Create(url) as HttpWebRequest;

request.Method = "GET";
request.ContentType = "application/x-www-form-urlencoded";
request.UserAgent = DefaultUserAgent;
request.CookieContainer = new System.Net.CookieContainer();

if (!string.IsNullOrEmpty(referer))
request.Referer = referer;

if (timeout.HasValue)
request.Timeout = timeout.Value;
else
request.Timeout = 25000;

if (cookies != null)
{
request.CookieContainer = new CookieContainer();
request.CookieContainer.Add(cookies);
}

return request.GetResponse() as HttpWebResponse;
}

private bool CheckValidationResult(object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors)
{
return true;
}

接下来就是如何将Web Api返回的Json还原为对象,这里需要用到 Newtonsoft.Json,添加引用的方法如下: 1,右击解决方案中的Components文件夹,选择view datails, 2.找到 jons.Net ,3.点击 Add to Projct(需要账号验证,使用注册的学者账号即可)。

d. 调试及部署
按F5运行, 第一次需要创建一个Android模拟器,这里只需要稍微注意一下你所选择Android系统版本,推荐使用2.2, 因为Mono For Android项目默认情况下使用的是2.2 的API。


Mono框架没有JIT(个人对这个没有研究,可能表达得不准确,欢迎拍砖),因此在断点调试的时候你会发现与平时调试C#略有不同,无法拖动断点,无法修改代码。
关于部署,必须使用Release进行编译,将并生成的APK文件传到手机安装后即可运行,UI与速度与Java开发的应用无差。

Mono for Android, Android开发我是新手中的新手,向大家学习,让C# running anywhere.
...全文
63646 321 9 打赏 收藏 举报
写回复
321 条回复
切换为时间正序
当前发帖距今超过3年,不再开放新的回复
发表回复
恠娃-_- 2016-02-23
感谢分享,楼主辛苦
  • 打赏
  • 举报
回复
u010195703 2016-02-22
  • 打赏
  • 举报
回复
楼主有几个问题需要请教一下:
1、Vs2010不支持这个的调试吧,我使用的是真机,不是模拟器;
2、怎么老提示我未连接网络,wifi和4G手机网络我都试了,所以一直都查不出天气来,2G网络也不得行,是不是手机的问题,未连接4G如下:
未连接wifi如下:

但是我的网络正常的啊,请教谢谢。
  • 打赏
  • 举报
回复
sinat_21289405 2014-09-26
网速真心好慢
  • 打赏
  • 举报
回复
alanchan_ch 2014-08-14
jifen 积分积分啊 啊啊啊 啊
  • 打赏
  • 举报
回复
alanchan_ch 2014-08-14
今天下载下来试试。没用过
  • 打赏
  • 举报
回复
kingredapple 2014-05-27
Mark先标记一下,有时间试试
  • 打赏
  • 举报
回复
zapdos 2014-05-27
引用 229 楼 hyblusea 的回复:
[quote=引用 228 楼 Tom_and_Jerry_zhao 的回复:] 说Java山寨C#?还不是不要带太强烈的个人主义!这样只会显得自己的水平很次。连起码的语言发展史都没了解。另外建议楼猪要是真的C#牛B,就勇敢的做出牛B的东西出来晒晒。效率要靠数据说话。否则好意西出来大谈语言优劣逞口舌之快。 PS:C++路过
玩C++的, 好了不起, 但是你可知计算机专业, C或C++是每个大学生必修课,每个人都是从C++走过来的。那么为什么有了C, C++还为什么会继续出现 Java, C#?自己Google其中的必然性。 很多人喜欢拿性能说事,也可以自己Google一下, C++ 与C#在各个方面的性能比较,凡事没有绝对。 另外我确实不才,没什么拿得出手的东西可以炫耀的, 只是分享一些有趣的,玩好的东西。 我就觉得Java已经落后C#了, 事实就是如此啊, 哪天 Java在语法上超越C#了, 我相信C#也会去学习Java,就是这样一个循环的过程。有不同观点可以绕道。[/quote] 说来说去不就是个语法合你口味而已,JAVA在各种领域上的表现足以让.NET望尘莫及 不单是移动系统,分布式计算方面JAVA也已甩开.NET好几条街了 语法这种东西要抄还不简单,但是在实际应用方面.NET永远也赶不上
引用 141 楼 hyblusea 的回复:
[quote=引用 140 楼 jiuchongtiann 的回复:] 很不错,没想到C#还可以这样。其实想起来原理也很简单,编写C#程序后,在发布的时候根据不同的目标平台生成不同的代码即可,没想到真的做到了。
微软设计 C#的初衷就是如此...[/quote] 这不也是抄JAVA的吗
  • 打赏
  • 举报
回复
叶黑 2014-05-27
  • 打赏
  • 举报
回复
lj_a_b 2014-05-12
有时间学学。。。
  • 打赏
  • 举报
回复
修仙之人 2014-05-09
开发模式完全跟andriod的一模一样嘛,只是用了C#的语法,但是这个过程中,并不是C#代码直接编译后andriod就能运行,其中是把C#代码编译成和JAVA一样的代码然后在打包成APK文件,其中C#转android就是MONO干的事.
  • 打赏
  • 举报
回复
灵影葬 2014-05-03
怎么使用release编译,然后部署到手机啊?
  • 打赏
  • 举报
回复
caoqinghua 2014-04-29
感谢分享,希望微软收了它
  • 打赏
  • 举报
回复
多啦A孟 2014-04-29
学习,刘明,谢谢
  • 打赏
  • 举报
回复
XXSingle 2014-04-29
先收藏了啊,以后用
  • 打赏
  • 举报
回复
紫色蚂蚁 2014-04-29
好东西 ,值得收藏和顶起来。
  • 打赏
  • 举报
回复
lvsezhufu 2014-04-29
学习~~~~~~~~~~~~~~~~~~~
  • 打赏
  • 举报
回复
何小宝 2014-04-28
引用 楼主 hyblusea 的回复:
C#是一种优秀的编程语言,语法之优雅,代码之简洁使得众多软粉多年来对她不离不弃。 但是如何将C#程序部署到Linux, Android等平台,这当然是得依靠众所周知的Mono。 本文Demo程序比较简单,实现了通过HttpRequest 查询天气,最终效果如下: 1. 下载并安装 Xamarin http://xamarin.com/download 注册Xamarin账号, Role 选择 Academic(学者)即可; 运行 XamarinInstaller.exe在线安装程序,需要一个良好的网络环境,所需的安装程序体积大约 1.44G ,安装过程一路默认即可。 2.破解 网上找的破解:http://download.csdn.net/detail/flydoos/5820325 解压缩后,将文件覆盖到指定位置即可。 3.开发 开发Mono Android程序可以使用Xamarin Studio,也可以使用Visual Studio,建议使用VS,因为Xamarin对VS提供有强大的插件 再配合VS本身强大的功能,会使开发工作如鱼得水,另外Xamarin Studio目前还不够完善,比如添加引用 之后,需要重启。 a.创建 Android项目 ,如下图: b.项目文件结构,如下图: c.页面文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">
    <LinearLayout
        android:orientation="horizontal"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:id="@+id/linearLayout1"
        android:layout_marginBottom="5dip"
        android:layout_marginLeft="5dip"
        android:layout_marginRight="5dip"
        android:layout_marginTop="5dip">
        <TextView
            android:id="@+id/LblCity"
            android:text="@string/PressCity"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:layout_weight="1" />
        <EditText
            android:id="@+id/TxtWeather"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:lines="1" />
    </LinearLayout>
我们可以看出,所有的控件均放在LinearLayout中,这是一个部局控件,LinearLayout又分为水平布局和垂直布局,比如一行中需要放置多个控件,这时候就需要用到水平布局。 d. cs文件 我们所熟悉的C#,将在这里大展拳脚,一切看上去都是那么亲切。

using System;

using Android.App;
using Android.Content;
using Android.Runtime;
using Android.Views;
using Android.Widget;
using Android.OS;
using System.IO;


namespace AndroidHelloWorld
{
	[Activity(Label = "EasyWeather", MainLauncher = true, Icon = "@drawable/icon")]
	public class Activity1 : Activity
	{
		protected override 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
			Button button = FindViewById<Button>(Resource.Id.MyButton);
			EditText txtWeather = FindViewById<EditText>(Resource.Id.TxtWeather);
			
            // 天气查询结果
            TextView lblCity = FindViewById<TextView>(Resource.Id.LblCityRst);                  // 城市 
            TextView lblCurTmp = FindViewById<TextView>(Resource.Id.LabCurTempRst);             // 当前温度  
            TextView lblWeather = FindViewById<TextView>(Resource.Id.LabWeatherRst);            // 天气
            TextView lblRange = FindViewById<TextView>(Resource.Id.LabRangeRst);                // 范围
            TextView lblUptTime = FindViewById<TextView>(Resource.Id.LabUptTimeRst);            // 更新时间


			button.Click += (sender, e) => {
				HttpHelper helper = new HttpHelper();

				string sUrl = String.Format(@"http://cgi.appx.qq.com/cgi/qqweb/weather/wth/weather.do?
											retype=1&city={0}&t={1}", 
											Uri.EscapeDataString(txtWeather.Text), 
											DateTime.Now.ToFileTime().ToString());

                try
                {
                    var v = helper.HttpGetRequest(sUrl, null, 10000, null);
                    var rst = new StreamReader(v.GetResponseStream(), System.Text.Encoding.GetEncoding(v.CharacterSet));


                    var vWeather = Newtonsoft.Json.JsonConvert.DeserializeObject<EtWeather>(rst.ReadToEnd());
                    //var vWeather = jss.Deserialize<EtWeather>(rst.ReadToEnd());

                    lblCity.Text = vWeather.city;
                    lblCurTmp.Text = vWeather.now_temperature;
                    lblWeather.Text = vWeather.now_pic_name;
                    lblRange.Text = vWeather.temperature_range;
                    lblUptTime.Text = vWeather.update_time;
                }
                catch (Exception Err)
                {
                    var builder = new AlertDialog.Builder(this);
                    builder.SetMessage(Err.Message);
                    builder.SetCancelable(false);
                    builder.SetPositiveButton("OK", delegate { });
                    var dialog = builder.Create();
                    dialog.Show();

                }
				
			};
		}
	}
}
需要注意的是,我们怎样使用前台页面所定义的控件?目前我知道的这种方法,比较繁琐,不知道是否还有更直接一点的方式:
TextView lblCity = FindViewById<TextView>(Resource.Id.LblCityRst);
然后编写HttpHelper类:

public class HttpHelper
{
private readonly string DefaultUserAgent = "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.2; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)";

public HttpWebResponse HttpGetRequest(string url, string referer, int? timeout, CookieCollection cookies)
{
	HttpWebRequest request = null;  

	if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase))
	{
		ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateValidationCallback(CheckValidationResult);
		request = WebRequest.Create(url) as HttpWebRequest;
		request.ProtocolVersion = HttpVersion.Version10;
	}
	else
		request = WebRequest.Create(url) as HttpWebRequest;

	request.Method = "GET";
	request.ContentType = "application/x-www-form-urlencoded";
	request.UserAgent = DefaultUserAgent;
	request.CookieContainer = new System.Net.CookieContainer();

	if (!string.IsNullOrEmpty(referer))
		request.Referer = referer;

	if (timeout.HasValue)
		request.Timeout = timeout.Value;
	else
		request.Timeout = 25000;

	if (cookies != null)
	{
		request.CookieContainer = new CookieContainer();
		request.CookieContainer.Add(cookies);
	}

	return request.GetResponse() as HttpWebResponse;  
}

private  bool CheckValidationResult(object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors)
{
	return true;
}
接下来就是如何将Web Api返回的Json还原为对象,这里需要用到 Newtonsoft.Json,添加引用的方法如下: 1,右击解决方案中的Components文件夹,选择view datails, 2.找到 jons.Net ,3.点击 Add to Projct(需要账号验证,使用注册的学者账号即可)。 d. 调试及部署 按F5运行, 第一次需要创建一个Android模拟器,这里只需要稍微注意一下你所选择Android系统版本,推荐使用2.2, 因为Mono For Android项目默认情况下使用的是2.2 的API。 Mono框架没有JIT(个人对这个没有研究,可能表达得不准确,欢迎拍砖),因此在断点调试的时候你会发现与平时调试C#略有不同,无法拖动断点,无法修改代码。 关于部署,必须使用Release进行编译,将并生成的APK文件传到手机安装后即可运行,UI与速度与Java开发的应用无差。 Mono for Android, Android开发我是新手中的新手,向大家学习,让C# running anywhere.
这个东西最后生存的也是apk文件?还是?
  • 打赏
  • 举报
回复
arSavior 2014-04-28
过来学习下,楼主辛苦了
  • 打赏
  • 举报
回复
苦逼de小Coder 2014-04-28
mark 收藏一下
  • 打赏
  • 举报
回复
加载更多回复
相关推荐
发帖
C#
加入

10.6w+

社区成员

.NET技术 C#
申请成为版主
帖子事件
创建了帖子
2013-07-30 10:07
社区公告

让您成为最强悍的C#开发者