Android传感器编程入门

一、前言

我很喜欢电脑,可是笔记本还是太大,笔记本电脑再小还是要弄个小包背起来的,智能手机则不同,它完全就是一个手机,可以随意装在一个口袋里随身携带。因此我在2002年左右时最喜欢玩装备是Dell的PDA,2007年的时候最喜欢玩的是N73,而在2010年最喜欢玩的则是Milestone。眼见着手机的功能越来越强,时至今日智能手机甚至在某些方面已经强过了台式机和笔记本。本节课讲的就是智能手机强过台式机和笔记本的地方:传感器。

2008年的时候我很喜欢我的小白笔记本Macbook,喜欢玩它的一个小软件,一拍桌子,笔记本感受到了震动,它就转换了一个桌面出来,这让我像个小孩子一样没事就拍拍桌子。这一功能这得益于苹果笔记本内置有传感器。

我不知道iPhone手机是不是第一个把各种各样的传感器运用在手机上的,不过我知道iPhone是把传感器运用在手机上最成功的第一个。随后的Android系统也内置了大量的传感器,这让Android系统手机和普通的诺基亚智能机和Windows CE智能机相比牛气了许多,在拥有了Milestone之后,我的N73就被仍在抽屉的角落里了。

从Android1.5开始,系统内置了对多达八种传感器的支持,他们分别是:加速度传感器(accelerometer),陀螺仪(gyroscope),环境光照传感器(light),磁力传感器(magnetic field),方向传感器(orientation),压力传感器(pressure),距离传感器(proximity)和温度传感器(temperature)。

利用这些传感器我们可以制作出各种有趣的应用程序和游戏。譬如在口袋里晃一晃手机,手机就开始神不知鬼不觉的录音,不要着急这个很容易做,我们在本文的结尾就一起制作这个小应用。

本讲的学习方式还是在实战中学习,需要提醒的是模拟器中无法模拟传感器,因此你需要准备一款Android真机才能运行本讲的例子。

二、实例:手机传感器清单

我们还是先看程序后解释,

1、创建一个项目 Lesson37_HelloSensor , 主Activity名字叫 mainActivity.java

2、UI布局文件main.xml的内容如下:

1<?xml version="1.0" encoding="utf-8"?>
2<linearlayout android:layout_height="fill_parent"android:layout_width="fill_parent" android:orientation="vertical"xmlns:android="/apk/res/android">
3<textview android:layout_height="wrap_content"android:layout_width="fill_parent" android:text=""android:id="@+id/TextView01">
4</textview></linearlayout>

3、mainActivity.java的内容如下:

01package basic.android.lesson37;
02 
03import java.util.List;
04 
05import android.app.Activity;
06import android.content.Context;
07import android.hardware.Sensor;
08import android.hardware.SensorManager;
09import android.os.Bundle;
10import android.widget.TextView;
11 
12public class MainActivity extends Activity {
13 
14    /** Called when the activity is first created. */
15    @Override
16    public void onCreate(Bundle savedInstanceState) {
17        super.onCreate(savedInstanceState);
18        setContentView(R.layout.main);
19 
20        //准备显示信息的UI组建
21        final TextView tx1 = (TextView) findViewById(R.id.TextView01);
22 
23        //从系统服务中获得传感器管理器
24        SensorManager sm = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
25 
26        //从传感器管理器中获得全部的传感器列表
27        List<sensor> allSensors = sm.getSensorList(Sensor.TYPE_ALL);
28 
29        //显示有多少个传感器
30        tx1.setText("经检测该手机有" + allSensors.size() + "个传感器,他们分别是:\n");
31 
32        //显示每个传感器的具体信息
33        for (Sensor s : allSensors) {
34 
35            String tempString = "\n" "  设备名称:" + s.getName() + "\n""  设备版本:" + s.getVersion() + "\n" "  供应商:"
36                    + s.getVendor() + "\n";
37 
38            switch (s.getType()) {
39            case Sensor.TYPE_ACCELEROMETER:
40                tx1.setText(tx1.getText().toString() + s.getType() + " 加速度传感器accelerometer" + tempString);
41                break;
42            case Sensor.TYPE_GYROSCOPE:
43                tx1.setText(tx1.getText().toString() + s.getType() + " 陀螺仪传感器gyroscope" + tempString);
44                break;
45            case Sensor.TYPE_LIGHT:
46                tx1.setText(tx1.getText().toString() + s.getType() + " 环境光线传感器light" + tempString);
47                break;
48            case Sensor.TYPE_MAGNETIC_FIELD:
49                tx1.setText(tx1.getText().toString() + s.getType() + " 电磁场传感器magnetic field" + tempString);
50                break;
51            case Sensor.TYPE_ORIENTATION:
52                tx1.setText(tx1.getText().toString() + s.getType() + " 方向传感器orientation" + tempString);
53                break;
54            case Sensor.TYPE_PRESSURE:
55                tx1.setText(tx1.getText().toString() + s.getType() + " 压力传感器pressure" + tempString);
56                break;
57            case Sensor.TYPE_PROXIMITY:
58                tx1.setText(tx1.getText().toString() + s.getType() + " 距离传感器proximity" + tempString);
59                break;
60            case Sensor.TYPE_TEMPERATURE:
61                tx1.setText(tx1.getText().toString() + s.getType() + " 温度传感器temperature" + tempString);
62                break;
63            default:
64                tx1.setText(tx1.getText().toString() + s.getType() + " 未知传感器" + tempString);
65                break;
66            }
67        }
68 
69    }
70}</sensor>

