内射老阿姨1区2区3区4区_久久精品人人做人人爽电影蜜月_久久国产精品亚洲77777_99精品又大又爽又粗少妇毛片

android按鍵精靈。C++編寫。-創(chuàng)新互聯(lián)

可以記錄屏幕鍵盤等傳感器對系統(tǒng)的輸入。android按鍵精靈。C++編
寫。

上一篇文章做的那個稍微有點復雜了,還需要把板子的輸出拿回電腦處理再倒回去。這個就簡單多了用法如下

建網站原本是網站策劃師、網絡程序員、網頁設計師等,應用各種網絡程序開發(fā)技術和網頁設計技術配合操作的協(xié)同工作。創(chuàng)新互聯(lián)專業(yè)提供成都網站設計、做網站、成都外貿網站建設公司,網頁設計,網站制作(企業(yè)站、響應式網站建設、電商門戶網站)等服務,從網站深度策劃、搜索引擎友好度優(yōu)化到用戶體驗的提升,我們力求做到極致!

usage:
event record /path/file1
event replay /path/file1

給我女友寫的程序直接搬過來了,所以注釋有些冗余。

"stdafx.h"

#include <sys/types.h>
#include<sys/stat.h>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<stdint.h>
#include<fcntl.h>
#include<sys/ioctl.h>
#include<errno.h>
#include<asm/types.h>
#include<unistd.h>
#include<iostream>
#include<vector>
struct input_event {
    timeval time;
    __u16 type;
    __u16 code;
    __s32 value;
};
struct myinputevent:public input_event
//繼承了16字節(jié)的數(shù)據(jù)結構,添加了一個deviceID{
int deviceID;

};

main.cpp

#include "stdafx.h"void record(char* filename);
void replay(char* filename);
int main(int argc, char *argv[])
{
if(argc!=3 || (strcmp(argv[1],"record") & strcmp(argv[1],"replay")))
    {
        std::cout<<

  "usage:"<<std::endl<<
  "event record /path/file1"<<std::endl<<
  "event replay /path/file1"<<std::endl;
return 0;
    }
if(!strcmp(argv[1],"record"))
        record(argv[2]);
else
        replay(argv[2]);
return 0;
}

replay.cpp

#include "stdafx.h"http://#define DEBUGint timedif(timeval& a,timeval& b)
//這是個計算時間差的函數(shù),對于一個timeval的數(shù)據(jù)結構,前面4字節(jié)秒,后面4字節(jié)微秒
//這個函數(shù)接受兩個這樣的結構,返回一個微秒單位的時間差{
return (b.tv_sec-a.tv_sec)*1000000+(b.tv_usec-a.tv_usec);
}


void replay(char* filename)
{

int fi = open(filename,O_RDONLY);
if(fi==-1)
    {
        std::cout<<filename<<" cannot be opened"<<std::endl;
        exit(0);
    }


char dev[40]="/dev/input/eventX";


int DeviceFile[10];
for(char i='0';i<='9';++i)
    {
        dev[16]=i;
int f=open(dev,O_RDWR);
//把所有的event都打開準備寫入        DeviceFile[i-'0']=f;
//把句柄保存下來,如果打不開的話f值是-1#ifdef DEBUG
if(f!=-1)
            std::cout<<"Device opened:"<<i<<std::endl;
#endif

    }
    myinputevent event0,event1;
    read(fi,&event0,sizeof(myinputevent));
//先讀取第一個event到event0中    timeval timetemp=event0.time;
//臨時存儲event中的時間,因為往設備寫的時候,時間要清空    memset(&event0.time, 0, sizeof(struct timeval));
//清空時間    write(DeviceFile[event0.deviceID], &event0.time, sizeof(input_event));
//寫回設備    event0.time=timetemp;
//把剛剛保存的時間拿回來  while(read(fi,&event1,sizeof(myinputevent)))
//只要文件還能讀,就讀取下一個event到event1中    {

int sleeptime=timedif(event0.time,event1.time);
//計算event0和event1的時間差#ifdef DEBUG
        std::cout<<"sleep"<<sleeptime<<std::endl;
#endif
        usleep(sleeptime);
//進行時間間隔        event0.time=event1.time;
//把event1的時間賦值給event0,這樣下次循環(huán)的時候繼續(xù)對比下一個時間差        memset(&event1.time, 0, sizeof(struct timeval));
//清空event1的時間        write(DeviceFile[event1.deviceID], &event1.time, sizeof(input_event));
//把event1寫回設備    }
for(int i=0;i<=9;++i)
    {
if(DeviceFile[i]!=-1)
            close(DeviceFile[i]);
//關閉所有設備
    }
    std::cout<<"Replay completed"<<std::endl;
}

