Android利用方向傳感器獲得手機(jī)的相對(duì)角度實(shí)例說(shuō)明
x軸的方向是沿著屏幕的水平方向從左向右,如果手機(jī)不是正方形的話,較短的邊需要水平放置,較長(zhǎng)的邊需要垂直放置。
Y軸的方向是從屏幕的左下角開(kāi)始沿著屏幕的的垂直方向指向屏幕的頂端。
將手機(jī)放在桌子上,z軸的方向是從手機(jī)指向天空。
2.方向傳感器
在方向傳感器中values變量的3個(gè)值都表示度數(shù),它們的含義如下:
values[0]:該值表示方位,也就是手機(jī)繞著Z軸旋轉(zhuǎn)的角度。0表示北(North);90表示東(East);180表示南(South);270表示西(West)。如果values[0]的值正好是這4個(gè)值,并且手機(jī)是水平放置,表示手機(jī)的正前方就是這4個(gè)方向??梢岳眠@個(gè)特性來(lái)實(shí)現(xiàn)電子羅盤,實(shí)例76將詳細(xì)介紹電子羅盤的實(shí)現(xiàn)過(guò)程。
values[1]:該值表示傾斜度,或手機(jī)翹起的程度。當(dāng)手機(jī)繞著X軸傾斜時(shí)該值發(fā)生變化。values[1]的取值范圍是-180≤values[1]≤180。
假設(shè)將手機(jī)屏幕朝上水平放在桌子上,這時(shí)如果桌子是完全水平的,values[1]的值應(yīng)該是0(由于很少有桌子是絕對(duì)水平的,因此,該值很可能不為0,但一般都是-5和5之間的某個(gè)值)。這時(shí)從手機(jī)頂部開(kāi)始抬起,直到將手機(jī)沿X軸旋轉(zhuǎn)180度(屏幕向下水平放在桌面上)。在這個(gè)旋轉(zhuǎn)過(guò)程中,values[1]會(huì)在0到-180之間變化,也就是說(shuō),從手機(jī)頂部抬起時(shí),values[1]的值會(huì)逐漸變小,直到等于-180。如果從手機(jī)底部開(kāi)始抬起,直到將手機(jī)沿X軸旋轉(zhuǎn)180度,這時(shí)values[1]會(huì)在0到180之間變化。也就是values[1]的值會(huì)逐漸增大,直到等于180??梢岳胿alues[1]和下面要介紹的values[2]來(lái)測(cè)量桌子等物體的傾斜度。
values[2]:表示手機(jī)沿著Y軸的滾動(dòng)角度。取值范圍是-90≤values[2]≤90。假設(shè)將手機(jī)屏幕朝上水平放在桌面上,這時(shí)如果桌面是平的,values[2]的值應(yīng)為0。將手機(jī)左側(cè)逐漸抬起時(shí),values[2]的值逐漸變小,直到手機(jī)垂直于桌面放置,這時(shí)values[2]的值是-90。將手機(jī)右側(cè)逐漸抬起時(shí),values[2]的值逐漸增大,直到手機(jī)垂直于桌面放置,這時(shí)values[2]的值是90。在垂直位置時(shí)繼續(xù)向右或向左滾動(dòng),values[2]的值會(huì)繼續(xù)在-90至90之間變化。
下面以一個(gè)實(shí)例說(shuō)明其應(yīng)用方法
package com.example.sensortest;
import java.util.List;
import android.app.Activity;
import android.hardware.Sensor;
import android.hardware.SensorEvent;
import android.hardware.SensorEventListener;
import android.hardware.SensorManager;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
public class SensorTest extends Activity implements SensorEventListener{
private SensorManager sensorManager = null;
private Sensor gyroSensor = null;
private TextView vX;
private TextView vY;
private TextView vZ;
private TextView v;
private Button button;
private static final float NS2S = 1.0f / 1000000000.0f;
private float timestamp;
private float[] angle = new float[3];
@SuppressWarnings("deprecation")
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_sensor_test);
vX = (TextView) findViewById(R.id.vx);
vY = (TextView)findViewById(R.id.vy);
vZ = (TextView)findViewById(R.id.vz);
v = (TextView)findViewById(R.id.v);
button = (Button)findViewById(R.id.button);
sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
gyroSensor = sensorManager
.getDefaultSensor(Sensor.TYPE_ORIENTATION);
vX.setText("!!!!!!");
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View arg0) {
// TODO Auto-generated method stub
//聲明可變字符串
StringBuffer sb = new StringBuffer();
//獲取手機(jī)全部的傳感器
List<Sensor> sensors = sensorManager.getSensorList(Sensor.TYPE_ALL);
//迭代輸出獲得上的傳感器
for (Sensor sensor : sensors) {
//System.out.println(sensor.getName().toString());
sb.append(sensor.getName().toString());
sb.append("\n");
Log.i("Sensor", sensor.getName().toString());
}
//給文本控件賦值
v.setText(sb.toString());
}
});
}
public SensorTest() {
// TODO Auto-generated constructor stub
angle[0] = 0;
angle[1] = 0;
angle[2] = 0;
timestamp = 0;
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.activity_sensor_test, menu);
return true;
}
@Override
protected void onPause() {
// TODO Auto-generated method stub
super.onPause();
sensorManager.unregisterListener(this); // 解除監(jiān)聽(tīng)器注冊(cè)
}
@Override
protected void onResume() {
// TODO Auto-generated method stub
super.onResume();
sensorManager.registerListener(this, gyroSensor,
SensorManager.SENSOR_DELAY_NORMAL); //為傳感器注冊(cè)監(jiān)聽(tīng)器
}
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
// TODO Auto-generated method stub
}
@Override
public void onSensorChanged(SensorEvent event) {
// TODO Auto-generated method stub
// if (event.accuracy == SensorManager.SENSOR_STATUS_UNRELIABLE)
// {
// return;
// }
// if (timestamp != 0) {
// final float dT = (event.timestamp - timestamp) * NS2S;
// angle[0] += event.values[0] * dT * 100;
// angle[1] += event.values[1] * dT * 100;
// angle[2] += event.values[2] * dT * 100;
// }
// timestamp = event.timestamp;
//
//
// vX.setText("X: " + Float.toString(angle[0]));
// vY.setText("Y: " + Float.toString(angle[1]));
// vZ.setText("Z: " + Float.toString(angle[2]));
// 方向傳感器提供三個(gè)數(shù)據(jù),分別為azimuth、pitch和roll。
//
// azimuth:方位,返回水平時(shí)磁北極和Y軸的夾角,范圍為0°至360°。
// 0°=北,90°=東,180°=南,270°=西。
//
// pitch:x軸和水平面的夾角,范圍為-180°至180°。
// 當(dāng)z軸向y軸轉(zhuǎn)動(dòng)時(shí),角度為正值。
//
// roll:y軸和水平面的夾角,由于歷史原因,范圍為-90°至90°。
// 當(dāng)x軸向z軸移動(dòng)時(shí),角度為正值。
vX.setText("Orientation X: " + event.values[0]);
vY.setText("Orientation Y: " + event.values[1]);
vZ.setText("Orientation Z: " + event.values[2]);
}
}
布局文件如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".SensorTest"
android:orientation="vertical"
>
<Button
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="獲取傳感器"
/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/v"
android:textSize="30px"
></TextView>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/vx"
android:textSize="50px"
/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/vy"
android:textSize="50px"
/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/vz"
android:textSize="50px"
/>
</LinearLayout>
- android 傳感器(OnSensorChanged)使用介紹
- Android 利用方向傳感器實(shí)現(xiàn)指南針具體步驟
- Android利用Sensor(傳感器)實(shí)現(xiàn)指南針小功能
- Android編程實(shí)現(xiàn)獲取所有傳感器數(shù)據(jù)的方法
- Android開(kāi)發(fā)中的重力傳感器用法實(shí)例詳解
- Android利用Sensor(傳感器)實(shí)現(xiàn)水平儀功能
- Android實(shí)現(xiàn)電子羅盤(指南針)方向傳感器的應(yīng)用
- Android開(kāi)發(fā)中方向傳感器定義與用法詳解【附指南針實(shí)現(xiàn)方法】
- Android實(shí)現(xiàn)計(jì)步傳感器功能
- Android傳感器的簡(jiǎn)單使用方法
相關(guān)文章
AndroidStudio利用android-support-multidex解決64k的各種異常
這篇文章主要為大家詳細(xì)介紹了AndroidStudio利用android-support-multidex解決64k的各種異常,感興趣的小伙伴們可以參考一下2016-09-09Android中自定義View的實(shí)現(xiàn)方式總結(jié)大全
這篇文章主要總結(jié)了Android中自定義View的實(shí)現(xiàn)方式的相關(guān)資料,文中介紹的非常詳細(xì),對(duì)各位Android開(kāi)發(fā)者們學(xué)習(xí)或者使用自定義View具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-04-04Android 個(gè)人理財(cái)工具六:顯示賬單明細(xì) 下
本文主要節(jié)誒是Android 個(gè)人理財(cái)工具顯示賬單明細(xì),主要實(shí)現(xiàn)此窗口的查詢和刪除功能,這里提供實(shí)現(xiàn)代碼,有興趣的小伙伴可以參考下2016-08-08Android Activity Results API代替onActivityResul
說(shuō)到onActivityResult,我們已經(jīng)非常熟悉來(lái),通過(guò)在A activity啟動(dòng)B activity并且傳入數(shù)據(jù)到B中,然后在A中通過(guò)onActivityResult來(lái)接收B中返回的數(shù)據(jù)。在最新的activity-ktx的beta版本中,谷歌已經(jīng)廢棄了onActivityResult2022-09-09詳解androidstudio項(xiàng)目上傳到github方法以及步驟
在使用studio開(kāi)發(fā)的項(xiàng)目過(guò)程中有時(shí)候我們想將項(xiàng)目發(fā)布到github上,studio其實(shí)是自帶這種功能的,那么如何使用呢,下面我們就一起來(lái)了解一下2019-01-01Android 實(shí)現(xiàn)九宮格抽獎(jiǎng)功能
這篇文章主要介紹了Android 實(shí)現(xiàn)九宮格抽獎(jiǎng)功能,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-03-03android popuwindow點(diǎn)擊外部窗口不消失的實(shí)例
下面小編就為大家?guī)?lái)一篇android popuwindow點(diǎn)擊外部窗口不消失的實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04Android編程中FileOutputStream與openFileOutput()的區(qū)別分析
這篇文章主要介紹了Android編程中FileOutputStream與openFileOutput()的區(qū)別,結(jié)合實(shí)例形式分析了FileOutputStream與openFileOutput()的功能,使用技巧與用法區(qū)別,需要的朋友可以參考下2016-02-02