点击上方“罗晓胜”,马上关注,您的支持对我帮助很大
上期文章
/ 前言 /
何为Android特色开发,就是只有在移动设备上开发使用的。例如定位,传感器。微信的摇一摇就是借助的传感器实现的,还有游戏狂野飙车的传感器感应,各种健康计步器等,都是使用传感器开发的特色功能。
/ 正文 /
基于位置的服务简称LBS,主要的工作原理就是利用无线电通讯网络或GPS 等定位方式来确定出移动设备所在的位置
找到自己的位置
用GPS 定位,精准度比较高,但是非常耗电,室内无法接受卫星信号 使用网络定位,的精准度稍差,但耗电量比较少
android原生定位:国内有墙,无法使用 第三方SDK:高德或者百度,参考官方文档
手机中内置的传感器是一种微型的物理设备,它能够探测、感受到外界的信号,并按一定规律转换成我们所需要的信息。Android 手机通常都会支持多种类型的传感器,如光照传感右器、加速度传感器、地磁传感器、压力传感器、温度传感器等。
Android 中每个传感器的用法其实都比较类似,真的可以说是一通百通了。
首先第一步要获取到SensorManager 的实例,方法如下:
SensorManager senserManager = (SensorManager)
getSystemService(Context.SENSOR_SERVICE);
SensorManager是系统所有传感器的管理器,有了它的实例之后就可以调用getDefaultSensor()方法来得到任意的传感器类型了,如下所示:
Sensor sensor = senserManager.getDefaultSensor(Sensor.TYPE_LIGHT);
这里使用Sensor.TYPE_LIGHT 常量来指定传感器类型,此时的Sensor 实例就代表着一个光照传感器。Sensor 中还有很多其他传感器类型的常量,。
接下来我们需要对传感器输出的信号进行监听,这就要借助SensorEventListener 来实现了。SensorEventListener 是一个接口,其中定义了onSensorChanged()和onAccuracyChanged() 这两个方法,如下所示:
SensorEventListener listener = new SensorEventListener() {
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
}
@Override
public void onSensorChanged(SensorEvent event) {
}
};
当传感器的精度发生变化时就会调用onAccuracyChanged()方法, 当传感器监测到的数值发生变化时就会调用onSensorChanged()方法。可以看到onSensorChanged()方法中传入了一个SensorEvent 参数,这个参数里又包含了一个values 数组,所有传感器输出的信息都是存放在这里的。
下面我们还需要调用SensorManager 的registerListener()方法来注册SensorEventListener才能使其生效, registerListener()方法接收三个参数, 第一个参数就是SensorEventListener 的实例, 第二个参数是Sensor 的实例,这两个参数我们在前面都已经成功得到了。第三个参数是用于表示传感器输出信息的更新速率,共有SENSOR_DELAY_UI、SENSOR_DELAY_NORMAL、SENSOR_DELAY_GAME 和SENSOR_DELAY_FASTEST 这四种值可选,它们的更新速率是依次递增的。因此,注册一个SensorEventListener 就可以写成:
senserManager.registerListener(listener, senser, SensorManager.SENSOR_DELAY_NORMAL);
另外始终要记得,当程序退出或传感器使用完毕时,一定要调用unregisterListener ()方法将使用的资源释放掉,如下所示:
sensorManager.unregisterListener(listener);
Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_LIGHT);
Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
public class MainActivity extends Activity {
private SensorManager sensorManager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
sensorManager = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
sensorManager.registerListener(listener, sensor, SensorManager.SENSOR_DELAY_NORMAL);
}
@Override
protected void onDestroy() {
super.onDestroy();
if (sensorManager != null) {
sensorManager.unregisterListener(listener);
}
}
private SensorEventListener listener = new SensorEventListener() {
@Override
public void onSensorChanged(SensorEvent event) {
// 加速度可能会是负值,所以要取它们的绝对值
float xValue = Math.abs(event.values[0]);
float yValue = Math.abs(event.values[1]);
float zValue = Math.abs(event.values[2]);
if (xValue > 15 || yValue > 15 || zValue > 15) {
// 认为用户摇动了手机,触发摇一摇逻辑
Toast.makeText(MainActivity.this, "摇一摇",
Toast.LENGTH_SHORT).show();
}
}
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
}
};
}
Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ORIENTATION);
之后在onSensorChanged()方法中通过SensorEvent 的values 数组,就可以得到传感器输 出的所有值了。方向传感器会记录手机在所有方向上的旋转角度,如图12.3 所示。其中,values[0]记录着手机围绕Z 轴的旋转角度,values[1] 记录着手机围绕X 轴的旋转角度,values[2] 记录着手机围绕Y 轴的旋转角度。
看起来很美好是吗?但遗憾的是,Android 早就废弃了Sensor.TYPE_ORIENTATION 这种传感器类型,虽然代码还是有效的,但已经不再推荐这么写了。事实上,Android 获取手机旋转的方向和角度是通过加速度传感器和地磁传感器共同计算得出的,这也是Android 目前推荐使用的方式。
首先我们需要分别获取到加速度传感器和地磁传感器的实例,并给它们注册监听器,如下所示:
Sensor accelerometerSensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
Sensor magneticSensor = sensorManager.getDefaultSensor(Sensor.TYPE_MAGNETIC_FIELD);
sensorManager.registerListener(listener, accelerometerSensor,SensorManager.SENSOR_DELAY_GAME);
sensorManager.registerListener(listener, magneticSensor,SensorManager.SENSOR_DELAY_GAME);
由于方向传感器的精确度要求通常都比较高,这里我们把传感器输出信息的更新速率提高了一些,使用的是SENSOR_DELAY_GAME。
接下来在onSensorChanged()方法中可以获取到SensorEvent 的values 数组,
分别记录着加速度传感器和地磁传感器输出的值。然后将这两个值传入到SensorManager 的
getRotationMatrix()方法中就可以得到一个包含旋转矩阵的R 数组,如下所示:
SensorManager.getRotationMatrix(R, null, accelerometerValues, magneticValues);
第一个参数R 是一个长度为9 的float 数组,getRotationMatrix()方法计算出的旋转数据就会赋值到这个数组当中。
第二个参数是一个用于将地磁向量转换成重力坐标的旋转矩阵,通常指定为null 即可。
第三和第四个参数则分别就是加速度传感器和地磁传感器输出的values 值。
得到了R 数组之后,接着就可以调用SensorManager 的getOrientation()方法来计算手机的旋转数据了,如下所示:
SensorManager.getOrientation(R, values);
values 是一个长度为3 的float 数组,手机在各个方向上的旋转数据都会被存放到这个数组当中。
values[0]记录着手机围绕着图12.3 中Z 轴的旋转弧度,
values[1]记录着手机围绕X 轴的旋转弧度,
values[2]记录着手机围绕Y 轴的旋转弧度。
注意这里计算出的数据都是以弧度为单位的,因此如果你想将它们转换成角度还需要调用如下方法:
Math.toDegrees(values[0]);
public class MainActivity extends Activity {
private SensorManager sensorManager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
sensorManager = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
Sensor magneticSensor = sensorManager.getDefaultSensor(Sensor.TYPE_MAGNETIC_FIELD);
Sensor accelerometerSensor = sensorManager.getDefaultSensor(
Sensor.TYPE_ACCELEROMETER);
sensorManager.registerListener(listener, magneticSensor,
SensorManager.SENSOR_DELAY_GAME);
sensorManager.registerListener(listener, accelerometerSensor,
SensorManager.SENSOR_DELAY_GAME);
}
@Override
protected void onDestroy() {
super.onDestroy();
if (sensorManager != null) {
sensorManager.unregisterListener(listener);
}
}
private SensorEventListener listener = new SensorEventListener() {
float[] accelerometerValues = new float[3];
float[] magneticValues = new float[3];
@Override
public void onSensorChanged(SensorEvent event) {
// 判断当前是加速度传感器还是地磁传感器
if (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER) {
// 注意赋值时要调用clone()方法
accelerometerValues = event.values.clone();
} else if (event.sensor.getType() == Sensor.TYPE_MAGNETIC_FIELD) {
// 注意赋值时要调用clone()方法
magneticValues = event.values.clone();
}
float[] R = new float[9];
float[] values = new float[3];
SensorManager.getRotationMatrix(R, null, accelerometerValues,
magneticValues);
SensorManager.getOrientation(R, values);
Log.d("MainActivity", "value[0] is " + Math.toDegrees(values[0]));
}
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
}
};
}
public class MyApplication extends Application {
private static Context context;
@Override
public void onCreate() {
context = getApplicationContext();
}
public static Context getContext() {
return context;
}
}
在Intent 中添加一些附加数据,以达到传值的效果, 比如在FirstActivity 中添加如下代码:
Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
intent.putExtra("string_data", "hello");
intent.putExtra("int_data", 100);
startActivity(intent);
这里调用了Intent 的putExtra()方法来添加要传递的数据,之后在SecondActivity 中就可
以得到这些值了,代码如下所示:
getIntent().getStringExtra("string_data");
getIntent().getIntExtra("int_data", 0);
但是不知道你有没有发现,putExtra()方法中所支持的数据类型是有限的,虽然常用的一 些数据类型它都会支持,但是当你想去传递一些自定义对象的时候就会发现无从下手。
Serializable 方式
public class Person implements Serializable{
。。。
Person person = new Person();
person.setName("Tom");
person.setAge(20);
Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
intent.putExtra("person_data", person);
startActivity(intent);
获取这个对象也很简单,写法如下:
Person person = (Person) getIntent().getSerializableExtra("person_data");
Parcelable 方式
public class Person implements Parcelable {
private String name;
private int age;
……
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(name); // 写出name
dest.writeInt(age); // 写出age
}
public static final Parcelable.Creator<Person> CREATOR = new Parcelable.Creator<Person>() {
@Override
public Person createFromParcel(Parcel source) {
Person person = new Person();
person.name = source.readString(); // 读取name
person.age = source.readInt(); // 读取age
return person;
}
@Override
public Person[] newArray(int size) {
return new Person[size];
}
};
}
获取对象
Person person = (Person) getIntent().getParcelableExtra("person_data");
/ 总结 /
本文主要讲了传Android特色开发中位置、传感器等使用方式,不知道这些特色开发有没有酷炫到你,手机功能已经越来越丰富,将来还会有各种各样的设备借助Andorid这样的系统去实现,可能还会有更加酷炫的传感器应用。
往期推荐:
关注我的公众号,学习技术或投稿
长按上图,识别图中二维码即可关注
文章浏览阅读3.8k次,点赞9次,收藏28次。直接上一个工作中碰到的问题,另外一个系统开启多线程调用我这边的接口,然后我这边会开启多线程批量查询第三方接口并且返回给调用方。使用的是两三年前别人遗留下来的方法,放到线上后发现确实是可以正常取到结果,但是一旦调用,CPU占用就直接100%(部署环境是win server服务器)。因此查看了下相关的老代码并使用JProfiler查看发现是在某个while循环的时候有问题。具体项目代码就不贴了,类似于下面这段代码。while(flag) {//your code;}这里的flag._main函数使用while(1)循环cpu占用99
文章浏览阅读347次。idea shift f6 快捷键无效_idea shift +f6快捷键不生效
文章浏览阅读135次。Ecmacript 中没有DOM 和 BOM核心模块Node为JavaScript提供了很多服务器级别,这些API绝大多数都被包装到了一个具名和核心模块中了,例如文件操作的 fs 核心模块 ,http服务构建的http 模块 path 路径操作模块 os 操作系统信息模块// 用来获取机器信息的var os = require('os')// 用来操作路径的var path = require('path')// 获取当前机器的 CPU 信息console.log(os.cpus._node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是
文章浏览阅读10w+次,点赞435次,收藏3.4k次。SPSS 22 下载安装过程7.6 方差分析与回归分析的SPSS实现7.6.1 SPSS软件概述1 SPSS版本与安装2 SPSS界面3 SPSS特点4 SPSS数据7.6.2 SPSS与方差分析1 单因素方差分析2 双因素方差分析7.6.3 SPSS与回归分析SPSS回归分析过程牙膏价格问题的回归分析_化工数学模型数据回归软件
文章浏览阅读7.5k次。如何利用hutool工具包实现邮件发送功能呢?1、首先引入hutool依赖<dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>5.7.19</version></dependency>2、编写邮件发送工具类package com.pc.c..._hutool发送邮件
文章浏览阅读867次,点赞2次,收藏2次。docker安装elasticsearch,elasticsearch-head,kibana,ik分词器安装方式基本有两种,一种是pull的方式,一种是Dockerfile的方式,由于pull的方式pull下来后还需配置许多东西且不便于复用,个人比较喜欢使用Dockerfile的方式所有docker支持的镜像基本都在https://hub.docker.com/docker的官网上能找到合..._docker安装kibana连接elasticsearch并且elasticsearch有密码
文章浏览阅读1.3w次,点赞57次,收藏92次。整理 | 郑丽媛出品 | CSDN(ID:CSDNnews)近年来,随着机器学习的兴起,有一门编程语言逐渐变得火热——Python。得益于其针对机器学习提供了大量开源框架和第三方模块,内置..._beeware
文章浏览阅读7.9k次。//// ViewController.swift// Day_10_Timer//// Created by dongqiangfei on 2018/10/15.// Copyright 2018年 飞飞. All rights reserved.//import UIKitclass ViewController: UIViewController { ..._swift timer 暂停
文章浏览阅读986次,点赞2次,收藏2次。1.硬性等待让当前线程暂停执行,应用场景:代码执行速度太快了,但是UI元素没有立马加载出来,造成两者不同步,这时候就可以让代码等待一下,再去执行找元素的动作线程休眠,强制等待 Thread.sleep(long mills)package com.example.demo;import org.junit.jupiter.api.Test;import org.openqa.selenium.By;import org.openqa.selenium.firefox.Firefox.._元素三大等待
文章浏览阅读3k次,点赞4次,收藏14次。Java软件工程师职位分析_java岗位分析
文章浏览阅读2k次。Java:Unreachable code的解决方法_java unreachable code
文章浏览阅读1w次。1、html中设置标签data-*的值 标题 11111 222222、点击获取当前标签的data-url的值$('dd').on('click', function() { var urlVal = $(this).data('ur_如何根据data-*属性获取对应的标签对象