record.cpp

#include "stdafx.h"#define DEBUG
inlinebool eventcmp(const myinputevent& a,const myinputevent& b)
//這個比較函數(shù)為了之后排序用,因為兩個自定義的數(shù)據(jù)結構,你用標準排序函數(shù),排序肯定不知道你這倆結構應該怎么比大小
//定義了我的數(shù)據(jù)結構之間如果做小于比較時,時間小就算小。
//這樣就可以采用標準排序函數(shù)對事件進行排序了。{
if(a.time.tv_sec<b.time.tv_sec)
return 1;
if(a.time.tv_sec>b.time.tv_sec)
return 0;
if(a.time.tv_usec<b.time.tv_usec)
return 1;
else   return 0;
}



std::vector<myinputevent> AllRecordedEvent;
//這里有個數(shù)組,用來添加所有錄制到的eventchar fn[200];
void afterstop(int x)
//按下ctrl+c和回車之后,錄制停止,做數(shù)據(jù)整理和保存工作
//主要是按時間順序對事件排序,然后保存{
#ifdef DEBUG
    std::cout<<"Recorded events:"<<AllRecordedEvent.size()<<std::endl;
#endif
    std::cout<<"Sorting & saving..."<<std::endl;
    std::stable_sort(AllRecordedEvent.begin(),AllRecordedEvent.end(),eventcmp);
    std::cout<<"Sort completed"<<std::endl;
//這是排序語句,對所有的錄制的event排序,排序時大小比較用eventcmp函數(shù),也就是根據(jù)時間先后排序啦
//因為有的時候,設備的輸出并不按照時間順序,比如你按開機鍵,它涉及的不止一個event,有時會有微微的時間上的錯序
//任何錯序都會嚴重影響replay,所以這里必須排序啦,這里一定要stable_sort,也就是排序必須是穩(wěn)定的
//對于不穩(wěn)定的排序,有時兩個事件的時間完全相等,可能會使原來在后面的跑前面去  int f=open(fn,O_WRONLY | O_CREAT | O_TRUNC);
if(f==-1)
    {
        std::cout<<fn<<" cannot be opened!";
        exit(0);
    }
