第一篇:安卓NFC开发学习笔记
大家学习android开发建议首选android开发文档,该文档在你下载的sdk中,路径:/sdk/docs/index.html
目前NFC应用的大的框架上的理解:
我使用的API LEVEL是19,支持的API有三个:android.nfc,android.nfc.cardemulator,android.nfc.tech NFC在手机上的应用大体分为两类:读卡器和卡
android.nfc.cardemulator接口是为NFC作为卡应用提供的接口,在较低版本的API上是没有的
android.nfc.tech,android.nfc接口是为NFC作为读卡器应用提供的接口
首先说作为卡,nfc有两种实现方式,一个是使用NFC芯片作为卡,另一个是使用SIM作为卡
Figure 1.NFC card emulation with a secure element.至于从读卡器发送的指令到底是传递到NFC芯片还是SIM由NFC Controler控制,图中Secure Element是指SIM,Host-CPU指NFC芯片 android提供HostApduService用于NFC芯片,OffHostApduService用于SIM芯片,传递方向在res/xml文件中通过AID来控制
ps:Host-Based Card Emulator 简称为HCE
代码实现:
AndroidManifest.xml 中 配置service,因为作为卡实现的话,NFC功能是作为service存在的
android:permission=“android.permission.BIND_NFC_SERVICE”> android:resource=“@xml/apduservice”/>
res/xml/apduservice.xml 中配置service响应的AID android:requireDeviceUnlock=“false”> android:category=“other”>
配置文件完成后编写service的处理方法:
NFCService需要继承HostApduService,如果需要与Activity通信,建议采用广播方式
也可以自己实现观察者模式,只是这样就需要持有Activity的引用,感觉不太好 NFCService.java public class NFCService extends HostApduService {
private Intent intent = new Intent(“com.example.communication.RECEIVER”);
@Override public void onCreate(){
//启动Acivity
Intent i = new Intent();
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);//需要启动的Activity不是当前Activity的时候需要用FLAG_ACTIVITY_NEW_TASK
i.setAction(“com.apdu.nfc”);
getApplication().startActivity(i);
Toast.makeText(getApplicationContext(), “Service启动”, Toast.LENGTH_LONG).show();
} @Override public byte[] processCommandApdu(byte[] commandApdu, Bundle extras){//当注册的AID被选中后,后续指令被分发到这个处理函数中
byte[] sw = new byte[]{(byte)0x90,(byte)0x00};
byte[] response = new byte[5];
if(commandApdu[0]==(byte)0x00 &&commandApdu[1]==(byte)0xA4&& commandApdu[2]==(byte)0x04
&& commandApdu[4]==(byte)0x07&& commandApdu[5]==(byte)0xF0)
{
return sw;
}
else
{
//apdu处理逻辑
switch(commandApdu[1])
{
case(byte)0xA8:
break;
case(byte)0xAE:
break;
default:
return sw;
}
}
intent.putExtra(“command”, commandApdu);
intent.putExtra(“response”, response);
sendBroadcast(intent);//利用广播与Activity通信
return response;//SW值需要包含在response中
}
@Override public void onDeactivated(int reason){
if(reason==HostApduService.DEACTIVATION_DESELECTED)
{
Toast.makeText(getApplicationContext(), “已选择其它应用”, Toast.LENGTH_LONG).show();
}
else
{
Toast.makeText(getApplicationContext(), “连接断开”, Toast.LENGTH_LONG).show();
} }
@Override
public void onDestroy()
{
Toast.makeText(getApplicationContext(), “Service关闭”, Toast.LENGTH_LONG).show();super.onDestroy();
}
框架搭建好剩余的事情就很简单了,apdu的处理逻辑在processCommandApdu方法中实现即可
以上是Host-CPU方式的实现,SIM方式,API介绍中说该方式没有提供可供操作的API,也就是说Android不会监听SIM卡与读卡器之间的通信
所以NFCOffService 只需要实现onBind接口,这样绑定该Service的Activity可以对NFCOffService进行有限操作 public class NFCOffService extends OffHostApduService {
@Override public IBinder onBind(Intent intent){
// TODO Auto-generated method stub
return null;} } 上面没有提到的就是,如果你需要使用NFC,需要在Manifest中申请NFC权限:
android.nfc.tech,android.nfc接口是为NFC作为读卡器应用提供的接口 接口定义了三种Action Tags:ACTION_NDEF_DISCOVERED,ACTION_TECH_DISCOVERED,ACTION_TAG_DISCOVERED。
当你在Manifest文件中将Activity的action-filter设置为这三个Tag中的一种或几种时,NFC响应事件会按照如图流程处理
我的理解是ACTION_NDEF_DISCOVERED 是用于两台NFC手机之间传输文件的
ACTION_TECH_DISCOVERED,ACTION_TAG_DISCOVERED才是用于NFC与卡进行通讯的 所以开发第一步是在Manifest中配置你的Action:
android:resource=“@xml/nfc_tech_filter” />
当然API中说明你可以将多个tech写在一个tech-list中,我做了尝试,这样做会引出一个问题,在程序未启动的情况下当手机刷卡时不会自动打开程序 如果想要自动打开需要按照上面这种写法,tech的个数可以根据你想要支持的卡类型进行调整
配置完成后,可以开始编写自己的Activity的java代码了
在onCreate方法中,需要获取NfcAdapter的引用,从名字可以看出这是一个适配器
NfcAdapter nfcAdapter;PendingIntent pendingIntent;@Override protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.pos_main);
dc =(Button)findViewById(R.id.button4DC);
ecc =(Button)findViewById(R.id.button4ECC);
qpboc =(Button)findViewById(R.id.button4QPBOC);
logWindow=(TextView)findViewById(R.id.communication4Financy);
nfcAdapter=NfcAdapter.getDefaultAdapter(this);
pendingIntent = PendingIntent.getActivity(this, 0, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);
onNewIntent(getIntent());} 这里使用PendingIntent,该Intent与普通的Intent不同的是它是有一个延迟启动的功能,它启动时会回调onNewIntent函数,这样能够实现NFC与Activity的交互
pendingIntent = PendingIntent.getActivity(this, 0, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);的含义是将Intent传递给this Activity 在onPause与onResume中需要添加代码
public void onPause(){
super.onPause();
nfcAdapter.disableForegroundDispatch(this);}
public void onResume(){
super.onResume();
nfcAdapter.enableForegroundDispatch(this, pendingIntent, FILTERS, TECHLISTS);} enableForegroundDispatch的作用是,当NFC事件发生时如果当前Activity不是注册了NFC action-filter的Activity,手机会显示注册了NFC事件的Activity供用户选择
如果当前Activity注册了NFC action 则将事件优先交由当前Activity处理。onNewIntent实现:
@Override public void onNewIntent(Intent intent){
Parcelable p = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
if(p==null)
{
return;
}
Tag nfcTag =(Tag)p;
final IsoDep isodep = IsoDep.get(nfcTag);// final NfcA isodep = NfcA.get(nfcTag);
final byte[] cmd = {(byte)0x00, // CLA Class
(byte)0xB4, // INS Instruction
(byte)0x04, // P1 Parameter 1
(byte)0x00, // P2 Parameter 2
(byte)0x00, // Le
};
try {
isodep.connect();
byte[] reaponse=null;
logWindow.append(“00B4040000”+'n');
try {
reaponse = isodep.getHistoricalBytes();
logWindow.append(Util.bytes2HexString(reaponse)+'n');
reaponse =isodep.transceive(cmd);
logWindow.append(Util.bytes2HexString(reaponse)+'n');
} catch(IOException e){
// TODO Auto-generated catch block
e.printStackTrace();
}
} catch(IOException e1){
// TODO Auto-generated catch block
e1.printStackTrace();
}finally{
try {
isodep.close();
} catch(IOException e){
// TODO Auto-generated catch block
e.printStackTrace();
}
} } 首先需要获取Tag,Tag是操作NFC的基础
Parcelable p = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
if(p==null)
{
return;
}
Tag nfcTag =(Tag)p;再将tag转换成特定的通讯协议 final IsoDep isodep = IsoDep.get(nfcTag);连接读卡器: isodep.connect();获取历史字节:
reaponse = isodep.getHistoricalBytes();响应指令:
reaponse =isodep.transceive(cmd);最后断开连接:(断开动作应该在Activity关闭或者pause的时候发生)finally{
try {
isodep.close();
} catch(IOException e){
// TODO Auto-generated catch block
e.printStackTrace();
}
} onNewIntent里面可以实现业务逻辑的处理以及UI
第二篇:如何学习安卓开发
如何学习安卓开发?安卓开发学习已经成为IT行业的新潮流。时下,Android也以其创造力、前瞻性、延续性和实现能力成为行业首领,可是怎么学好Android呢?今天,欧柏泰克的老师告诉你如何学好Android。
熟悉Java基础知识
Android应用的开发语言用的是Java语言,并且在Android中也用到了Java核心类库的大量的类,因此,在学习Android开发之前,可以先把Java基本语法和Java SE的基础类库好好学习一下。Android应用程序开发是以Java语言为基础的,所以没有扎实的Java基础知识,只是机械的照抄别人的代码,是没有任何意义的。建议在Android课程前期的Java学习阶段中,需要用心的学好。
熟悉一门编程语言
现在大学里面和计算机相关的专业甚至理工类专业一般都会开设C语言课程,只是很多同学在大学期间并没有好好学习,如果对它掌握的不太好或者很久没用了,建议先从将其好好复习一下,将其基本的语法再好好回顾一下,最好能搭建一个环境来运行、调试它。如果没有学过,不妨也提前学习一下。大部分的高校所开设的C语言使用的教材都是用它作为教材,因此无论是购买还是借阅,都容易找到;
熟悉数据结构和算法基础知识
如果后续有志于游戏方面的开发,最好具备一定的数据结构和算法基础知识。虽然现代的高级编程语言中,其类库中已经帮我们实现了大部分的数据结构,一般情况下,我们直接使用即可。但如果能对其原理有所了解,当需要在这些数据结构和算法中间的时候,可以更加的清楚到底应该选择哪个数据结构或者算法。另外,在图形图像处理上面,线性代数的作用也非常重要,如果能掌握一点这方面的基础知识,无疑也会在后续的学习中如虎添翼。
第三篇:如何学习安卓
如何学习安卓
想学编程开发,那要先会一门编程语言,现在可以试着去学学C语言,虽然这个安卓沾不上边,但是,C语言的编程思想还是很重要的,学完了C语言之后,他的语法基本上和所有的编程语言都很相像,能影响你的思维,帮助你理解其他的编程语言的。之后呢,在好好看看《数据结构》,这很重要。然后再去学学Java语言,因为Android的应用的开发语言用的是Java,所以一定要好好学习。
最后了解下数据库,我们在学习数据库之前都先学了《数据库原理》《离散数学》《关系代数》,有了这些基础之后再去学数据库,数据库也有很多可以选择的,推荐mysql。
加油!
第四篇:安卓学习
安卓开发学习准备要点介绍
要说当下IT行业当中最具创造力、前瞻性、延续性和实现能力,想必有相当的人会把票投给Google的安卓,安卓开发学习也成为新潮流。安卓开发学习要做什么准备?下面就由福州卓跃教育具体介绍。
首先,最好先熟悉一门编程语言,现在大学里面和计算机相关的专业甚至理工类专业一般都会开设C语言课程,只是很多同学在大学期间并没有好好学习,如果对它掌握的不太好或者很久没用了,建议先从将其好好复习一下,将其基本的语法再好好回顾一下,最好能搭建一个环境来运行、调试它。如果没有学过,不妨也提前学习一下,可以参考清华大学出版社出版的谭浩强老师的《C语言程序设计》,推荐这本书的原因一是它已经经过了多年的考验,应该说还是比较严谨的;其次就是大部分的高校所开设的C语言使用的教材都是用它作为教材,因此无论是购买还是借阅,都容易找到。
其次,如果后续有志于游戏方面的开发,最好具备一定的数据结构和算法基础知识。虽然现代的高级编程语言中,其类库中已经帮我们实现了大部分的数据结构,一般情况下,我们直接使用即可。但如果能对其原理有所了解,当需要在这些数据结构和算法中间的时候,可以更加的清楚到底应该选择那个数据结构或者算法。另外,在图形图像处理上面,线性代数的作用也非常重要,如果能掌握一点这方面的基础知识,无疑也会在后续的学习中如虎添翼。举个例子,在Android中,有一个用于图形变换的类Matrix,用起来稍有点难。
第三,因为Android的应用的开发语言用的是Java语言,并且在Android中也用到了Java核心类库的大量的类,因此,在学习Android开发之前,可以先把Java基本语法和Java SE的基础类库好好学习一下Android应用程序开发是以Java语言为基础的,所以没有扎实的Java基础知识,只是机械的照抄别人的代码,是没有任何意义的。
至少要掌握以下两个方面的内容:a)Java基础语法:具体的知识点列表可以在这里下载:《Java知识点列表》V1.0。这部分内容没有讨价还价的余地,必须烂熟于胸。至于具体的学习方法,可以看书或者是看视频,但是关键是要多加练习,无论是书上的练习还是视频里面的练习,都需要仔仔细细的完成;b)设计模式:由于在Android系统的框架层当中,使用了大量的设计模式,如果没有这个方面的知识,对于Android的理解就会大打折扣。设计模式的种类非常之多,一个一个的全部掌握,是不现实的,必须首先掌握面向对象的基础设计原则,有了这些基础原则的支持,就可以举一反三。这部分内容可以在《Effective Java》和《Agile.Software.Development:
Principles,Patterns.and.Practices》这两本书中找到。
第五篇:何避免安卓开发学习中的弯路
何避免安卓开发学习中的弯路
最初参加安卓开发培训是因为公司安排,想通过有规律的学习来提升安卓开发能力,也是因为避免自学安卓开发中的弯路。
本人在学习安卓开发培训的这期间,感觉学到的远比自己自学一年时间,所学习到的、了解到的要多很多,也收获了很多人脉。
首先,自学的时候没有方向性、没有目的性,这种看到哪个知识点就去学习那方面知识点的方式,在一定程度上造成学习的事倍功半;而在在安卓开发培训的课堂上,则明确设置了学习方向,由浅显入深,循序渐进;
第二,自学的时候遇到问题除了百度或谷歌外,别无他法;而在安卓开发培训学习,除了百度或谷歌外,还有一批优秀的、经验丰富的培训师可以咨询,这样就可以避免自己的学习陷入死循环;
第三,自学的时候随意性比较大,没有时间上的观念,想学就学一下,不想学就丢一边,特别是在学习过程中遇到难题的时候;而在安卓开发培训学习的话,则有良好的时间安排,一方面既保证了足够的学习时间,另一方面又保证了自我钻研时间;
第四,自学的时候不会想着自己要去做某些项目;但当你身处安
卓开发培训的课堂上,就会在你阶段性的学习过程中要求你独立完成具有某些功能的项目,这样就能更快,更全面的掌握技能,从而更有助于加深对安卓开发技能方面的理解。
选择千锋安卓开发培训机构,起初是因为公司经过多家机构的筛选和比较,了解到千锋的讲师背景、学习氛围、课程体系在业内都是名列前矛,加之口碑非常好,所以选择了千锋这家培训机构。
以上只是针对我一个过来人,对于自己学习安卓开发培训的四个月的真实领悟和总结,想对以后想要加入安卓开发学习的朋友们提供一些有用的建议吧!