《Android NFC开发实战详解》——6.4节Android NFC P2P开发进阶

简介:

本节书摘来自异步社区《Android NFC开发实战详解》一书中的第6章,第6.4节Android NFC P2P开发进阶,作者 赵波,更多章节内容可以访问云栖社区“异步社区”公众号查看

6.4 Android NFC P2P开发进阶
Android NFC开发实战详解
本节将介绍Android API 16+中引入的针对NFC P2P功能开发的新功能——文件传输进行介绍。该功能包括setBeamPushUrisCallback和setBeamPushUris两个方法。通过本节的介绍,大家可以结合NFC和蓝牙或WiFi很轻松的实现Android设备之间大数据(如图片、音乐等)的传输。

6.4.1 Beam实现文件传输的方法
Android4.1(Jelly Bean,Android API 16)新增了一种基于NFC和蓝牙(或WiFi)实现Android设备之间大数据传递的新方法,其中包含setBeamPushUrisCallback( )和setBeamPushUris( )两个方法。该方法比传统的蓝牙传输少了显示配对的过程,在你的应用无需考虑更多的其他工作同时还能发挥蓝牙传输的优势。

1.setBeamPushUrisCallback( )方法的原型
public void setBeamPushUrisCallback (NfcAdapter.CreateBeamUrisCallback callback, Activity activity)

其中,callback为通过Android Beam发送一个或多个动态生成的Uri(s)的回调接口;activity为当前调用该方法的activity,即push Uri(s)的Activity。

使用setBeamPushUrisCallback()方法时,应注意以下几点:

(1)该方法可以在Activity中任何位置调用(onDestroy()之前)。因为Uri只有Activity在前台状态(resume()状态)下才可用,所以Android的官方建议在OnCreate()中调用该方法。同时,由于该方法并不阻塞线程,因此可以在UI主线程中使用。

(2)使用该方法时,如果callback为null,则调用该方法的Activity的Uri Push功能将会disable。

(3)当同时使用该方法和setNdefPushMessage(NdefMessage, Activity, Activity...) 或setNdefPush MessageCallback(NfcAdapter.CreateNdefMessageCallback, Activity, Activity...)方法时,该方法具有较高优先级。

(4)关于该方法的使用,官方提供的使用范例如下(关于更详细的使用方法,读者可以参考本节后面的具体实例)。

protected void onCreate(Bundle savedInstanceState)
{
 super.onCreate(savedInstanceState);
 NfcAdapter nfcAdapter = NfcAdapter.getDefaultAdapter(this);
 if (nfcAdapter == null)
  return; // NFC not available on this device
 nfcAdapter.setBeamPushUrisCallback(callback, this);
}

(5)如果用户的手机不支持蓝牙或WiFi,那么该方法将失效,使用该方法时则不会有任何反应。

(6)使用该方法需要在AndroidManifest.xml中添加NFC权限。

(7)在Android API 16+以上的系统中使用。

2.setBeamPushUris ( )方法的原型
public void setBeamPushUris (Uri[] uris, Activity activity):其中,uris为用于Android Beam的一个或多个Uri(s);activity为当前调用该方法的activity,即push Uri(s)的Activity。

使用setBeamPushUris()方法时,应注意以下几点:

(1)提供的每一个Uri中的Uri Scheme必须包含“file”或“content”。

(2)该方法可以在Activity中任何位置调用(onDestroy()之前)。因为Uri只有Activity在前台状态(resume()状态)下才可用,所以Android的官方建议在OnCreate()中调用该方法。同时,由于该方法并不阻塞线程,所以可以在UI主线程中使用。

(3)使用该方法时,如果Uri为null,则调用该方法的Activity的Uri Push功能将会disable。

(4)当同时使用该方法和setNdefPushMessage(NdefMessage, Activity, Activity...)或setNdefPush MessageCallback(NfcAdapter.CreateNdefMessageCallback, Activity, Activity...)方法时,该方法具有较高优先级。

(5)关于该方法的使用,官方提供的使用范例如下(关于更详细的使用方法,读者可以参考本节后面的具体实例)。