4、连接真机Milestone,编译并运行程序,显示结果如下:

5、结合上面的程序我们做一些解释。

  1. Android所有的传感器都归传感器管理器 SensorManager 管理,获取传感器管理器的方法很简单:

    String service_name = Context.SENSOR_SERVICE;

    SensorManager sensorManager = (SensorManager)getSystemService(service_name);

  2. 现阶段Android支持的传感器有8种,它们分别是:
    传感器类型常量内部整数值中文名称
    Sensor.TYPE_ACCELEROMETER1加速度传感器
    Sensor.TYPE_MAGNETIC_FIELD2磁力传感器
    Sensor.TYPE_ORIENTATION3方向传感器
    Sensor.TYPE_GYROSCOPE4陀螺仪传感器
    Sensor.TYPE_LIGHT5环境光照传感器
    Sensor.TYPE_PRESSURE6压力传感器
    Sensor.TYPE_TEMPERATURE7温度传感器
    Sensor.TYPE_PROXIMITY8距离传感器
  3. 从传感器管理器中获取其中某个或者某些传感器的方法有如下三种:

    第一种:获取某种传感器的默认传感器

    Sensor defaultGyroscope = sensorManager.getDefaultSensor(Sensor.TYPE_GYROSCOPE);

    第二种:获取某种传感器的列表

    List<Sensor> pressureSensors = sensorManager.getSensorList(Sensor.TYPE_PRESSURE);

    第三种:获取所有传感器的列表,我们这个例子就用的第三种

    List<Sensor> allSensors = sensorManager.getSensorList(Sensor.TYPE_ALL);

  4. 对于某一个传感器,它的一些具体信息的获取方法可以见下表:
    方法描述
    getMaximumRange()最大取值范围
    getName()设备名称
    getPower()功率
    getResolution()精度
    getType()传感器类型
    getVentor()设备供应商
    getVersion()设备版本号

三、实例:窈窈录音器

通过上面的例子我们学会了如何获得某种类型的传感器,下面我通过一个实例来学会如何使用某一个类型的传感器。我们这里使用加速度传感器来实现这样一个功能:开启我们的录音程序放在你的口袋或者提包里,需要录音的时候把衣服整理一下,或者把提包挪动个位置,那么此时手机就会感受到变化从而开始录音。由此达到神不知鬼不觉的录音效果。说起来似乎有点神,其实做起来很简单,让我们开始吧。

简单的录音程序已经在第28讲的时候做过了,我们在28讲程序的基础上写本讲的代码。

1、新建一个项目 Lesson37_YYRecorder , 主文件叫 MainActivity.java ,具体信息都可以参见第二十八讲的“窈窈录音”的例子。

2、这里只贴出于28讲不同的 MainActivity.java  的代码,请注意看注释:

