Android入門(13)| Android權限 與 內容提供器

文章目錄

  • 普通權限與危險權限
    • 運行時申請權限
  • 內容提供器
    • 運用安卓封裝好的內容提供器
    • 自實現的內容提供器
      • 概念
      • 實現


普通權限與危險權限

主要用于不同應用程序之間在保證被訪數據的安全性的基礎上,實現數據共享的功能

在 Android 6.0 開始引入了運行時權限的功能,用戶在安裝軟件時不需要一次性授權所有的權限,而是在軟件的使用過程中再對某一項權限進行申請。Android 將權限分為兩類:

  • 普通權限: 不會直接影響到用戶的安全和隱私的權限,對于這部分權限,系統自動授權。
  • 危險權限: 可能會涉及到用戶的隱私或者對設備安全性造成影響的權限。

危險權限如下,這些權限需要進行運行時權限處理,不在表中的權限只需要在 AndroidManifest.xml 添加權限聲明即可:
在這里插入圖片描述
表中的每一個危險權限都屬于一個權限組,雖然在進行權限處理的時候使用的是權限名,但是一旦用戶同意授權,那么該權限名對應的權限組中的所有權限也會同時被授權

運行時申請權限

給按鈕注冊點擊事件:

        Button button1 = findViewById(R.id.button_1);button1.setOnClickListener((View view)->{try {/*// 打開撥號界面,無需聲明權限Intent intent = new Intent(Intent.ACTION_DIAL);*/// 打電話,需要生命權限Intent intent = new Intent(Intent.ACTION_CALL);intent.setData(Uri.parse("tel:15309276440"));startActivity(intent);} catch (SecurityException e){e.printStackTrace();}});

在注冊表中加入:
在這里插入圖片描述
這樣的程序在 Android 6.0 之前都可以正常運行,但是在更高的版本點擊按鈕后沒有任何效果,錯誤信息如下:
在這里插入圖片描述
權限被禁止。

修復這個問題,申請運行時權限的流程:

將打電話的行為封裝成函數 call()

    private void call(){try {/*// 打開撥號界面,無需聲明權限Intent intent = new Intent(Intent.ACTION_DIAL);*/// 打電話,需要聲明權限Intent intent = new Intent(Intent.ACTION_CALL);intent.setData(Uri.parse("tel:15309276440"));startActivity(intent);} catch (SecurityException e){e.printStackTrace();}}

修改 onCreate 方法內的點擊按鈕行為:

        Button button1 = findViewById(R.id.button_1);button1.setOnClickListener((View view)->{// 相等說明用戶已授權,不等說明未授權if(ContextCompat.checkSelfPermission(this, Manifest.permission.CALL_PHONE)!= PackageManager.PERMISSION_GRANTED){// 申請授權ActivityCompat.requestPermissions(this,new String[] { Manifest.permission.CALL_PHONE}, 1);} else {call();}});
  • 通過 ContextCompat.checkSelfPermission() 方法檢測用戶是否已授權,該方法有兩個參數:
    1. context
    2. 具體權限名
  • 未授權則需要調用 ActivityCompat.requestPermissions() 方法來向用戶申請授權,該方法接受三個參數:
    1. Activity 實例,也就是當前活動。
    2. String 數組,也就是要申請的權限名。
    3. 請求碼,必須唯一,這里傳入 1。
  • 調用 requestPermissions 方法后,系統會彈出一個權限申請的對話框,用戶可以選擇同意或拒絕權限申請,不論同意與否,都會回調 onRequestPermissionsResult 方法,該方法有三個參數:
    1. 唯一的請求碼
    2. 存儲被申請權限名的 String 數組
    3. 授權結果 grantResults
    // 權限申請對話框點擊結果回調@Overridepublic void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {switch (requestCode) {case 1:if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {call();} else {Toast.makeText(this, "用戶拒絕授權", Toast.LENGTH_LONG).show();}break;default:}if(!ActivityCompat.shouldShowRequestPermissionRationale(this,Manifest.permission.CALL_PHONE)){AlertDialog.Builder dialog = new AlertDialog.Builder(this);dialog.setTitle("電話權限不可用").setMessage("請在-應用設置-權限中,允許APP使用電話權限。");dialog.setCancelable(false);dialog.setPositiveButton("立即設置", (dialog1, which) -> goToAppSetting());dialog.setNegativeButton("取消", (dialog2, which) -> dialog2.dismiss());dialog.show();}}

shouldShowRequestPermissionRationale 方法的返回值:

  • 應用第一次安裝,并且權限被禁用時,返回 true
  • 權限第一次被禁用時,返回 true
  • 權限被禁用且不再提示時,返回 false
  • 已授權時返回 false

總結:該方法返回值表示需不需要向用戶解釋一下你的 app 為什么需要這個權限。當用戶已經授權或者用戶明確禁止(權限被禁用且不再提示)的時候就不需要再去解釋了,所以此時會返回 false

權限不可用時引導用戶手動啟用權限:

	// 跳轉到權限設置界面private void goToAppSetting() {Intent intent = new Intent();intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);Uri uri = Uri.fromParts("package", getPackageName(), null);intent.setData(uri);startActivity(intent);}

上述代碼的運行邏輯是:

  • 通過 checkSelfPermission 檢驗用戶是否已授權:
    • 已授權則直接調用 call 打電話;
    • 未授權則通過 requestPermissions 申請授權:
      • 第一次申請授權被拒絕,點擊按鈕仍會二次調用 requestPermissions,此時 shouldShowRequestPermissionRationale 返回值為 true
      • 第二次申請授權被拒絕,權限被視為禁止使用,調用 requestPermissions 不會再彈出詢問彈窗,但是仍會回調 onRequestPermissionsResult,此時 shouldShowRequestPermissionRationale 返回值為 false,因此會彈出對話框詢問用戶是否要跳轉到設置界面開啟權限,用戶可以通過 “立即設置” 跳轉到 setting界面 來開放權限,此后再點擊按鈕會因為已授權而不再調用 requestPermissions

點擊按鈕的運行結果:
在這里插入圖片描述
點擊 DENY:
在這里插入圖片描述


內容提供器

內容提供器有兩種:已有的(如 Android 系統自帶的電話簿、短信等程序提供的供其他程序訪問部分內部數據外部訪問接口)、自實現的

ContentResolver類 是內容提供器的具體類,可以通過 Context類 中的 getContentResolver()方法 獲取該類的實例,該類提供了一系列的 CRUD 操作,這些增刪改查方法都使用 Uri參數 替代 表名參數內容URI 主要由三部分組成:

  • content: 協議聲明;
  • authority: 用于區分不同應用程序,一般采用程序包名命名;
  • path: 用區分同一程序中不同表。

舉個例子:在這里插入圖片描述

內容URI 只是一串字符,還需通過 Uri.parse() 方法解析成 Uri對象 才可做為參數。

關于內容提供器的增刪查改方法,這里僅解釋較為復雜的 query() 方法:
在這里插入圖片描述
查詢完后返回一個 Cursor對象,可以通過遍歷其所有行來得到每一行數據。


運用安卓封裝好的內容提供器

運用聯系人應用的內容提供器,讀取聯系人信息并在 ListView 中顯示。

聲明權限:
在這里插入圖片描述
布局文件 contacts_layout.xml:

<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"><ListViewandroid:id="@+id/contacts_view"android:layout_width="match_parent"android:layout_height="match_parent"/>
</LinearLayout>

活動文件 ContactsActivity:

public class ContactsActivity extends AppCompatActivity {ArrayAdapter<String> adapter;List<String> contactsList = new ArrayList<>();@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.contacts_layout);ListView contactsView = findViewById(R.id.contacts_view);adapter = new ArrayAdapter<>(this, android.R.layout.simple_list_item_1, contactsList);contactsView.setAdapter(adapter);if(ContextCompat.checkSelfPermission(this, Manifest.permission.READ_CONTACTS)!= PackageManager.PERMISSION_GRANTED){ActivityCompat.requestPermissions(this, new String[] {Manifest.permission.READ_CONTACTS}, 1);}else {readContacts();}}private  void readContacts() {Cursor cursor = null;try {Uri uri = ContactsContract.CommonDataKinds.Phone.CONTENT_URI;cursor = getContentResolver().query(uri, null, null,null, null, null);if(cursor != null){while(cursor.moveToNext()){// 獲取聯系人姓名String name = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));// 獲取聯系人手機號String number = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));contactsList.add(name + "\n" + number);}// 刷新ListViewadapter.notifyDataSetChanged();// 關閉 Cursor 對象cursor.close();}} catch (Exception e) {e.printStackTrace();} finally {// 和上面的關閉二選一/*if(cursor != null){cursor.close();}*/}}@Overridepublic void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,@NonNull int[] grantResults) {switch (requestCode){case 1:if(grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){readContacts();}else {Toast.makeText(this, "用戶拒絕授權", Toast.LENGTH_LONG).show();}break;}}
}

運行結果:
在這里插入圖片描述


自實現的內容提供器

概念

可以通過新建一個 ContentProvider子類 的方式來創建自己的內容提供器。ContentProvider類6抽象方法需要我們重寫:onCreate()query()insert()update()delete()getType()。這里重點介紹 onCreategetType 兩個方法:

  • onCreate():ContentProvider 嘗試訪問程序中數據時,初始化內容提供器,通常在這里完成對數據庫的創建和升級等操作。返回 true 表內容提供器初始化成功,false 表失敗。
  • getType(): 根據傳入的 內容URI 來返回相應的 MIME 類型。MIME字符串 主要由三部分組成:
    1. 必須要以 vnd 開頭
    2. 如果 內容URI路徑 結尾,則后接 android.cursor.dir/,如果以 id 結尾,則后接 android.cursor.item/
    3. 最后接上 vnd.<authority>.<path>

內容URI 的格式主要有兩種:

  • 路徑結尾表示期望訪問表中所有數據content://com.example.app.provider/table (訪問 table 表中所有數據)
  • id 結尾表示期望訪問表中擁有相應 id 的數據content://com.example.app.provider/table/1 (訪問 table 表中 id 為 1 的數據)

還可以使用通配符:

  • 匹配任意表:content://com.example.app.provider/*
  • 匹配 table 表中任意一行數據:content://com.example.app.provider/table/#

內容URI 對應的 MIME類型

  • content://com.example.app.provider/tablevnd.android.cursor.dir/vnd.com.example.app.provider.table
  • content://com.example.app.provider/table/1vnd.android.cursor.item/vnd.com.example.app.provider.table

如何匹配 內容URI 呢?

首先借助 UriMatcher.addURI() 方法,將 內容URI的相關信息 添加進匹配器中,相關信息對應方法的三個參數:authoritypath(int)code。前兩者之前講過這里不再贅述,code 用以唯一標識要訪問的資源

再借助 UriMatcher.match() 方法,傳入一個 Uri對象 ,通過返回的 code 來匹配對應的操作。

如何保證隱私數據不泄露?

因為所有的 CRUD操作 都需要匹配到相應的 內容URI 格式才能進行,只要不向 UriMatcher 中添加 隱私數據的URI 就好。


實現

那現在開始自實現內容提供器,操作的數據庫是該篇博客中的例子:

AndroidManifest.xml 文件中注冊:
在這里插入圖片描述
自定義的內容提供器 MyContentProvider

public class MyContentProvider extends ContentProvider {public static final int STUDENT_DIR = 0;public static final int STUDENT_ITEM = 1;public static final int CLASS_DIR = 2;public static final int CLASS_ITEM = 3;public static final String AUTHORITY = "com.example.activitytest.CustomType.provider";private static UriMatcher uriMatcher;private MyDatabaseHelper dbHelper;static {uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);uriMatcher.addURI(AUTHORITY, "student", STUDENT_DIR);uriMatcher.addURI(AUTHORITY, "student/#", STUDENT_ITEM);uriMatcher.addURI(AUTHORITY, "class", CLASS_DIR);uriMatcher.addURI(AUTHORITY, "class/#", CLASS_ITEM);}@Overridepublic int delete(Uri uri, String selection, String[] selectionArgs) {SQLiteDatabase db = dbHelper.getWritableDatabase();int deleteRows = 0;switch (uriMatcher.match(uri)){case STUDENT_DIR:deleteRows = db.delete("Student", selection, selectionArgs);break;case STUDENT_ITEM:String studentId = uri.getPathSegments().get(1);deleteRows = db.delete("Student", "id = ?", new String[]{studentId});break;case CLASS_DIR:deleteRows = db.delete("Class", selection, selectionArgs);break;case CLASS_ITEM:String classId = uri.getPathSegments().get(1);deleteRows = db.delete("Class","id = ?", new String[]{classId});break;}return deleteRows;}@Overridepublic String getType(Uri uri) {switch (uriMatcher.match(uri)){case STUDENT_DIR:return "vnd.android.cursor.dir/vnd.com.example.activitytest.CustomType.provider.student";case STUDENT_ITEM:return "vnd.android.cursor.item/vnd.com.example.activitytest.CustomType.provider.student";case CLASS_DIR:return "vnd.android.cursor.dir/vnd.com.example.activitytest.CustomType.provider.class";case CLASS_ITEM:return "vnd.android.cursor.item/vnd.com.example.activitytest.CustomType.provider.class";}return null;}@Overridepublic Uri insert(Uri uri, ContentValues values) {SQLiteDatabase db = dbHelper.getWritableDatabase();Uri uriReturn = null;switch (uriMatcher.match(uri)){case STUDENT_DIR:case STUDENT_ITEM:long studentId = db.insert("Student", null, values);uriReturn = Uri.parse("content://" + AUTHORITY + "/student/" + studentId);break;case CLASS_DIR:case CLASS_ITEM:long classId = db.insert("Class", null, values);uriReturn = Uri.parse("content://" + AUTHORITY + "/class/" + classId);break;default:break;}return uriReturn;}@Overridepublic boolean onCreate() {dbHelper = new MyDatabaseHelper(getContext(), "Student.db", null, 4);return true;}@Overridepublic Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,String sortOrder) {SQLiteDatabase db = dbHelper.getReadableDatabase();Cursor cursor = null;switch (uriMatcher.match(uri)){case STUDENT_DIR:cursor = db.query("Student", projection, selection, selectionArgs,null, null, sortOrder);break;case STUDENT_ITEM:// Uri字符串中以 “/” 作為分割,0部分是路徑,1部分則是id。即獲取Uri字符串中的id部分。String studentId = uri.getPathSegments().get(1);cursor = db.query("Student", projection, "id = ?", new String[]{ studentId }, null, null, sortOrder);break;case CLASS_DIR:cursor = db.query("Class", projection, selection, selectionArgs,null, null, sortOrder);break;case CLASS_ITEM:String classId = uri.getPathSegments().get(1);cursor = db.query("Class", projection, "id = ?", new String[]{ classId }, null, null, sortOrder);break;default:break;}return cursor;}@Overridepublic int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {SQLiteDatabase db = dbHelper.getWritableDatabase();int updateRows = 0;switch (uriMatcher.match(uri)){case STUDENT_DIR:updateRows = db.update("Student", values, selection, selectionArgs);break;case STUDENT_ITEM:String studentId = uri.getPathSegments().get(1);updateRows = db.update("Student", values, "id = ?", new String[]{studentId});break;case CLASS_DIR:updateRows = db.update("Class", values, selection, selectionArgs);break;case CLASS_ITEM:String classId = uri.getPathSegments().get(1);updateRows = db.update("Class", values, "id = ?", new String[]{classId});break;default:break;}return updateRows;}
}

onCreate()

  1. 初始化一個 MyDatabaseHelper 實例;
  2. 返回 true 表示內容提供器初始化成功。

query()

  1. 通過 MyDatabaseHelper 獲取 SQLiteDatabase 實例;
  2. 通過 uriMatcher.match(uri) 分析用戶想訪問的表;
  3. 通過 SQLiteDatabase.query() 進行查詢,并返回 Cursor 對象:
    • 訪問單條數據時,調用 uri.getPathSegments()內容URI 權限之后的部分以 “/” 作為分割,并將結果放入一個字符串列表,列表的第0個位置是路徑,第1個位置則是id

insert()

  • 前兩步同 query()
  1. 通過 SQLiteDatabase.insert() 進行添加,但由于該方法要求返回一個 Uri對象,因此需要調用 Uri.parse()URI字符串 解析成 Uri對象

接下來新建一個程序,用來調用上面的內容提供器:

public class MainActivity extends AppCompatActivity {private static final String TAG = "MainActivity";private String newId;public static final String AUTHORITY = "content://com.example.activitytest.CustomType.provider/";@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button button_add = findViewById(R.id.button_add);button_add.setOnClickListener(v->{Uri uri = Uri.parse(AUTHORITY + "student/");ContentValues values = new ContentValues();values.put("name", "zj");values.put("age", 21);values.put("weight", 90);values.put("gender", "girl");Uri insertUri = getContentResolver().insert(uri, values);newId = insertUri.getPathSegments().get(1);Log.e(TAG, "咕咕:"+insertUri.toString());});Button button_query = findViewById(R.id.button_query);button_query.setOnClickListener(v->{Uri uri = Uri.parse(AUTHORITY + "student");Cursor cursor = getContentResolver().query(uri, null, null,null, null);while(cursor.moveToNext()){String name = cursor.getString(cursor.getColumnIndex("name"));int age = cursor.getInt(cursor.getColumnIndex("age"));double weight = cursor.getDouble(cursor.getColumnIndex("weight"));String gender = cursor.getString(cursor.getColumnIndex("gender"));String res = name + " " + age + " " + weight + " " + gender;Toast.makeText(this, res, Toast.LENGTH_LONG).show();}cursor.close();Log.e(TAG, "表中數據顯示完畢");});Button button_update = findViewById(R.id.button_update);button_update.setOnClickListener(v->{Uri uri = Uri.parse(AUTHORITY + "student/" + newId);ContentValues values = new ContentValues();values.put("name", "cjl");values.put("weight", 95);getContentResolver().update(uri, values, null, null);});Button button_delete = findViewById(R.id.button_delete);button_delete.setOnClickListener(v->{if(newId!=null && newId.compareTo("0") > 0){Uri uri = Uri.parse(AUTHORITY + "student/" + newId);getContentResolver().delete(uri, null, null);newId = String.valueOf(Integer.valueOf(newId)-1);Log.e(TAG, "最后一個id:" + newId);}else{Toast.makeText(this, "表中已經沒有數據了", Toast.LENGTH_LONG).show();}});}
}

如果模擬器是 Android 11,那么該程序的清單文件需要加上 <queries> 標簽,原因見本博客:
在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/443699.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/443699.shtml
英文地址,請注明出處:http://en.pswp.cn/news/443699.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Java實現身份證號碼的驗證,JAVA后臺驗證身份證號碼

代碼如下&#xff1a; package cn.gov.csrc.util;/*** 18 位身份證驗證器* * author admin* */ public class IDCard {final int[] wi { 7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2, 1 };final int[] vi { 1, 0, X, 9, 8, 7, 6, 5, 4, 3, 2 };private int[] ai n…

Android入門(14)| 通知

文章目錄創建通知點擊效果其它小功能實例創建通知 創建通知的步驟&#xff1a; 管理通知的 NotificationManager&#xff0c;通常通過當前 Context 的 getSystemService() 獲取實例。它接受一個字符串參數用于確定獲取系統的什么服務。Android 8.0(O) 版本后需要通知通道&…

Android開發(3) | 權限和內容提供器的應用——調用相機和相冊

文章目錄拍照并保存到 ImageView 控件布局文件 notice_layout.xml按鈕 button_takePhoto 的點擊操作隱式 Intent 啟動后的回調AndroidManifest.xml從相冊選取照片并在 ImageView 控件中顯示布局文件 notice_layout.xml按鈕 button_takePhoto 的點擊操作自定義打開相冊的方法 op…

Android開發(4) | 系統權限、MediaPlayer類 和 VideoView類 的應用——播放多媒體文件

文章目錄MediaPlayer類播放音頻的實例VideoView類播放視頻的實例MediaPlayer類 對多種格式的音頻文件提供了全面的控制方法&#xff1a; 如何獲得MediaPlayer實例&#xff1f; 通過構造函數&#xff1a; MediaPlayer mp new MediaPlayer();調用 MediaPlayer.create() 方法&…

Android入門(15)| 網絡

文章目錄WebViewHTTP使用HttpURLConnection使用OkHttp封裝網絡操作封裝HttpURLConnection封裝OkHttpWebView WebView 可以在 應用程序中&#xff08;而不是瀏覽器&#xff09; 展示一些網頁。 布局文件 web_layout.xml&#xff1a; <LinearLayoutxmlns:android"http…

Java-單例模式

單例模式相信大家都不陌生&#xff0c;在JAVAEE應用中&#xff0c;單例模式是一種應用非常廣泛的設計模式&#xff0c;應用中許多組件都只需要單個實例&#xff0c;下面介紹單例模式。 使用單例模式的優點&#xff1a; 1.減少創建JAVA實例所帶來的系統開銷。 2.便于系統跟蹤單…

Android入門(16)| 服務

文章目錄概念Android 多線程繼承 Thread繼承 Runable 接口匿名類異步消息處理AsyncTask使用服務框架啟動/停止服務綁定/解綁服務服務的生命周期前臺服務IntentService完整版下載示例下載過程的回調接口&#xff1a;DownloadListener繼承 AsyncTask 實現下載功能&#xff1a;Dow…

2020德勤面試開始了嗎_2020國考面試開始,近期面試公告匯總,附結構化小組面試流程...

2020年國家公務員考試面試環節逐步恢復考試&#xff0c;各個招錄部門已經發布面試考察公告&#xff0c;對于進入面試環節的國考考生來說&#xff0c;有必要了解近期國考面試的招錄動態&#xff0c;提前做好面試準備。2020國考國家統計局機關面試面試確認&#xff1a;請進入面試…

項目積壓需求項目計劃_需求變更頻繁,項目經理如何做好需求管理?

項目實施過程中&#xff0c;項目經理常常面臨一個重大挑戰——需求變更。需求變更無處不在&#xff0c;市場條件變化、新業務出現、戰略目標調整、客戶需求修改、資源限制等&#xff0c;都會造成需求變更。需求變更會影響項目的時間、成本和質量&#xff0c;對整個項目和團隊成…

Android | Sensor.TYPE_ORIENTATION被廢棄后的解決辦法

文章目錄概述getOrientation 方法根據 旋轉矩陣R 獲取 設備旋轉弧度getRotationMatrix 方法根據 地磁場、加速度傳感器對象 獲取 旋轉矩陣R代碼參考資料概述 Sensor.TYPE_ORIENTATION 常數在 API 8 中已棄用&#xff0c;官方推薦使用 SensorManager.getOrientation() 替代。關…

【JAVA 開發小問題】 | String操作合集

文章目錄截取特定兩個字符之間的字符串截取特定兩個字符之間的字符串 利用正則表達式&#xff0c;圖片來源

uniapp 刷新后數據都沒有了_環境溫度傳感器都沒有連接,竟然還會有數據?

福田歐曼GTL(福康發動機、康明斯2880系統)匹配ECoffit尿素泵●故障現象&#xff1a;OBD故障燈點亮&#xff0c;不燒尿素&#xff0c;油耗高&#xff0c;動力不足●故障碼&#xff1a;●維修分析&#xff1a;①故障指出加熱器問題&#xff0c;摸下尿素箱溫度&#xff0c;發現燙手…

Android | 再探 RecyclerView 之名詞解析

文章目錄Adapter、ViewHolderchild viewLayoutManagerRecyclerScrapDirtyIndexPositionlayout position 和 adapter position四級緩存瀏覽本文前推薦先閱讀 Android入門&#xff08;九&#xff09;| 滾動控件 ListView 與 RecyclerView Adapter、ViewHolder Adapter: A subcla…

Linux學習:第一章-Linux簡介

一 UNIX發展史1 1965年&#xff0c;美國麻省理工學院&#xff08;MIT&#xff09;、通用電氣公司&#xff08;GE&#xff09;及AT&T的貝爾實驗室聯合開發Multics工程計劃&#xff0c;其目標是開發一種交互式的具有多道程序處理能力的分時操作系統&#xff0c;但因Multics追…

尼爾機器人技能快捷鍵_《尼爾機械紀元》連招操作技巧

《尼爾機械紀元》中的每個角色都可以裝備一個輕武器和一個重武器&#xff0c;技能招式也很豐富&#xff0c;下面為大家帶來了《尼爾機械紀元》連招操作技巧&#xff0c;希望對你們有所幫助。連擊技能展示視頻視頻原址&#xff1a;點擊進入基本沒有什么太難的連段&#xff0c;只…

保姆級教學!Xcode 配置 OpenGL 環境

文章目錄GLFW獲取 GLFWGLAD獲取 GLAD在 Xcode 中配置下載好的 GLFW 和 GLAD配置流程檢測是否配置成功無關配置的題外話——Xcode 下安全的刪除移動操作GLFW Graphics Library Framework&#xff08;圖形庫框架&#xff09;&#xff0c;可以讓我們通過其封裝好的 通用API 來正確…

Android入門(17)| 百度提供的 Android定位SDK

文章目錄配置百度提供的 Android定位SDK用于發布的 SHA1用于測試的 SHA1使用百度定位實例配置百度提供的 Android定位SDK 詳情參見官方文檔&#xff0c;這里僅對獲取 SHA1 做詳細介紹&#xff1a; 用于發布的 SHA1 用于測試的 SHA1 使用百度定位實例 public class LocationAc…

ios 不被遮擋 陰影_為何你沒見到日環食?你不知道的天象常識原來還有這么多 | 返樸...

關注風云之聲提升思維層次導讀說好的日環食呢&#xff0c;為什么上周很多人只等到了日偏食?日食月食的時間和種類是怎么預測的?你真的弄懂了各種日食和月食的成因嗎&#xff1f;你了解它們有什么區別和聯系&#xff0c;又遵循什么樣的時間規律嗎? 日食和月食發生的頻率一樣嗎…

初識貝塞爾(bezier)曲線

文章目錄資料援引貝塞爾曲線的用途一階貝塞爾&#xff08;bezier&#xff09;曲線二階貝塞爾&#xff08;bezier&#xff09;曲線三階貝塞爾&#xff08;bezier&#xff09;曲線高階貝塞爾&#xff08;bezier&#xff09;曲線三階貝塞爾曲線求插值&#xff08;Slerp&#xff09…

python代碼測試 vim_用 Hypothesis 快速測試你的 Python 代碼

點擊上方“Python編程時光”&#xff0c;選擇“加為星標”第一時間關注Python技術干貨&#xff01;介紹無論你使用哪種編程語言或框架&#xff0c;測試都非常重要。Hypothesis是 Python 的一個高級測試庫。它允許編寫測試用例時參數化&#xff0c;然后生成使測試失敗的簡單易懂…