protected void onCreate(Bundle savedInstanceState)
{
 super.onCreate(savedInstanceState);
 NfcAdapter nfcAdapter = NfcAdapter.getDefaultAdapter(this);
 if (nfcAdapter == null)
  return; // NFC not available on this device
 nfcAdapter.setBeamPushUris(new Uri[]
 { uri1, uri2 }, this);
 }
}

(6)如果用户的手机不支持蓝牙或WiFi,那么该方法将失效,使用该方法时则不会有任何反应。

(7)如果希望动态的提供Uri(s),那么可以使用上述的setBeamPushUrisCallback()方法。

(8)使用该方法需要在AndroidManifest.xml中添加NFC权限。

(9)在Android API 16+以上的系统中使用。

3.setBeamPushUrisCallback( ) 和setBeamPushUris( ) 的区别和选择
setBeamPushUrisCallback( )提供的是一个回调接口。当使用这个回调接口时,用户每Beam一次,该回调函数执行一次。如果待分享的Uri是动态的,会随着activity中的用户环境变化而变化,就可以使用该回调方法;反之,当待分享的Uri是静态的,是不会改变的,就可以使用setBeam PushUris( )方法提前定义好它们。总之,setBeamPushUris( )方法是在初始化Beam的同时立即提供Uri对象值,而setBeamPushUrisCallback( )回调方法在初始化Beam时并无需提供Uri对象值,而是在建立Beam连接时提供Uri对象值。

6.4.2 Beam文件传输实例1:setBeamPushUris
在Beam文件传输实例1中,本书对setBeamPushUris ( )方法实现Android Beam传输Uri功能进行了实例描述。由于Android Beam实现文件传输的方法中接收端可以由操作系统完成的,无需在App中处理,因此该实例中仅包括Uri发送端。Uri的Push部分主要包括两个步骤,分别为:

(1)在Activity中创建文件Uri。

(2)在需要的地方调用setBeamPushUris。

具体参见实例代码中对应的注释,详细代码如下:

1. package skyseraph.nfc_demo.p2p.beam.app;   //声明包