001package basic.android.lesson37;
002 
003import java.io.File;
004import java.io.IOException;
005import java.util.Calendar;
006import java.util.Locale;
007 
008import android.app.Activity;
009import android.content.Context;
010import android.hardware.Sensor;
011import android.hardware.SensorEvent;
012import android.hardware.SensorEventListener;
013import android.hardware.SensorManager;
014import android.media.MediaRecorder;
015import android.os.Bundle;
016import android.text.format.DateFormat;
017import android.view.View;
018import android.widget.Button;
019import android.widget.TextView;
020import android.widget.Toast;
021 
022public class MainActivity extends Activity {
023 
024    //录音和停止按钮
025    private Button recordButton;
026    private Button stopButton;
027 
028    //检测摇动相关变量
029    private long initTime = 0;
030    private long lastTime = 0;
031    private long curTime = 0;
032    private long duration = 0;
033 
034    private float last_x = 0.0f;
035    private float last_y = 0.0f;
036    private float last_z = 0.0f;
037 
038    private float shake = 0.0f;
039    private float totalShake = 0.0f;
040 
041    //媒体录音器对象
042    private MediaRecorder mr;
043 
044    //是否正在录音
045    private boolean isRecoding = false;
046 
047    @Override
048    public void onCreate(Bundle savedInstanceState) {
049        super.onCreate(savedInstanceState);
050        setContentView(R.layout.main);
051 
052        // UI组件
053        recordButton = (Button) this.findViewById(R.id.Button01);
054        stopButton = (Button) this.findViewById(R.id.Button02);
055        final TextView tx1 = (TextView)this.findViewById(R.id.TextView01);
056 
057        // 录音按钮点击事件
058        recordButton.setOnClickListener(new View.OnClickListener() {
059 
060            @Override
061            public void onClick(View v) {
062                //如果没有在录音,那么点击按钮可以开始录音
063                if(!isRecoding){
064                    startRecord();
065                }
066            }
067        });
068 
069        // 停止按钮点击事件
070        stopButton.setOnClickListener(new View.OnClickListener() {
071 
072            @Override
073            public void onClick(View v) {
074                initShake();
075                //如果正在录音,那么可以停止录音
076                if (mr != null) {
077                    mr.stop();
078                    mr.release();
079                    mr = null;
080                    recordButton.setText("录音");
081                    Toast.makeText(getApplicationContext(), "录音完毕", Toast.LENGTH_LONG).show();
082                    isRecoding = false;
083 
084                }
085            }
086        });
087 
088        // 获取传感器管理器
089        SensorManager sm = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
090        // 获取加速度传感器
091        Sensor acceleromererSensor = sm.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
092 
093        // 定义传感器事件监听器
094        SensorEventListener acceleromererListener = newSensorEventListener() {
095 
096            @Override
097            public void onAccuracyChanged(Sensor sensor, int accuracy) {
098                //什么也不干
099            }
100 
101            //传感器数据变动事件
102            @Override
103            public void onSensorChanged(SensorEvent event) {   
104 
105                //如果没有开始录音的话可以监听是否有摇动事件,如果有摇动事件可以开始录音
106                if(!isRecoding){
107                    //获取加速度传感器的三个参数
108                    float x = event.values[SensorManager.DATA_X];
109                    float y = event.values[SensorManager.DATA_Y];
110                    float z = event.values[SensorManager.DATA_Z];
111 
112                    //获取当前时刻的毫秒数
113                    curTime = System.currentTimeMillis();
114 
115                    //100毫秒检测一次
116                    if ((curTime - lastTime) > 100) {
117 
118                        duration = (curTime - lastTime);
119 
120                        // 看是不是刚开始晃动
121                        if (last_x == 0.0f && last_y == 0.0f && last_z ==0.0f) {
122                            //last_x、last_y、last_z同时为0时,表示刚刚开始记录
123                            initTime = System.currentTimeMillis();
124                        else {
125                            // 单次晃动幅度
126                            shake = (Math.abs(x - last_x) + Math.abs(y - last_y) + Math.abs(z - last_z)) / duration * 100;
127                        }
128 
129                        //把每次的晃动幅度相加,得到总体晃动幅度
130                        totalShake += shake;
131 
132                        // 判断是否为摇动,这是我自己写的标准,不准确,只是用来做教学示例,别误会了^_^
133                        if (totalShake > 10 && totalShake / (curTime - initTime) * 1000 10) {
134                            startRecord();
135                            initShake();
136                        }
137 
138                        tx1.setText("总体晃动幅度="+totalShake+ "\n平均晃动幅度="+totalShake / (curTime - initTime) * 1000 );
139                    }
140 
141                    last_x = x;
142                    last_y = y;
143                    last_z = z;
144                    lastTime = curTime;
145                }
146            }
147 
148        };
149 
150        //在传感器管理器中注册监听器
151        sm.registerListener(acceleromererListener, acceleromererSensor, SensorManager.SENSOR_DELAY_NORMAL);
152 
153    }
154 
155    // 开始录音
156    public void startRecord() {
157        //把正在录音的标志设为真
158        isRecoding = true;
159        //存放文件
160        File file = new File("/sdcard/" "YY"
161                new DateFormat().format("yyyyMMdd_hhmmss", Calendar.getInstance(Locale.CHINA)) + ".amr");
162 
163        Toast.makeText(getApplicationContext(), "正在录音,录音文件在" + file.getAbsolutePath(), Toast.LENGTH_LONG).show();
164 
165        // 创建录音对象
166        mr = new MediaRecorder();
167 
168        // 从麦克风源进行录音
169        mr.setAudioSource(MediaRecorder.AudioSource.DEFAULT);
170 
171        // 设置输出格式
172        mr.setOutputFormat(MediaRecorder.OutputFormat.DEFAULT);
173 
174        // 设置编码格式
175        mr.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT);
176 
177        // 设置输出文件
178        mr.setOutputFile(file.getAbsolutePath());
179 
180        try {
181            // 创建文件
182            file.createNewFile();
183            // 准备录制
184            mr.prepare();
185        catch (IllegalStateException e) {
186            e.printStackTrace();
187        catch (IOException e) {
188            e.printStackTrace();
189        }
190        // 开始录制
191        mr.start();
192        recordButton.setText("录音中……");
193    }
194 
195    //摇动初始化
196    public void initShake() {
197        lastTime = 0;
198        duration = 0;
199        curTime = 0;
200        initTime = 0;
201        last_x = 0.0f;
202        last_y = 0.0f;
203        last_z = 0.0f;
204        shake = 0.0f;
205        totalShake = 0.0f;
206    }
207}

3、连接真机Milestone,编译并运行程序:

晃动机器,开始录音

查看录音文件,效果还可以:

4、我们小结一下:

到Android2.2版本为止,系统并没有给开发者提供多少可用的包装好的传感器信息,只是提供了传感器发出的原始数据,这些原始数据存放在  event.values 的数组里,开发人员需要从这些裸数据总自行发掘有用的信息,譬如从加速度传感器的3维裸数据中获得摇动的判断(我的摇动判断很弱智,有时间再改吧……)。

好了本讲就先到这里,关于传感器有机会我们展开再谈,下次再见吧。

转载于:https://www.cnblogs.com/java-time/archive/2011/05/25/tt150.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/261415.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python引入redis_使用python向Redis批量导入数据

1.使用pipeline进行批量导入数据class Redis_Handler(Handler):def connect(self):#print self.host,self.port,self.tableself.conn Connection(self.host,self.port,self.table)def execute(self, action_name):filename "/tmp/temp.txt"batch_size 10000with o…

使用R语言绘制中国地图

R语言环境 R3.1.1 Windows8.1 需要安装的packages: maptools,gp 绘图所需要的数据 中国地图的GIS数据 &#xff08;可以此下面的网址下载) http://cos.name/wp-content/uploads/2009/07/chinaprovinceborderdata_tar_gz.zip 是一个压缩包&#xff0c;完全解压后包含三个文件&am…

ASP.NET MVC 5 学习教程:控制器传递数据给视图

起飞网 ASP.NET MVC 5 学习教程目录&#xff1a; 添加控制器添加视图修改视图和布局页控制器传递数据给视图添加模型创建连接字符串通过控制器访问模型的数据生成的代码详解使用 SQL Server LocalDBEdit方法和Edit视图详解添加查询Entity Framework 数据迁移之添加字段添加验证…

如何将ListT转换相应的Html(xsl动态转换)(二)

一、前言 紧跟着上一篇随笔&#xff0c;本文主要涉及到如何将xml与xsl动态转换成html&#xff0c;这个才是最关键的地方&#xff0c;所有的内容都是围绕这个主题来进行开展的。根据指定的xsl样式将List<T>转换相应的Html&#xff0c;相关的随笔如下&#xff1a; &#xf…

echarts折线图背景线_echarts设置折线线条颜色和折线点颜色的实例

本文主要介绍了jQuery插件echarts设置折线图中折线线条颜色和折线点颜色的方法,结合实例形式分析了jQuery图表插件echarts设置折线图的相关操作技巧,需要的朋友可以参考下&#xff0c;希望能帮助到大家。1、问题背景设计一条折线图&#xff0c;但是图形中不用插件自带的颜色&am…

saxbuilder用法(转)

xml为我们在网络交换数据带来很大方便,在java中可以使用saxbuilder来操作xml格式文件,下面介绍一下saxbuilder的常用方法.import org.jdom.Document;import org.jdom.Element;import org.jdom.JDOMException;import org.jdom.input.SAXBuilder;import org.jdom.output.XMLOutpu…

页面添加复制代码功能

简单的"复制代码" 功能<a href"javascript:" οnclick"CopyCode(document.getElementById(code1));">复制代码</a> <div id"code1"> 点击复制 这里放代码这里放代码这里放代码这里放代码这里放代码这里放代码这里放…

Linux CentOS 5.5 服务器安装图文教程

下面开始&#xff1a; 系统版本&#xff1a;CentOS 5.5 将镜像刻成光盘&#xff0c;设置BIOS将CDROM设置为第一启动 启动画面&#xff1a; 通过提示&#xff0c;按ENTER进入图形安装模式&#xff08;E文不好的&#xff0c;赶紧补习去哈~~~&#xff09; 我们按ENTER键 信息检测&…

win10大文件无法移动到U盘

想着把自己的一些文件整理整理&#xff0c;发现稍微大点的文件竟然不能copy到U盘&#xff0c;百度了一番&#xff08;不好意思&#xff0c;最近百度犯错查的正狠^_^&#xff09;,发现因为U盘的格式不是NTFS,只要把U盘格式改为NTFS就可以copy大文件了。 方法一&#xff1a;用命令…

为多孔介质的当量直径_新型纳米多孔碳材料在催化剂载体方面的应用

成分&#xff1a;基本成分为碳91-95%&#xff0c;氧4-8%&#xff0c;氢1%&#xff0c;具有良好的导电性&#xff08;2-10 S/cm&#xff09;和优良的化学稳定性。 结构&#xff1a;内部是三维贯通的纳米孔道结构&#xff0c;每个孔与周围12个孔相连&#xff0c;孔道结构有序&…

Shell 变量的操作方法

在Shell中&#xff0c;经常会用到:和:-这样的操作符。 这两个操作符功能很相似&#xff0c;下面简单比较下: #color"red" echo "color is ${color}" //这样应该会打印color is (空白)稍微改造下: echo "color is ${color:-"no color"}"…

maven-settings.xml的那些事

1、代理<proxies> <proxies><proxy><id>optional</id>//代理名<active>true</active>//true激活该代理<protocol>http</protocol>//代理协议<username>proxyuser</username>//帐号密码<password>prox…

替换空格

题目描述&#xff1a; 请实现一个函数&#xff0c;将一个字符串中的空格替换成“%20”。例如&#xff0c;当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy。 时间复杂度o(n)的算法 先遍历一次字符串统计出空格的总数&#xff0c;计算出替换之后的字符串的总长…

备份数据 宝塔linux_宝塔面板旧版本升级教程汇总—升级宝塔面板

往期教程&#xff1a;宝塔面板教程&#xff08;1&#xff09;基于云服务器搭建宝塔面板教程最全详解宝塔面板教程&#xff08;2&#xff09;宝塔面板添加WordPress站点详细图文教程宝塔面板教程&#xff08;3&#xff09;基于宝塔面板成功配置网站SSL安全证书宝塔面板教程&…

执行全文索引时出现权限不足的解决方法

&#xfeff;在Oracle中&#xff0c;SYS用户是整个数据库的所有者&#xff0c;SYS拥有数据库中几乎所有的权限。今天我们就来介绍一下使用SYS执行全文索引的建立时&#xff0c;出现了权限不足的错误。   许多人刚刚接触Oracle时&#xff0c;对于权限并不是很了解&#xff0c;…

Swift面向对象基础(上)——Swift中的类和结构体(下)

学习来自《极客学院》1 import Foundation2 3 class User {4 var name:String5 var age:Int6 init(name:String,age:Int){7 self.age age8 self.name name9 } 10 } 11 var u1 User(name:"何杨",age:23) 12 var u2 User(name:&qu…

ultraedit26 运行的是试用模式_如何并行运行你的自动化测试?

点击上方蓝色字关注我们~在自动化测试过程中的某些时刻&#xff0c;你将遇到一个共同的问题&#xff0c;那就是&#xff1a;需要花费很长时间去执行大量的UI自动化测试用例。最为有效地加速用例执行的方式就是并行地运行自动化测试。不幸地是&#xff0c;使各个自动化测试用例并…

js实现IE/Firefox的收藏功能

代码&#xff1a; function addFavorite(el){var url window.location.href,title $(head).find(title).html();try{//IEwindow.external.AddFavorite(url, title);}catch(e){ //Firefoxtry{window.sidebar.addPanel(title,url,);}catch(e){if(window.opera){ //operael.hre…

最小点割集

prim算法不仅仅可以求最小生成树&#xff0c;也可以求“最大生成树”。最小割集Stoer-Wagner算法就是典型的应用实例。 求解最小割集普遍采用Stoer-Wagner算法&#xff0c;不提供此算法证明和代码&#xff0c;只提供算法思路&#xff1a; 1.minMAXINT&#xff0c;固定一个顶点P…

问题:从键盘读取特定类型的数据(使用Scanner读取int类型)

import java.util.Scanner; public class ScannerIntTest{public static void main(String [] args){int num1,num2;num1 getIntNumber();//调用函数getIntNumber()来获取一个从键盘输入的整数num2 getIntNumber();System.out.println("first number "num1"-…