//打開一開始event record file中那個file文件進行寫入  for(unsigned int i=0;i<AllRecordedEvent.size();++i)
    {
        write(f,&AllRecordedEvent[i],sizeof(myinputevent));
    }
    close(f);
    std::cout<<"Record completed. Filename:"<<fn<<std::endl;
    exit(0);
}
void record(char* filename)
{
    std::cout<<"If using windows adb, you have to "busybox stty intr ^X" to change the INTR from Ctrl-C to Ctrl-X"<<std::endl;
    std::cout<<"And the stop recording command will be ctrl-X."<<std::endl
<<"If using linux adb, you are fine."<<std::endl;
//注意如果用windows中的adb,按下ctrl+c之后windows的shell也會收到命令從而終止adb shell,這樣達不到終止錄制的功能
//所以如果非要在windows中使用adb,就需要用這個命令busybox stty intr ^X,把板子的終止組合鍵注冊到ctrl-X上去
//這樣按ctrl-X再回車,就可以終止錄制    strcpy(fn,filename);
int f=open(fn,O_WRONLY | O_CREAT);
//先嘗試打開一下參數(shù)中的文件,若失敗就不繼續(xù)了  if(f==-1)
    {
        std::cout<<fn<<" cannot be opened!";
        exit(0);
    }
    close(f);
struct sigaction act;
    act.sa_handler=afterstop;
    sigemptyset(&act.sa_mask);
    act.sa_flags= 0;
    sigaction(SIGINT,&act, NULL);
//這里注冊了ctrl+c事件,這樣錄制過程中按下ctrl+c然后回車,可以執(zhí)行afterstop函數(shù)。  char dev[40]="/dev/input/eventX";

int DeviceFile[10];
for(char i='0';i<='9';++i)
    {
        dev[16]=i;
int f=open(dev,O_RDONLY | O_NONBLOCK);
//把所有的event都打開,這樣保證能包括鍵盤和屏幕
//這里O_RDONLY是只讀,O_NONBLOCK是非阻塞。非阻塞的意思就是讀取的時候,如果設備里沒有,就返回-1。
//如果是阻塞的話,設備里沒有,read函數(shù)是不返回的,要等到有了才返回。
//所以阻塞肯定不行,因為我有一堆設備要循環(huán)讀取的,如果第一個設備沒數(shù)據(jù)就會卡住,后面的設備都讀不到。        DeviceFile[i-'0']=f;
//把句柄保存下來,如果打不開的話f值是-1#ifdef DEBUG
if(f!=-1)
            std::cout<<"Device opened:"<<i<<std::endl;
#endif

    }
char buffer[2048];


    std::cout<<"Recording started, press Ctrl+C then press ENTER to stop recording"<<std::endl;

while(1)
    {
for(int i=0;i<10;++i)
        {
if(DeviceFile[i]==-1)
continue;
int ReadSize=read(DeviceFile[i],buffer,2048);
//不斷的循環(huán),把打開了的文件反復讀取,這里緩存區(qū)必須是16的倍數(shù),因為生成的數(shù)據(jù)是16字節(jié)的數(shù)據(jù)結構,這個基本結構不能破壞了
//16字節(jié)的數(shù)據(jù)結構中,前4字節(jié)是秒,然后4字節(jié)微秒,最后8字節(jié)是實際的設備發(fā)送的信息
//這里read會返回讀取到的字節(jié)數(shù)量,如果啥也沒讀到會返回-1 if(ReadSize!=-1)
            {
#ifdef DEBUG
                std::cout<<"device"<<i<<"read"<<ReadSize<<std::endl;
#endif  for(input_event *p=(input_event*)buffer;(char*)p<buffer+ReadSize;++p)
                {
                    myinputevent my;

                    my.code=p->code;
                    my.type=p->type;
                    my.value=p->value;
                    my.time=p->time;
                    my.deviceID=i;
                    AllRecordedEvent.push_back(my);
//這里可能難理解,buffer里存儲了大量16字節(jié)的event,所以讓指針p是數(shù)據(jù)結構指針,然后16字節(jié)16字節(jié)的挪動
//我有一個自己的數(shù)據(jù)結構叫myinputevent,比這個16字節(jié)來說,尾部添加了一個4字節(jié)的int,存儲設備的event號
//你可以在stdafx.h中看到這些數(shù)據(jù)結構
//把那16字節(jié)的原封不動的放到我的數(shù)據(jù)結構中,再把我自己的設備號賦值,就完成了這一個事件的錄制。
//然后就存儲到AllRecordedEvent里面去。                }
            }
        }
    }

}

新聞標題:android按鍵精靈。C++編寫。-創(chuàng)新互聯(lián)
轉載來源:http://m.rwnh.cn/article30/iiipo.html

成都網站建設公司_創(chuàng)新互聯(lián),為您提供服務器托管、App設計ChatGPT、響應式網站、域名注冊、網站收錄

廣告

聲明:本網站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯(lián)

商城網站建設
富平县| 贵港市| 喀什市| 阜城县| 赤城县| 新晃| 嘉荫县| 洪江市| 阿勒泰市| 九江市| 天峻县| 托克逊县| 长春市| 桃园县| 固安县| 云南省| 桂东县| 山阳县| 永清县| 浏阳市| 社旗县| 重庆市| 水城县| 丹棱县| 墨竹工卡县| 平乡县| 汨罗市| 得荣县| 昭苏县| 石棉县| 包头市| 郧西县| 宝清县| 翁牛特旗| 那坡县| 霍邱县| 乌海市| 白玉县| 酉阳| 长武县| 望谟县|