2. import skyseraph.android.util.LogUtil;  //导入相关类
3. ……//该处省略了导入相关类的代码
4. public class BeamFileDemo1 extends Activity implements OnClickListener
5. {
6.  private static final String Tag_ASSIST = "[BeamFileDemo1]-";
7.  private Context mContext = null;
8.  private boolean flagNFC = false;
9.  private boolean flagVersion = false;
10.  private static int mRequestCode = 1001;
11.  // NFC相关
12.  private NfcAdapter mNfcAdapter = null;
13. 
14.  @Override
15.  protected void onCreate(Bundle savedInstanceState)
16.  {
17.   // TODO Auto-generated method stub
18.   super.onCreate(savedInstanceState);
19.   mContext = this;
20.   LogUtil.i(MyConstant.Tag, Tag_ASSIST + "into onCreate");
21.   flagNFC = checkNFCFunction();
22.   LogUtil.i(MyConstant.Tag, Tag_ASSIST + "flagNFC=" + flagNFC);
23.   if (flagNFC)
24.   {
25.    checkSystemVersion();
26.    LogUtil.i(MyConstant.Tag, Tag_ASSIST + "flagVersion=" + flagVersion);
27.    if(flagVersion)
28.    {
29.     // BNM步骤1: Create 文件Uri
30.     Intent i = new Intent(Intent.ACTION_GET_CONTENT);   
     //用户选择某种特殊数据并返回
31.     i.setType("image/*");   
     //查看类型,如果是其他类型,如视频则替换成 video/*,或 */*
32.     startActivityForResult(i, mRequestCode);
33.    }
34.   }
35.  }
36. 
37. 
38.  /*
39.   * (non-Javadoc)
40.   * 
41.   * @see android.app.Activity#onActivityResult(int, int,
42.   * android.content.Intent)
43.   */
44.  @Override
45.  protected void onActivityResult(int requestCode, int resultCode, Intent data)
46.  {
47.   // TODO Auto-generated method stub
48.   // super.onActivityResult(requestCode, resultCode, data);
49.   if (requestCode == mRequestCode && resultCode == RESULT_OK) 
50.   {
51.    Uri[] uriArray = new Uri[]
52.    { data.getData() };
53.    LogUtil.i(MyConstant.Tag, Tag_ASSIST + "uriArray=" + uriArray);
54.    
55.    // BNM步骤2:call setBeamPushUris anywhere your want
56.    mNfcAdapter.setBeamPushUris(uriArray, this);
57. 
58.    Button btn = new Button(this);
59.    btn.setText("Touch this to finish!");
60.    btn.setTextSize(30);
61.    btn.setOnClickListener(this);
62.    setContentView(btn);
63.   }
64.  }
65. 
66.  @Override
67.  public void onClick(View v)
68.  {
69.   finish();
70.  }
71. 
72.  /**
73.   * NFC Function Check By skyseraph 2013-2
74.   */
75.  private boolean checkNFCFunction()
76.  {
77.   // TODO Auto-generated method stub
78.   mNfcAdapter = NfcAdapter.getDefaultAdapter(this);
79.   if (mNfcAdapter == null)
80.   {
81.    // mTextView.setText("NFC apdater is not available");
82.    Dialog dialog = null;
83.    CustomDialog.Builder customBuilder = new CustomDialog.Builder (mContext);
84.    customBuilder.setTitle(getString(R.string.inquire)).setMessage(getString(R.string.  
 nfc_notice2))
85.      .setIcon(R.drawable.dialog_icon2)
86.      .setNegativeButton(getString(R.string.no), new Dialog   
      Interface.OnClickListener()
87.      {
88.       public void onClick(DialogInterface dialog, int which)
89.       {
90.        dialog.dismiss();
91.        finish();
92.       }
93.      }).setPositiveButton(getString(R.string.yes), new Dialog   
      Interface.OnClickListener()
94.      {
95.       public void onClick(DialogInterface dialog, int which)
96.       {
97.        dialog.dismiss();
98.        finish();
99.       }
100.      });
101.    dialog = customBuilder.create();
102.    dialog.setCancelable(false);// back
103.    dialog.setCanceledOnTouchOutside(false);
104.    SetDialogWidth(dialog).show();
105.    return false;
106.   } else
107.   {
108.    if (!mNfcAdapter.isEnabled())
109.    {
110.     Dialog dialog = null;
111.     CustomDialog.Builder customBuilder = new CustomDialog.Builder   
      (mContext);
112.     customBuilder.setTitle(getString(R.string.inquire)).setMessage(getString(R.string.  
 nfc_notice3))
113.       .setIcon(R.drawable.dialog_icon2)
114.       .setNegativeButton(getString(R.string.no), new   
       DialogInterface.OnClickListener()
115.       {
116.         public void onClick(DialogInterface dialog, int which)  
117.        {
118.         dialog.dismiss();
119.         finish();
120.        }
121.       }).setPositiveButton(getString(R.string.yes), new   
       DialogInterface.OnClickListener()
122.       {  
123.         public void onClick(DialogInterface dialog, int which)
124.        {
125.         dialog.dismiss();
126.         Intent setnfc = new Intent(Settings.   
         ACTION_WIRELESS_SETTINGS);
127.         // Intent setnfc = new
128.         // Intent(Settings.ACTION_NFC_SETTINGS);
129.         startActivity(setnfc);
130.        }
131.       });
132.     dialog = customBuilder.create();
133.     dialog.setCancelable(false);// back
134.     dialog.setCanceledOnTouchOutside(false);
135.     SetDialogWidth(dialog).show();
136.     return false;
137.    } else if (!mNfcAdapter.isNdefPushEnabled())
138.    {
139.     Intent setnfc = new Intent(Settings.ACTION_NFCSHARING_SETTINGS);
140.     startActivity(setnfc);
141.     return false;
142.    } else
143.    {
144.     return true;
145.    }
146.   }
147.  }
148. 
149.  /**
150.   * @param dialog
151.   * @return
152.   */
153.  private Dialog SetDialogWidth(Dialog dialog)
154.  {
155.   DisplayMetrics dm = new DisplayMetrics();
156.   // 取得窗口属性
157.   getWindowManager().getDefaultDisplay().getMetrics(dm);
158.   // 窗口的宽度
159.   int screenWidth = dm.widthPixels;
160.   // 窗口高度
161.   int screenHeight = dm.heightPixels;
162.   WindowManager.LayoutParams params = dialog.getWindow().getAttributes();
163.   if (screenWidth > screenHeight)
164.   {
165.    params.width = (int) (((float) screenHeight) * 0.875);
166. 
167.   } else
168.   {
169.    params.width = (int) (((float) screenWidth) * 0.875);
170.   }
171.   dialog.getWindow().setAttributes(params);
172. 
173.   return dialog;
174.  }
175.  
176.  /**
177.   * check System Version()
178.   */
179.  private void checkSystemVersion()
180.  {
181.   // TODO Auto-generated method stub
182.   // if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN)
183. 
184.   LogUtil.i(MyConstant.Tag, Tag_ASSIST + "Product Model=" + android.os.     Build.MODEL + ", "+ android.os.Build.VERSION.SDK + ", " + android.os.Build. 
185.   VERSION.RELEASE);
186.   String systemModel;
187.   String releaseVersion;
188.   String sdkVersion;
189.   systemModel = android.os.Build.MODEL;
190.   sdkVersion = android.os.Build.VERSION.SDK;
191.   releaseVersion = android.os.Build.VERSION.RELEASE;
192.   if (Integer.parseInt(sdkVersion) > 15)
193.   {
194.    flagVersion = true;
195.   } else
196.   {
197.    flagVersion = false;
198.    Toast.makeText(mContext, "Your android system version is low to API-16", Toast.LENGTH_SHORT).show();
199.   }
200.  }
201. }

第21行为实现NFC功能的检测。在使用Android Beam功能前,需要确保设备支持NFC功能、NFC功能可用,且Android Beam功能是enable。具体可通过isEnabled() 和 isNdefPushEnabled()函数实现,参考代码第72~147行checkNFCFunction()函数。
第25行为实现系统版本的检测,因为setBeamPushUris ( )方法是在Android API 16+中才提供,所以此处加入了系统版本检测的代码。具体参考代码第176~200行checkSystemVersion()函数,该函数中主要通过android.os.Build.VERSION.SDK检测Android系统SDK API以及通过android.os.Build.VERSION.RELEASE检测Android系统版本。
第20~32行为BNM步骤1阶段,即准备Uri数据。代码通过Intent.ACTIONGET CONTENT进行过滤选择需要生产Uri的文件类型,本例中采用的是图片。
第38~64行中为onActivityResult()的处理,即用户选择了某个图片文件后进入到该函数中(根据第32行调用的startActivityForResult(i, mRequestCode)),具体包含了BNM步骤2阶段部分(即调用setBeamPushUris ( )方法)和UI操控部分(设置Button,用户触控结束该Activity)。
在AndroidManifest.xml中声明Activity,并添加NFC权限,其代码如下:

1. <uses-permission android:name="android.permission.NFC" />

2. <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
3. <uses-feature
 android: name="android.hardware.nfc" 
 android:required="true" />
4. <activity
5.             android:name="skyseraph.nfc_demo.p2p.beam.app.BeamFileDemo1"
6.             android:label="NFC_Demo_BeamFile-1" >
7.             <intent-filter>
8.                 <action android:name="android.intent.action.MAIN" />
9. 
10.                 <category android:name="android.intent.category.skyseraph_nfc_demo" />
11.             </intent-filter>
12.         </activity>

第1行为APP添加NFC权限。
第2行为添加APP读取外部存储器权限。注意,在Android API 17+中已不再强调需要这个权限,但未来的平台版本可能会对想要从外部存储器上读取文件的应用程序要求这个权限。因此,为了确保兼容性,在它变成必须的要求之前,申请这个权限。
第3行为添加只有存在NFC功能的设备才能使用你的应用。
Beam文件传输实例1的具体效果如图6-13所示,其中,图6-13(a)为初次启动界面,即提示用户选择待分享的图片,用户选择完后,显示图6-13(b)所示的界面,此时,用户触碰界面结束当前分享界面。


d2369c9d563d2fd7a250d8df1737c9246b77de59

6.4.3 Beam文件传输实例2:setBeamPushUrisCallback
在Beam文件传输实例2中,本书对setBeamPushUrisCallback ( )方法实现Android Beam传输Uri功能进行了实例描述。同实例1,由于Android Beam实现文件传输的方法中接收端可以由操作系统完成的,无需在App中处理,因此实例2中也仅包括Uri发送端。Uri的Push部分主要包括三个步骤,分别为:

(1)在Activity中实现CreateBeamUrisCallback接口;

(2)在需要的地方调用setBeamPushUrisCallback;

(3)回调函数中实现Beam Data。

具体参见实例代码中对应的注释,详细代码如下:

1. package skyseraph.nfc_demo.p2p.beam.app;   //声明包

2. import skyseraph.android.util.LogUtil;  //导入相关类
3. ……//该处省略了导入相关类的代码
4. public class BeamFileDemo2 extends Activity implements CreateBeamUrisCallback,   
 OnClickListener
5. { // BNM步骤1:在你的Activity中实现CreateBeamUrisCallback接口(implements)
6.  private static final String Tag_ASSIST = "[BeamFileDemo2]-";
7.  private Context mContext = null;
8.  private boolean flagNFC = false;
9.  private static int mRequestCode = 1001;
10.  private Intent mIntent;
11.  // NFC相关
12.  private NfcAdapter mNfcAdapter = null;
13. 
14.  @Override
15.  protected void onCreate(Bundle savedInstanceState)
16.  {
17.   // TODO Auto-generated method stub
18.   super.onCreate(savedInstanceState);
19.   mContext = this;
20.   LogUtil.i(MyConstant.Tag, Tag_ASSIST + "into onCreate");
21.   flagNFC = checkNFCFunction();
22.   LogUtil.i(MyConstant.Tag, Tag_ASSIST + "flagNFC=" + flagNFC);
23. 
24.   if (flagNFC)
25.   {
26.    Intent i = new Intent(Intent.ACTION_GET_CONTENT);   
    // 用户选择某种特殊数据并返回
27.    i.setType("image/*"); // 查看类型,如果是其他类型,如视频则替换成 video/*,或 */*
28.    startActivityForResult(i, mRequestCode);
29.   }
30.  }
31. 
32.  // BNM步骤3:回调函数中实现Beam Data。
33.  // 当发现有支持Beam的手机时,该回调接口会自动激活,你只需将你需要Beam的文件Uri准备好  
  // 并返回即可。
34.  // 本例中Uri为用户选择的图片为例。
35.  @Override
36.  public Uri[] createBeamUris(NfcEvent event)
37.  {
38.   // TODO Auto-generated method stub
39.   LogUtil.i(MyConstant.Tag, Tag_ASSIST + "into createBeamUris");
40.   Uri[] uriArray = new Uri[]
41.   { mIntent.getData() };
42.   LogUtil.i(MyConstant.Tag, Tag_ASSIST + "uriArray=" + uriArray);
43.   return uriArray;
44.  }
45. 
46.  /*
47.   * (non-Javadoc)
48.   * 
49.   * @see android.app.Activity#onActivityResult(int, int,
50.   * android.content.Intent)
51.   */
52.  @Override
53.  protected void onActivityResult(int requestCode, int resultCode, Intent data)
54.  {
55.   // TODO Auto-generated method stub
56.   // super.onActivityResult(requestCode, resultCode, data);
57.   if (requestCode == mRequestCode && resultCode == RESULT_OK)
58.   {
59.    mIntent = data;
60.    // BNM步骤2:call setBeamPushUrisCallback anywhere your want
61.    mNfcAdapter.setBeamPushUrisCallback(this, this);
62. 
63.    Button btn = new Button(this);
64.    btn.setText("Touch this to finish!");
65.    btn.setTextSize(30);
66.    btn.setOnClickListener(this);
67.    setContentView(btn);
68.   }
69.  }
70. 
71.  @Override
72.  public void onClick(View v)
73.  {
74.   finish();
75.  }
76. 
77.  /**
78.   * NFC Function Check By skyseraph 2013-2
79.   */
80.  private boolean checkNFCFunction()
81.  {
82.   // Beam文件传输实例1中的checkNFCFunction()函数,该处省略
83.  }
84. 
85.  /**
86.   * @param dialog
87.   * @return
88.   */
89.  private Dialog SetDialogWidth(Dialog dialog)
90.  {
91.   // Beam文件传输实例1中的SetDialogWidth ()函数,该处省略
92.  }
93. }

第4行为BNM步骤1阶段,即实现CreateBeamUrisCallback接口。实现该接口后,该Activity中需重载createBeamUris()函数。在该函数中返回需要Beam 的文件的Uri。
第21行为实现NFC功能的检测。在使用Android Beam功能前,需要确保设备支持NFC功能、NFC功能可用,且Android Beam功能是enable,具体通过isEnabled() 和 isNdefPushEnabled()函数实现,参考代码第77~83行checkNFCFunction()函数。
第26~28行为用户选择Uri数据,代码中通过Intent.ACTION_GET_CONTENT进行过滤选择需要生产Uri的文件类型,本例中采用的是图片。
第60~61行为BNM步骤2阶段,即调用setBeamPushUrisCallback(),调用该方法后Activity中会接收一个回调接口,一旦有发现其他设备的Beam数据,createBeamUris()就会自动调用。
第32~44行为BNM步骤3阶段,即回调接口createBeamUris(NfcEvent)中实现Beam Data,具体为将第26描述的用户选择的Uri数据获取并返回,Uri数组中的图片文件就会自动Beam to目标设备。
AndroidManifest.xml中声明Activity,并添加NFC权限,其代码如下:

1. <uses-permission android:name="android.permission.NFC" />

2. <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
3. <uses-feature   
 android: name="android.hardware.nfc"   
 android:required="true" />
4. <activity
5.             android:name="skyseraph.nfc_demo.p2p.beam.app.BeamFileDemo2"
6.             android:label="NFC_Demo_BeamFile-2" >
7.             <intent-filter>
8.                 <action android:name="android.intent.action.MAIN" />
9. 
10.                 <category android:name="android.intent.category.skyseraph_nfc_demo" />
11.             </intent-filter>
12.         </activity>

第1行为APP添加NFC权限。
第2行为添加APP读取外部存储器权限。注意,虽然在Android 4.2.2 (API 17)中已不再强调需要这个权限,但未来的平台版本可能会对想要从外部存储器上读取文件的应用程序要求这个权限。因此,为了确保兼容性,在它变成必须的要求之前,申请这个权限。
第3行为添加只有存在NFC功能的设备才能使用应用。
Beam文件传输实例2的具体效果如图6-14所示。其中,图6-14(a)为初次启动界面,即提示用户选择待分享的图片。用户选择完后,显示图6-14(b)所示的界面。此时,用户触碰界面结束当前分享界面。


f40926aac0e608ea2936cb127b77134f6fa28e37
相关文章
|
2天前
|
Android开发 开发者 Kotlin
探索安卓开发中的新特性
【9月更文挑战第14天】本文将引导你深入理解安卓开发领域的一些最新特性,并为你提供实用的代码示例。无论你是初学者还是经验丰富的开发者,这篇文章都会给你带来新的启示和灵感。让我们一起探索吧!
|
6天前
|
IDE 开发工具 Android开发
安卓与iOS开发对比:平台选择对项目成功的影响
【9月更文挑战第10天】在移动应用开发的世界中,选择正确的平台是至关重要的。本文将深入探讨安卓和iOS这两大主要移动操作系统的开发环境,通过比较它们的市场份额、开发工具、编程语言和用户群体等方面,为开发者提供一个清晰的指南。我们将分析这两个平台的优势和劣势,并讨论如何根据项目需求和目标受众来做出最佳选择。无论你是初学者还是有经验的开发者,这篇文章都将帮助你更好地理解每个平台的特性,并指导你做出明智的决策。
|
2天前
|
XML 编解码 Android开发
安卓开发中的自定义视图控件
【9月更文挑战第14天】在安卓开发中,自定义视图控件是一种高级技巧,它可以让开发者根据项目需求创建出独特的用户界面元素。本文将通过一个简单示例,引导你了解如何在安卓项目中实现自定义视图控件,包括创建自定义控件类、处理绘制逻辑以及响应用户交互。无论你是初学者还是有经验的开发者,这篇文章都会为你提供有价值的见解和技巧。
|
4天前
|
API Android开发 iOS开发
安卓与iOS开发中的线程管理对比
【9月更文挑战第12天】在移动应用的世界中,安卓和iOS平台各自拥有庞大的用户群体。开发者们在这两个平台上构建应用时,线程管理是他们必须面对的关键挑战之一。本文将深入探讨两大平台在线程管理方面的异同,通过直观的代码示例,揭示它们各自的设计理念和实现方式,帮助读者更好地理解如何在安卓与iOS开发中高效地处理多线程任务。
|
5天前
|
搜索推荐 Android开发 UED
安卓开发中的自定义视图:打造个性化用户界面
【9月更文挑战第11天】在安卓应用开发领域,自定义视图是实现独特用户体验的基石。本文将引导你通过一个简单的自定义视图示例,探索如何从零开始创建并应用自定义组件,以增强你的应用界面。我们将一起学习如何扩展View类,重写onDraw方法,处理触摸事件,并最终在我们的安卓项目中使用这个自定义视图。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供清晰的步骤和实用的技巧,帮助你提升用户界面设计的能力。
|
6天前
|
开发框架 Android开发 iOS开发
探索安卓与iOS开发的差异:构建未来应用的指南
在移动应用开发的广阔天地中,安卓与iOS两大平台各占半壁江山。本文将深入浅出地对比这两大操作系统的开发环境、工具和用户体验设计,揭示它们在编程语言、开发工具以及市场定位上的根本差异。我们将从开发者的视角出发,逐步剖析如何根据项目需求和目标受众选择适合的平台,同时探讨跨平台开发框架的利与弊,为那些立志于打造下一个热门应用的开发者提供一份实用的指南。
18 5
|
6天前
|
开发工具 Android开发 iOS开发
安卓与iOS开发:平台选择的艺术与科学
在移动应用开发的广阔天地中,安卓与iOS两大平台如同东西方哲学的碰撞,既有共通之处又各具特色。本文将深入探讨这两个平台的设计理念、开发工具和市场定位,旨在为开发者提供一份简明扼要的指南,帮助他们在这场技术与商业的博弈中找到自己的道路。通过比较分析,我们将揭示每个平台的优势与局限,以及它们如何影响应用的性能、用户体验和市场接受度。无论你是初涉江湖的新手,还是经验丰富的老手,这篇文章都将为你的选择提供新的视角和思考。
17 5
|
7天前
|
安全 Android开发 开发者
探索安卓开发的未来:Kotlin的崛起与Flutter的挑战
在移动开发的广阔天地中,安卓平台始终占据着举足轻重的地位。随着技术的不断进步和开发者需求的多样化,Kotlin和Flutter成为了改变游戏规则的新玩家。本文将深入探讨Kotlin如何以其现代化的特性赢得开发者的青睐,以及Flutter凭借跨平台的能力如何挑战传统的安卓开发模式。通过实际案例分析,我们将揭示这两种技术如何塑造未来的安卓应用开发。
23 6
|
6天前
|
开发工具 Android开发 Swift
探索安卓与iOS开发的差异:从新手到专家的旅程
在数字时代的浪潮中,移动应用开发已成为连接世界的桥梁。本文将深入探讨安卓与iOS这两大主流平台的开发差异,带领读者从零基础出发,逐步了解各自的特点、开发环境、编程语言及市场策略。无论你是梦想成为移动应用开发者的初学者,还是希望扩展技能边界的资深开发者,这篇文章都将为你提供宝贵的见解和实用的建议。